mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-25 17:04:40 +02:00 
			
		
		
		
	Compare commits
	
		
			255 Commits
		
	
	
		
			draft/wond
			...
			export-com
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | f5c44e1f0b | ||
|   | 958ebeae61 | ||
|   | 31f51ca53b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5abbf73050 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7cf766630d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 59fccafeac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 19a6996e6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 86c4f90910 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4d88112021 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | de5c63e299 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | da0853a121 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 57cc4b6a29 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e2ddd7b27a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 693de8501e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c6df6d444e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ad5692c5f8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 60ab3337af | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dd847793d2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6d6e9f0dd3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0fe0d7ca6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | abcf1f1bae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7d0b03f754 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd8931d3d1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0d86c04939 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8436ebbf68 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 824f94b3df | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f9a8e686b2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e442a44ba8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f1fd29571a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6a482a7ba2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bfea434a55 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | acb22c5a64 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7cd1b621d1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c37b25bab | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a8bb9a78ef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e4aff04061 | ||
|   | c5cadc7de3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7dc0c0d96a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2c9c8c8e05 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b5d7ae57e5 | ||
|   | 0f66ee3a41 | ||
|   | 771372c66b | ||
|   | a7937681e9 | ||
|   | 792f238d16 | ||
|   | ba16416c75 | ||
|   | 6e0ac52a64 | ||
|   | 5bc40402a6 | ||
|   | df14c69977 | ||
|   | 1ea67ba93d | ||
|   | a7153d9d1d | ||
|   | e885057a71 | ||
|   | 7efa081976 | ||
|   | 5deb93a083 | ||
|   | e3908e6fe3 | ||
|   | fe3d0b5e8b | ||
|   | b6bb74d08d | ||
|   | c725f84334 | ||
|   | 11a3380d83 | ||
|   | 76a5bb060e | ||
|   | dac8dda4d4 | ||
|   | a1a62468a6 | ||
|   | ba3a723e99 | ||
|   | c5355c08cf | ||
|   | 6102380051 | ||
|   | 655e59a707 | ||
|   | d05745070b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 88c313bf86 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a7705848ec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 69e1bae8dd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d361757e4a | ||
|   | 0ef202f2df | ||
|   | bbfd2b3cd3 | ||
|   | 120c8f373c | ||
|   | 9135ebf2e2 | ||
|   | af31e9dcc2 | ||
|   | 50bc7e099a | ||
|   | 39d17c4a3c | ||
|   | d34c2a75db | ||
|   | de95c68d75 | ||
|   | cdf352d4c3 | ||
|   | 4712393b62 | ||
|   | fd48c2cf79 | ||
|   | 5feacd9a3b | ||
|   | ec35d5db51 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ddf088e428 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | adc1e585ff | ||
|   | 84b47a2ed5 | ||
|   | 6196fba286 | ||
|   | 5daff2d3cd | ||
|   | f1bc90e08a | ||
|   | aabcdc20fd | ||
|   | 269fbcc2f3 | ||
|   | d08179c215 | ||
|   | 90e739d444 | ||
|   | 4a9fac2d1e | ||
|   | 07ebd7c68c | ||
|   | 92f30f7ed6 | ||
|   | 605aa554d0 | ||
|   | bed9fca4a5 | ||
|   | b9968e2e72 | ||
|   | ab1a30073c | ||
|   | 31049d06e8 | ||
|   | ef8559d060 | ||
|   | 33bb23d2f3 | ||
|   | b27ac257e7 | ||
|   | d2cc76e52e | ||
|   | cad6097d60 | ||
|   | 2537b225ac | ||
|   | 4ee48d2729 | ||
|   | 68f23d652f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a078508c05 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | abf4dc9256 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ba8f12d588 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d57560db06 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0d26049b4e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f72e9b6ea5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 029cfb31b0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a288eb09c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 803909abb6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 56c75b769c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eea48d94d3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e29152ab30 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f4aa36b35d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2903a763a7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4a980ed5db | ||
|   | d2e687ed0a | ||
|   | 0d70690ec8 | ||
|   | a524eeb66e | ||
|   | 3d56ceb794 | ||
|   | 65c32b3319 | ||
|   | 9e8e047aae | ||
|   | 64d330a332 | ||
|   | 1ed1529f96 | ||
|   | b30066ca19 | ||
|   | aae8e2fa5d | ||
|   | 9e6d5fdbcb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 22b2e10ddb | ||
|   | d53ac2a61e | ||
|   | 6a0f800716 | ||
|   | aee1e2451e | ||
|   | da94eb1284 | ||
|   | ea51251fe6 | ||
|   | 399ce1e01a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7df8302ba2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | af8c59b5bb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cf0f00285b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b5c67a384c | ||
|   | af93cedc08 | ||
|   | b6a6f2d465 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6bcbf8b50a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 666516d7e9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b941c5b661 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8f8c85c64e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 116b0c48da | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aa2971e8c5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5656ac1e3e | ||
|   | e6a9ff1b96 | ||
|   | 832b88249c | ||
|   | 9902092fd1 | ||
|   | 8f0863d335 | ||
|   | 9423ac3263 | ||
|   | a66cfe2627 | ||
|   | 86cf28f2b4 | ||
|   | b5a46dd671 | ||
|   | cd942c3e3b | ||
|   | 55ccd5b79b | ||
|   | 4348c55c31 | ||
|   | a3fbe40b26 | ||
|   | 7431ca81d1 | ||
|   | 4d13dbf625 | ||
|   | 3840e2f4e6 | ||
|   | 52d10bb41e | ||
|   | 96c87f920a | ||
|   | 7d4189c624 | ||
|   | f3e17c90d3 | ||
|   | 70b3a9de49 | ||
|   | bf6d0eeef7 | ||
|   | 5359e4fec9 | ||
|   | 58fe639b8d | ||
|   | 327ed0e2d1 | ||
|   | c2fce6d8c4 | ||
|   | cb6b7559b4 | ||
|   | 77d789ed8e | ||
|   | 89471094ce | ||
|   | 670ceafc84 | ||
|   | 873afdacd3 | ||
|   | 880e4feede | ||
|   | 9ba7ca3845 | ||
|   | 734bb4d2ed | ||
|   | f2d2f97546 | ||
|   | 2fa69ddc32 | ||
|   | 1331cffe93 | ||
|   | f242721f3b | ||
|   | e940aeb1a3 | ||
|   | 580e719580 | ||
|   | 127af9db23 | ||
|   | 2209e2c1e8 | ||
|   | ed31980f84 | ||
|   | db28595302 | ||
|   | cded1cd63d | ||
|   | 8e447b4c32 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e29d3fc5e6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9da56e46f0 | ||
|   | 625ecc64ed | ||
|   | ceb43ed8fb | ||
|   | 8c0a0415de | ||
|   | 192debd829 | ||
|   | 1cfb4dfd8b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fb32886355 | ||
|   | 065df495ba | ||
|   | 558227f744 | ||
|   | 6d45430344 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3aa0c5ebc0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e940993e0e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8f90aeb8d5 | ||
|   | e92d133973 | ||
|   | b682d88167 | ||
|   | 7daf1a7944 | ||
|   | 5c0eff50a0 | ||
|   | 19056d635b | ||
|   | 4d5f00ff08 | ||
|   | 20de06ef50 | ||
|   | 1849ff6ee2 | ||
|   | 6765fc16be | ||
|   | 5ca4f5bbf4 | ||
|   | 9392ec276d | ||
|   | b26e4fcf99 | ||
|   | 45f3410da8 | ||
|   | 94b387ef7b | ||
|   | 6d0716eb6b | ||
|   | 8e26d5b500 | ||
|   | c5a7723185 | ||
|   | 49172ac2d3 | ||
|   | 618a846451 | ||
|   | d9f49ffd67 | ||
|   | 46e43baad1 | ||
|   | bd35b682fa | ||
|   | b6f9a8005e | ||
|   | 1acfaf6b6e | ||
|   | 5cf7087754 | ||
|   | b2d49155ef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9745461db7 | ||
|   | 21e9fcb2f5 | ||
|   | e203203993 | ||
|   | f224e4d596 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e0ca689759 | ||
|   | f792eb5ae7 | ||
|   | 4604c8d823 | ||
|   | 0896892f8a | ||
|   | 7fe225ee99 | ||
|   | d2fd7be457 | ||
|   | 5c61613a2e | ||
|   | b2767924de | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 59d0a77862 | ||
|   | 987526d1e5 | 
| @@ -4,5 +4,19 @@ 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_URL=https://libraries.excalidraw.com | ||||||
| REACT_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | REACT_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | ||||||
|  |  | ||||||
| REACT_APP_SOCKET_SERVER_URL=http://localhost:3002 | # 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"}' | 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"}' | ||||||
|  |  | ||||||
|  | # put these in your .env.local, or make sure you don't commit! | ||||||
|  | # must be lowercase `true` when turned on | ||||||
|  | # | ||||||
|  | # whether to enable Service Workers in development | ||||||
|  | REACT_APP_DEV_ENABLE_SW= | ||||||
|  | # whether to disable live reload / HMR. Usuaully what you want to do when | ||||||
|  | # debugging Service Workers. | ||||||
|  | REACT_APP_DEV_DISABLE_LIVE_RELOAD= | ||||||
|   | |||||||
| @@ -4,8 +4,14 @@ REACT_APP_BACKEND_V2_POST_URL=https://json.excalidraw.com/api/v2/post/ | |||||||
| REACT_APP_LIBRARY_URL=https://libraries.excalidraw.com | REACT_APP_LIBRARY_URL=https://libraries.excalidraw.com | ||||||
| REACT_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | REACT_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | ||||||
|  |  | ||||||
| REACT_APP_SOCKET_SERVER_URL=https://oss-collab-us1.excalidraw.com | 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"}' | 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 | # production-only vars | ||||||
| REACT_APP_GOOGLE_ANALYTICS_ID=UA-387204-13 | REACT_APP_GOOGLE_ANALYTICS_ID=UA-387204-13 | ||||||
|  |  | ||||||
|  | REACT_APP_PLUS_APP=https://app.excalidraw.com | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,37 +0,0 @@ | |||||||
| version: 2 |  | ||||||
| updates: |  | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: / |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       day: sunday |  | ||||||
|       time: "01:00" |  | ||||||
|     reviewers: |  | ||||||
|       - lipis |  | ||||||
|     assignees: |  | ||||||
|       - lipis |  | ||||||
|     open-pull-requests-limit: 20 |  | ||||||
|  |  | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: /src/packages/excalidraw/ |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       day: sunday |  | ||||||
|       time: "01:00" |  | ||||||
|     reviewers: |  | ||||||
|       - ad1992 |  | ||||||
|     assignees: |  | ||||||
|       - ad1992 |  | ||||||
|     open-pull-requests-limit: 20 |  | ||||||
|  |  | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: /src/packages/utils/ |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       day: sunday |  | ||||||
|       time: "01:00" |  | ||||||
|     reviewers: |  | ||||||
|       - ad1992 |  | ||||||
|     assignees: |  | ||||||
|       - ad1992 |  | ||||||
|     open-pull-requests-limit: 20 |  | ||||||
							
								
								
									
										2
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| name: Auto release @excalidraw/excalidraw-next | name: Auto release excalidraw next | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| name: Auto release preview @excalidraw/excalidraw-preview | name: Auto release excalidraw preview | ||||||
| on: | on: | ||||||
|   issue_comment: |   issue_comment: | ||||||
|     types: [created, edited] |     types: [created, edited] | ||||||
| @@ -6,7 +6,7 @@ on: | |||||||
| jobs: | jobs: | ||||||
|   Auto-release-excalidraw-preview: |   Auto-release-excalidraw-preview: | ||||||
|     name: Auto release preview |     name: Auto release preview | ||||||
|     if: github.event.comment.body == '@excalibot release package' && github.event.issue.pull_request |     if: github.event.comment.body == '@excalibot trigger release' && github.event.issue.pull_request | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: React to release comment |       - name: React to release comment | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,29 +0,0 @@ | |||||||
| 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 |  | ||||||
							
								
								
									
										47
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										47
									
								
								README.md
									
									
									
									
									
								
							| @@ -32,6 +32,10 @@ Last but not least, we're thankful to these companies for offering their service | |||||||
|  |  | ||||||
| [](https://vercel.com) [](https://sentry.io) [](https://crowdin.com) | [](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 | ## Documentation | ||||||
|  |  | ||||||
| ### Shortcuts | ### Shortcuts | ||||||
| @@ -124,14 +128,41 @@ For collaboration, you will need to set up [collab server](https://github.com/ex | |||||||
|  |  | ||||||
| #### Commands | #### Commands | ||||||
|  |  | ||||||
| | Command            | Description                       | | ##### Install the dependencies | ||||||
| | ------------------ | --------------------------------- | |  | ||||||
| | `yarn`             | Install the dependencies          | | ``` | ||||||
| | `yarn start`       | Run the project                   | | yarn | ||||||
| | `yarn fix`         | Reformat all files with Prettier  | | ``` | ||||||
| | `yarn test`        | Run tests                         | |  | ||||||
| | `yarn test:update` | Update test snapshots             | | ##### Run the project | ||||||
| | `yarn test:code`   | Test for formatting with Prettier | |  | ||||||
|  | ``` | ||||||
|  | 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 | ||||||
|  | ``` | ||||||
|  |  | ||||||
| #### Docker Compose | #### Docker Compose | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,12 +1,11 @@ | |||||||
| rules_version = '2'; | rules_version = '2'; | ||||||
| service firebase.storage { | service firebase.storage { | ||||||
|   match /b/{bucket}/o { |   match /b/{bucket}/o { | ||||||
|     match /{migrations} { |     match /{files}/rooms/{room}/{file} { | ||||||
|       match /{scenes}/{scene} { |  | ||||||
|     	allow get, write: if true; |     	allow get, write: if true; | ||||||
|         // redundant, but let's be explicit' |  | ||||||
|         allow list: if false; |  | ||||||
|     } |     } | ||||||
|  |     match /{files}/shareLinks/{shareLink}/{file} { | ||||||
|  |     	allow get, write: if true; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										39
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								package.json
									
									
									
									
									
								
							| @@ -22,22 +22,23 @@ | |||||||
|     "@sentry/browser": "6.2.5", |     "@sentry/browser": "6.2.5", | ||||||
|     "@sentry/integrations": "6.2.5", |     "@sentry/integrations": "6.2.5", | ||||||
|     "@testing-library/jest-dom": "5.16.2", |     "@testing-library/jest-dom": "5.16.2", | ||||||
|     "@testing-library/react": "12.1.2", |     "@testing-library/react": "12.1.5", | ||||||
|     "@tldraw/vec": "1.4.3", |     "@tldraw/vec": "1.7.1", | ||||||
|     "@types/jest": "27.4.0", |     "@types/jest": "27.4.0", | ||||||
|     "@types/pica": "5.1.3", |     "@types/pica": "5.1.3", | ||||||
|     "@types/react": "17.0.38", |     "@types/react": "17.0.39", | ||||||
|     "@types/react-dom": "17.0.11", |     "@types/react-dom": "17.0.11", | ||||||
|     "@types/socket.io-client": "1.4.36", |     "@types/socket.io-client": "1.4.36", | ||||||
|     "browser-fs-access": "0.23.0", |     "browser-fs-access": "0.29.1", | ||||||
|     "clsx": "1.1.1", |     "clsx": "1.1.1", | ||||||
|     "fake-indexeddb": "3.1.7", |     "fake-indexeddb": "3.1.7", | ||||||
|     "firebase": "8.3.3", |     "firebase": "8.3.3", | ||||||
|     "i18next-browser-languagedetector": "6.1.2", |     "i18next-browser-languagedetector": "6.1.4", | ||||||
|     "idb-keyval": "6.0.3", |     "idb-keyval": "6.0.3", | ||||||
|     "image-blob-reduce": "3.0.1", |     "image-blob-reduce": "3.0.1", | ||||||
|  |     "jotai": "1.6.4", | ||||||
|     "lodash.throttle": "4.1.1", |     "lodash.throttle": "4.1.1", | ||||||
|     "nanoid": "3.1.32", |     "nanoid": "3.3.3", | ||||||
|     "open-color": "1.9.1", |     "open-color": "1.9.1", | ||||||
|     "pako": "1.0.11", |     "pako": "1.0.11", | ||||||
|     "perfect-freehand": "1.0.16", |     "perfect-freehand": "1.0.16", | ||||||
| @@ -50,7 +51,7 @@ | |||||||
|     "react-dom": "17.0.2", |     "react-dom": "17.0.2", | ||||||
|     "react-scripts": "4.0.3", |     "react-scripts": "4.0.3", | ||||||
|     "roughjs": "4.5.2", |     "roughjs": "4.5.2", | ||||||
|     "sass": "1.49.7", |     "sass": "1.51.0", | ||||||
|     "socket.io-client": "2.3.1", |     "socket.io-client": "2.3.1", | ||||||
|     "typescript": "4.5.5" |     "typescript": "4.5.5" | ||||||
|   }, |   }, | ||||||
| @@ -58,20 +59,19 @@ | |||||||
|     "@excalidraw/eslint-config": "1.0.0", |     "@excalidraw/eslint-config": "1.0.0", | ||||||
|     "@excalidraw/prettier-config": "1.0.2", |     "@excalidraw/prettier-config": "1.0.2", | ||||||
|     "@types/chai": "4.3.0", |     "@types/chai": "4.3.0", | ||||||
|     "@types/lodash.throttle": "4.1.6", |     "@types/lodash.throttle": "4.1.7", | ||||||
|     "@types/pako": "1.0.3", |     "@types/pako": "1.0.3", | ||||||
|     "@types/resize-observer-browser": "0.1.6", |     "@types/resize-observer-browser": "0.1.7", | ||||||
|     "chai": "4.3.6", |     "chai": "4.3.6", | ||||||
|     "dotenv": "10.0.0", |     "dotenv": "16.0.1", | ||||||
|     "eslint-config-prettier": "8.3.0", |     "eslint-config-prettier": "8.5.0", | ||||||
|     "eslint-plugin-prettier": "3.3.1", |     "eslint-plugin-prettier": "3.3.1", | ||||||
|     "firebase-tools": "9.23.0", |  | ||||||
|     "husky": "7.0.4", |     "husky": "7.0.4", | ||||||
|     "jest-canvas-mock": "2.3.1", |     "jest-canvas-mock": "2.4.0", | ||||||
|     "lint-staged": "12.3.3", |     "lint-staged": "12.3.7", | ||||||
|     "pepjs": "0.5.3", |     "pepjs": "0.5.3", | ||||||
|     "prettier": "2.5.1", |     "prettier": "2.6.2", | ||||||
|     "rewire": "5.0.0" |     "rewire": "6.0.0" | ||||||
|   }, |   }, | ||||||
|   "resolutions": { |   "resolutions": { | ||||||
|     "@typescript-eslint/typescript-estree": "5.10.2" |     "@typescript-eslint/typescript-estree": "5.10.2" | ||||||
| @@ -94,7 +94,8 @@ | |||||||
|     "build:app:docker": "REACT_APP_DISABLE_SENTRY=true react-scripts build", |     "build:app:docker": "REACT_APP_DISABLE_SENTRY=true react-scripts build", | ||||||
|     "build:app": "REACT_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA react-scripts build", |     "build:app": "REACT_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA react-scripts build", | ||||||
|     "build:version": "node ./scripts/build-version.js", |     "build:version": "node ./scripts/build-version.js", | ||||||
|     "build": "yarn build:app && yarn build:version", |     "build:prebuild": "node ./scripts/prebuild.js", | ||||||
|  |     "build": "yarn build:prebuild && yarn build:app && yarn build:version", | ||||||
|     "eject": "react-scripts eject", |     "eject": "react-scripts eject", | ||||||
|     "fix:code": "yarn test:code --fix", |     "fix:code": "yarn test:code --fix", | ||||||
|     "fix:other": "yarn prettier --write", |     "fix:other": "yarn prettier --write", | ||||||
| @@ -112,6 +113,8 @@ | |||||||
|     "test:typecheck": "tsc", |     "test:typecheck": "tsc", | ||||||
|     "test:update": "yarn test:app --updateSnapshot --watchAll=false", |     "test:update": "yarn test:app --updateSnapshot --watchAll=false", | ||||||
|     "test": "yarn test:app", |     "test": "yarn test:app", | ||||||
|     "autorelease": "node scripts/autorelease.js" |     "autorelease": "node scripts/autorelease.js", | ||||||
|  |     "prerelease": "node scripts/prerelease.js", | ||||||
|  |     "release": "node scripts/release.js" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -52,6 +52,25 @@ | |||||||
|       content="Excalidraw is a whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them." |       content="Excalidraw is a whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them." | ||||||
|     /> |     /> | ||||||
|  |  | ||||||
|  |     <script> | ||||||
|  |       // Redirect Excalidraw+ users which have auto-redirect enabled. | ||||||
|  |       // | ||||||
|  |       // Redirect only the bare root path, so link/room/library urls are not | ||||||
|  |       // redirected. | ||||||
|  |       // | ||||||
|  |       // Putting into index.html for best performance (can't redirect on server | ||||||
|  |       // due to location.hash checks). | ||||||
|  |       if ( | ||||||
|  |         window.location.pathname === "/" && | ||||||
|  |         !window.location.hash && | ||||||
|  |         !window.location.search && | ||||||
|  |         // if its present redirect | ||||||
|  |         document.cookie.includes("excplus-autoredirect=true") | ||||||
|  |       ) { | ||||||
|  |         window.location.href = "https://app.excalidraw.com"; | ||||||
|  |       } | ||||||
|  |     </script> | ||||||
|  |  | ||||||
|     <link rel="shortcut icon" href="favicon.ico" type="image/x-icon" /> |     <link rel="shortcut icon" href="favicon.ico" type="image/x-icon" /> | ||||||
|  |  | ||||||
|     <!-- Excalidraw version --> |     <!-- Excalidraw version --> | ||||||
| @@ -72,12 +91,6 @@ | |||||||
|       crossorigin="anonymous" |       crossorigin="anonymous" | ||||||
|     /> |     /> | ||||||
|  |  | ||||||
|     <link |  | ||||||
|       href="%REACT_APP_SOCKET_SERVER_URL%/socket.io" |  | ||||||
|       rel="preconnect" |  | ||||||
|       crossorigin="anonymous" |  | ||||||
|     /> |  | ||||||
|  |  | ||||||
|     <link |     <link | ||||||
|       rel="manifest" |       rel="manifest" | ||||||
|       href="manifest.json" |       href="manifest.json" | ||||||
| @@ -85,6 +98,22 @@ | |||||||
|     /> |     /> | ||||||
|  |  | ||||||
|     <link rel="stylesheet" href="fonts.css" type="text/css" /> |     <link rel="stylesheet" href="fonts.css" type="text/css" /> | ||||||
|  |     <% if (process.env.REACT_APP_DEV_DISABLE_LIVE_RELOAD === "true") { %> | ||||||
|  |     <script> | ||||||
|  |       { | ||||||
|  |         const _WebSocket = window.WebSocket; | ||||||
|  |         window.WebSocket = function (url) { | ||||||
|  |           if (/ws:\/\/localhost:.+?\/sockjs-node/.test(url)) { | ||||||
|  |             console.info( | ||||||
|  |               "[!!!] Live reload is disabled via process.env.REACT_APP_DEV_DISABLE_LIVE_RELOAD [!!!]", | ||||||
|  |             ); | ||||||
|  |           } else { | ||||||
|  |             return new _WebSocket(url); | ||||||
|  |           } | ||||||
|  |         }; | ||||||
|  |       } | ||||||
|  |     </script> | ||||||
|  |     <% } %> | ||||||
|     <script> |     <script> | ||||||
|       window.EXCALIDRAW_ASSET_PATH = "/"; |       window.EXCALIDRAW_ASSET_PATH = "/"; | ||||||
|       // setting this so that libraries installation reuses this window tab. |       // setting this so that libraries installation reuses this window tab. | ||||||
| @@ -130,26 +159,6 @@ | |||||||
|         user-select: none; |         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 { |       #root { | ||||||
|         height: 100%; |         height: 100%; | ||||||
|         -webkit-touch-callout: none; |         -webkit-touch-callout: none; | ||||||
| @@ -158,8 +167,10 @@ | |||||||
|         -moz-user-select: none; |         -moz-user-select: none; | ||||||
|         -ms-user-select: none; |         -ms-user-select: none; | ||||||
|         user-select: none; |         user-select: none; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       @media screen and (min-width: 1200px) { |       @media screen and (min-width: 1200px) { | ||||||
|  |         #root { | ||||||
|           -webkit-touch-callout: default; |           -webkit-touch-callout: default; | ||||||
|           -webkit-user-select: auto; |           -webkit-user-select: auto; | ||||||
|           -khtml-user-select: auto; |           -khtml-user-select: auto; | ||||||
| @@ -176,10 +187,6 @@ | |||||||
|     <header> |     <header> | ||||||
|       <h1 class="visually-hidden">Excalidraw</h1> |       <h1 class="visually-hidden">Excalidraw</h1> | ||||||
|     </header> |     </header> | ||||||
|     <div id="root"> |     <div id="root"></div> | ||||||
|       <div class="LoadingMessage"> |  | ||||||
|         <span>Loading scene...</span> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   </body> |   </body> | ||||||
| </html> | </html> | ||||||
|   | |||||||
| @@ -17,11 +17,23 @@ | |||||||
|  * See https://goo.gl/2aRDsh
 |  * See https://goo.gl/2aRDsh
 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| importScripts("/workbox/workbox-sw.js"); | // in dev, `process` is undefined because this file is not compiled until build
 | ||||||
|  | const IS_DEVELOPMENT = | ||||||
|  |   typeof process === "undefined" || process.env.NODE_ENV !== "production"; | ||||||
| 
 | 
 | ||||||
| workbox.setConfig({ | if (IS_DEVELOPMENT) { | ||||||
|  |   importScripts( | ||||||
|  |     "https://storage.googleapis.com/workbox-cdn/releases/4.3.1/workbox-sw.js", | ||||||
|  |   ); | ||||||
|  |   workbox.setConfig({ | ||||||
|  |     debug: true, | ||||||
|  |   }); | ||||||
|  | } else { | ||||||
|  |   importScripts("/workbox/workbox-sw.js"); | ||||||
|  |   workbox.setConfig({ | ||||||
|     modulePathPrefix: "/workbox/", |     modulePathPrefix: "/workbox/", | ||||||
| }); |   }); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| self.addEventListener("message", (event) => { | self.addEventListener("message", (event) => { | ||||||
|   if (event.data && event.data.type === "SKIP_WAITING") { |   if (event.data && event.data.type === "SKIP_WAITING") { | ||||||
| @@ -30,14 +42,17 @@ self.addEventListener("message", (event) => { | |||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| workbox.core.clientsClaim(); | workbox.core.clientsClaim(); | ||||||
| workbox.precaching.precacheAndRoute(self.__WB_MANIFEST); |  | ||||||
| 
 | 
 | ||||||
| workbox.routing.registerNavigationRoute( | if (!IS_DEVELOPMENT) { | ||||||
|  |   workbox.precaching.precacheAndRoute(self.__WB_MANIFEST); | ||||||
|  | 
 | ||||||
|  |   workbox.routing.registerNavigationRoute( | ||||||
|     workbox.precaching.getCacheKeyForURL("./index.html"), |     workbox.precaching.getCacheKeyForURL("./index.html"), | ||||||
|     { |     { | ||||||
|       blacklist: [/^\/_/, /\/[^/?]+\.[^/]+$/], |       blacklist: [/^\/_/, /\/[^/?]+\.[^/]+$/], | ||||||
|     }, |     }, | ||||||
| ); |   ); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| // Cache relevant font files
 | // Cache relevant font files
 | ||||||
| workbox.routing.registerRoute( | workbox.routing.registerRoute( | ||||||
| @@ -5,22 +5,25 @@ const core = require("@actions/core"); | |||||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | ||||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; | const excalidrawPackage = `${excalidrawDir}/package.json`; | ||||||
| const pkg = require(excalidrawPackage); | const pkg = require(excalidrawPackage); | ||||||
|  | const isPreview = process.argv.slice(2)[0] === "preview"; | ||||||
|  |  | ||||||
| const getShortCommitHash = () => { | const getShortCommitHash = () => { | ||||||
|   return execSync("git rev-parse --short HEAD").toString().trim(); |   return execSync("git rev-parse --short HEAD").toString().trim(); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const publish = () => { | const publish = () => { | ||||||
|  |   const tag = isPreview ? "preview" : "next"; | ||||||
|  |  | ||||||
|   try { |   try { | ||||||
|     execSync(`yarn  --frozen-lockfile`); |     execSync(`yarn  --frozen-lockfile`); | ||||||
|     execSync(`yarn --frozen-lockfile`, { cwd: excalidrawDir }); |     execSync(`yarn --frozen-lockfile`, { cwd: excalidrawDir }); | ||||||
|     execSync(`yarn run build:umd`, { cwd: excalidrawDir }); |     execSync(`yarn run build:umd`, { cwd: excalidrawDir }); | ||||||
|     execSync(`yarn --cwd ${excalidrawDir} publish`); |     execSync(`yarn --cwd ${excalidrawDir} publish --tag ${tag}`); | ||||||
|     console.info("Published 🎉"); |     console.info(`Published ${pkg.name}@${tag}🎉`); | ||||||
|     core.setOutput( |     core.setOutput( | ||||||
|       "result", |       "result", | ||||||
|       `**Preview version has been shipped** :rocket: |       `**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!`, |     You can use [@excalidraw/excalidraw@${pkg.version}](https://www.npmjs.com/package/@excalidraw/excalidraw/v/${pkg.version}) for testing!`, | ||||||
|     ); |     ); | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     core.setOutput("result", "package couldn't be published :warning:!"); |     core.setOutput("result", "package couldn't be published :warning:!"); | ||||||
| @@ -51,27 +54,19 @@ exec(`git diff --name-only HEAD^ HEAD`, async (error, stdout, stderr) => { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   // update package.json |   // update package.json | ||||||
|   pkg.name = "@excalidraw/excalidraw-next"; |  | ||||||
|   let version = `${pkg.version}-${getShortCommitHash()}`; |   let version = `${pkg.version}-${getShortCommitHash()}`; | ||||||
|  |  | ||||||
|   // update readme |   // update readme | ||||||
|   let data = fs.readFileSync(`${excalidrawDir}/README_NEXT.md`, "utf8"); |  | ||||||
|  |  | ||||||
|   const isPreview = process.argv.slice(2)[0] === "preview"; |  | ||||||
|   if (isPreview) { |   if (isPreview) { | ||||||
|     // use pullNumber-commithash as the version for preview |     // use pullNumber-commithash as the version for preview | ||||||
|     const pullRequestNumber = process.argv.slice(3)[0]; |     const pullRequestNumber = process.argv.slice(3)[0]; | ||||||
|     version = `${pkg.version}-${pullRequestNumber}-${getShortCommitHash()}`; |     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; |   pkg.version = version; | ||||||
|  |  | ||||||
|   fs.writeFileSync(excalidrawPackage, JSON.stringify(pkg, null, 2), "utf8"); |   fs.writeFileSync(excalidrawPackage, JSON.stringify(pkg, null, 2), "utf8"); | ||||||
|  |  | ||||||
|   fs.writeFileSync(`${excalidrawDir}/README.md`, data, "utf8"); |  | ||||||
|   console.info("Publish in progress..."); |   console.info("Publish in progress..."); | ||||||
|   publish(); |   publish(); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -36,6 +36,7 @@ const crowdinMap = { | |||||||
|   "ru-RU": "en-ru", |   "ru-RU": "en-ru", | ||||||
|   "si-LK": "en-silk", |   "si-LK": "en-silk", | ||||||
|   "sk-SK": "en-sk", |   "sk-SK": "en-sk", | ||||||
|  |   "sl-SI": "en-sl", | ||||||
|   "sv-SE": "en-sv", |   "sv-SE": "en-sv", | ||||||
|   "ta-IN": "en-ta", |   "ta-IN": "en-ta", | ||||||
|   "tr-TR": "en-tr", |   "tr-TR": "en-tr", | ||||||
| @@ -47,6 +48,8 @@ const crowdinMap = { | |||||||
|   "lv-LV": "en-lv", |   "lv-LV": "en-lv", | ||||||
|   "cs-CZ": "en-cs", |   "cs-CZ": "en-cs", | ||||||
|   "kk-KZ": "en-kk", |   "kk-KZ": "en-kk", | ||||||
|  |   "vi-vn": "en-vi", | ||||||
|  |   "mr-in": "en-mr", | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const flags = { | const flags = { | ||||||
| @@ -86,6 +89,7 @@ const flags = { | |||||||
|   "ru-RU": "🇷🇺", |   "ru-RU": "🇷🇺", | ||||||
|   "si-LK": "🇱🇰", |   "si-LK": "🇱🇰", | ||||||
|   "sk-SK": "🇸🇰", |   "sk-SK": "🇸🇰", | ||||||
|  |   "sl-SI": "🇸🇮", | ||||||
|   "sv-SE": "🇸🇪", |   "sv-SE": "🇸🇪", | ||||||
|   "ta-IN": "🇮🇳", |   "ta-IN": "🇮🇳", | ||||||
|   "tr-TR": "🇹🇷", |   "tr-TR": "🇹🇷", | ||||||
| @@ -93,6 +97,9 @@ const flags = { | |||||||
|   "zh-CN": "🇨🇳", |   "zh-CN": "🇨🇳", | ||||||
|   "zh-HK": "🇭🇰", |   "zh-HK": "🇭🇰", | ||||||
|   "zh-TW": "🇹🇼", |   "zh-TW": "🇹🇼", | ||||||
|  |   "eu-ES": "🇪🇦", | ||||||
|  |   "vi-VN": "🇻🇳", | ||||||
|  |   "mr-IN": "🇮🇳", | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const languages = { | const languages = { | ||||||
| @@ -133,6 +140,7 @@ const languages = { | |||||||
|   "ru-RU": "Русский", |   "ru-RU": "Русский", | ||||||
|   "si-LK": "සිංහල", |   "si-LK": "සිංහල", | ||||||
|   "sk-SK": "Slovenčina", |   "sk-SK": "Slovenčina", | ||||||
|  |   "sl-SI": "Slovenščina", | ||||||
|   "sv-SE": "Svenska", |   "sv-SE": "Svenska", | ||||||
|   "ta-IN": "Tamil", |   "ta-IN": "Tamil", | ||||||
|   "tr-TR": "Türkçe", |   "tr-TR": "Türkçe", | ||||||
| @@ -140,6 +148,8 @@ const languages = { | |||||||
|   "zh-CN": "简体中文", |   "zh-CN": "简体中文", | ||||||
|   "zh-HK": "繁體中文 (香港)", |   "zh-HK": "繁體中文 (香港)", | ||||||
|   "zh-TW": "繁體中文", |   "zh-TW": "繁體中文", | ||||||
|  |   "vi-VN": "Tiếng Việt", | ||||||
|  |   "mr-IN": "मराठी", | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const percentages = fs.readFileSync( | const percentages = fs.readFileSync( | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								scripts/prebuild.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								scripts/prebuild.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | const fs = require("fs"); | ||||||
|  |  | ||||||
|  | // for development purposes we want to have the service-worker.js file | ||||||
|  | // accessible from the public folder. On build though, we need to compile it | ||||||
|  | // and CRA expects that file to be in src/ folder. | ||||||
|  | const moveServiceWorkerScript = () => { | ||||||
|  |   const oldPath = "./public/service-worker.js"; | ||||||
|  |   const newPath = "./src/service-worker.js"; | ||||||
|  |  | ||||||
|  |   fs.rename(oldPath, newPath, (error) => { | ||||||
|  |     if (error) { | ||||||
|  |       throw error; | ||||||
|  |     } | ||||||
|  |     console.info("public/service-worker.js moved to src/"); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // ----------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | moveServiceWorkerScript(); | ||||||
							
								
								
									
										37
									
								
								scripts/prerelease.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								scripts/prerelease.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | const fs = require("fs"); | ||||||
|  | const util = require("util"); | ||||||
|  | const exec = util.promisify(require("child_process").exec); | ||||||
|  | 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 prerelease = async (nextVersion) => { | ||||||
|  |   try { | ||||||
|  |     await updateChangelog(nextVersion); | ||||||
|  |     updatePackageVersion(nextVersion); | ||||||
|  |     await exec(`git add -u`); | ||||||
|  |     await exec( | ||||||
|  |       `git commit -m "docs: release @excalidraw/excalidraw@${nextVersion}  🎉"`, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     console.info("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); | ||||||
|  | } | ||||||
|  | prerelease(nextVersion); | ||||||
| @@ -1,39 +1,44 @@ | |||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const util = require("util"); | const { execSync } = require("child_process"); | ||||||
| const exec = util.promisify(require("child_process").exec); |  | ||||||
| const updateReadme = require("./updateReadme"); |  | ||||||
| const updateChangelog = require("./updateChangelog"); |  | ||||||
|  |  | ||||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | ||||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; | const excalidrawPackage = `${excalidrawDir}/package.json`; | ||||||
|  | const pkg = require(excalidrawPackage); | ||||||
|  |  | ||||||
| const updatePackageVersion = (nextVersion) => { | const originalReadMe = fs.readFileSync(`${excalidrawDir}/README.md`, "utf8"); | ||||||
|   const pkg = require(excalidrawPackage); |  | ||||||
|   pkg.version = nextVersion; | const updateReadme = () => { | ||||||
|   const content = `${JSON.stringify(pkg, null, 2)}\n`; |   const excalidrawIndex = originalReadMe.indexOf("### Excalidraw"); | ||||||
|   fs.writeFileSync(excalidrawPackage, content, "utf-8"); |  | ||||||
|  |   // remove note for stable readme | ||||||
|  |   const data = originalReadMe.slice(excalidrawIndex); | ||||||
|  |  | ||||||
|  |   // update readme | ||||||
|  |   fs.writeFileSync(`${excalidrawDir}/README.md`, data, "utf8"); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const release = async (nextVersion) => { | const publish = () => { | ||||||
|   try { |   try { | ||||||
|     updateReadme(); |     execSync(`yarn  --frozen-lockfile`); | ||||||
|     await updateChangelog(nextVersion); |     execSync(`yarn --frozen-lockfile`, { cwd: excalidrawDir }); | ||||||
|     updatePackageVersion(nextVersion); |     execSync(`yarn run build:umd`, { cwd: excalidrawDir }); | ||||||
|     await exec(`git add -u`); |     execSync(`yarn --cwd ${excalidrawDir} publish`); | ||||||
|     await exec( |  | ||||||
|       `git commit -m "docs: release @excalidraw/excalidraw@${nextVersion}  🎉"`, |  | ||||||
|     ); |  | ||||||
|     /* eslint-disable no-console */ |  | ||||||
|     console.log("Done!"); |  | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     console.error(error); |     console.error(error); | ||||||
|     process.exit(1); |     process.exit(1); | ||||||
|   } |   } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const nextVersion = process.argv.slice(2)[0]; | const release = () => { | ||||||
| if (!nextVersion) { |   updateReadme(); | ||||||
|   console.error("Pass the next version to release!"); |   console.info("Note for stable readme removed"); | ||||||
|   process.exit(1); |  | ||||||
| } |   publish(); | ||||||
| release(nextVersion); |   console.info(`Published ${pkg.version}!`); | ||||||
|  |  | ||||||
|  |   // revert readme after release | ||||||
|  |   fs.writeFileSync(`${excalidrawDir}/README.md`, originalReadMe, "utf8"); | ||||||
|  |   console.info("Readme reverted"); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | release(); | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ const headerForType = { | |||||||
|   perf: "Performance", |   perf: "Performance", | ||||||
|   build: "Build", |   build: "Build", | ||||||
| }; | }; | ||||||
|  | const badCommits = []; | ||||||
| const getCommitHashForLastVersion = async () => { | const getCommitHashForLastVersion = async () => { | ||||||
|   try { |   try { | ||||||
|     const commitMessage = `"release @excalidraw/excalidraw@${lastVersion}"`; |     const commitMessage = `"release @excalidraw/excalidraw@${lastVersion}"`; | ||||||
| @@ -53,7 +53,9 @@ const getLibraryCommitsSinceLastRelease = async () => { | |||||||
|     const messageWithoutType = commit.slice(indexOfColon + 1).trim(); |     const messageWithoutType = commit.slice(indexOfColon + 1).trim(); | ||||||
|     const messageWithCapitalizeFirst = |     const messageWithCapitalizeFirst = | ||||||
|       messageWithoutType.charAt(0).toUpperCase() + messageWithoutType.slice(1); |       messageWithoutType.charAt(0).toUpperCase() + messageWithoutType.slice(1); | ||||||
|     const prNumber = commit.match(/\(#([0-9]*)\)/)[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 |       // return if the changelog already contains the pr number which would happen for package updates | ||||||
|       if (existingChangeLog.includes(prNumber)) { |       if (existingChangeLog.includes(prNumber)) { | ||||||
| @@ -65,7 +67,12 @@ const getLibraryCommitsSinceLastRelease = async () => { | |||||||
|         prMarkdown, |         prMarkdown, | ||||||
|       ); |       ); | ||||||
|       commitList[type].push(messageWithPRLink); |       commitList[type].push(messageWithPRLink); | ||||||
|  |     } else { | ||||||
|  |       badCommits.push(commit); | ||||||
|  |       commitList[type].push(messageWithCapitalizeFirst); | ||||||
|  |     } | ||||||
|   }); |   }); | ||||||
|  |   console.info("Bad commits:", badCommits); | ||||||
|   return commitList; |   return commitList; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -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; |  | ||||||
| @@ -7,6 +7,7 @@ import { t } from "../i18n"; | |||||||
|  |  | ||||||
| export const actionAddToLibrary = register({ | export const actionAddToLibrary = register({ | ||||||
|   name: "addToLibrary", |   name: "addToLibrary", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState, _, app) => { |   perform: (elements, appState, _, app) => { | ||||||
|     const selectedElements = getSelectedElements( |     const selectedElements = getSelectedElements( | ||||||
|       getNonDeletedElements(elements), |       getNonDeletedElements(elements), | ||||||
| @@ -24,9 +25,9 @@ export const actionAddToLibrary = register({ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     return app.library |     return app.library | ||||||
|       .loadLibrary() |       .getLatestLibrary() | ||||||
|       .then((items) => { |       .then((items) => { | ||||||
|         return app.library.saveLibrary([ |         return app.library.setLibrary([ | ||||||
|           { |           { | ||||||
|             id: randomId(), |             id: randomId(), | ||||||
|             status: "unpublished", |             status: "unpublished", | ||||||
| @@ -41,7 +42,7 @@ export const actionAddToLibrary = register({ | |||||||
|           commitToHistory: false, |           commitToHistory: false, | ||||||
|           appState: { |           appState: { | ||||||
|             ...appState, |             ...appState, | ||||||
|             toastMessage: t("toast.addedToLibrary"), |             toast: { message: t("toast.addedToLibrary") }, | ||||||
|           }, |           }, | ||||||
|         }; |         }; | ||||||
|       }) |       }) | ||||||
|   | |||||||
| @@ -43,6 +43,7 @@ const alignSelectedElements = ( | |||||||
|  |  | ||||||
| export const actionAlignTop = register({ | export const actionAlignTop = register({ | ||||||
|   name: "alignTop", |   name: "alignTop", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -72,6 +73,7 @@ export const actionAlignTop = register({ | |||||||
|  |  | ||||||
| export const actionAlignBottom = register({ | export const actionAlignBottom = register({ | ||||||
|   name: "alignBottom", |   name: "alignBottom", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -101,6 +103,7 @@ export const actionAlignBottom = register({ | |||||||
|  |  | ||||||
| export const actionAlignLeft = register({ | export const actionAlignLeft = register({ | ||||||
|   name: "alignLeft", |   name: "alignLeft", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -130,6 +133,8 @@ export const actionAlignLeft = register({ | |||||||
|  |  | ||||||
| export const actionAlignRight = register({ | export const actionAlignRight = register({ | ||||||
|   name: "alignRight", |   name: "alignRight", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|  |  | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -159,6 +164,8 @@ export const actionAlignRight = register({ | |||||||
|  |  | ||||||
| export const actionAlignVerticallyCentered = register({ | export const actionAlignVerticallyCentered = register({ | ||||||
|   name: "alignVerticallyCentered", |   name: "alignVerticallyCentered", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|  |  | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -184,6 +191,7 @@ export const actionAlignVerticallyCentered = register({ | |||||||
|  |  | ||||||
| export const actionAlignHorizontallyCentered = register({ | export const actionAlignHorizontallyCentered = register({ | ||||||
|   name: "alignHorizontallyCentered", |   name: "alignHorizontallyCentered", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|   | |||||||
							
								
								
									
										136
									
								
								src/actions/actionBoundText.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								src/actions/actionBoundText.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,136 @@ | |||||||
|  | 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,5 +1,5 @@ | |||||||
| import { ColorPicker } from "../components/ColorPicker"; | import { ColorPicker } from "../components/ColorPicker"; | ||||||
| import { zoomIn, zoomOut } from "../components/icons"; | import { eraser, zoomIn, zoomOut } from "../components/icons"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| import { DarkModeToggle } from "../components/DarkModeToggle"; | import { DarkModeToggle } from "../components/DarkModeToggle"; | ||||||
| import { THEME, ZOOM_STEP } from "../constants"; | import { THEME, ZOOM_STEP } from "../constants"; | ||||||
| @@ -11,22 +11,24 @@ import { getNormalizedZoom, getSelectedElements } from "../scene"; | |||||||
| import { centerScrollOn } from "../scene/scroll"; | import { centerScrollOn } from "../scene/scroll"; | ||||||
| import { getStateForZoom } from "../scene/zoom"; | import { getStateForZoom } from "../scene/zoom"; | ||||||
| import { AppState, NormalizedZoomValue } from "../types"; | import { AppState, NormalizedZoomValue } from "../types"; | ||||||
| import { getShortcutKey } from "../utils"; | import { getShortcutKey, updateActiveTool } from "../utils"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { Tooltip } from "../components/Tooltip"; | import { Tooltip } from "../components/Tooltip"; | ||||||
| import { newElementWith } from "../element/mutateElement"; | import { newElementWith } from "../element/mutateElement"; | ||||||
| import { getDefaultAppState } from "../appState"; | import { getDefaultAppState, isEraserActive } from "../appState"; | ||||||
| import ClearCanvas from "../components/ClearCanvas"; | import ClearCanvas from "../components/ClearCanvas"; | ||||||
|  | import clsx from "clsx"; | ||||||
|  |  | ||||||
| export const actionChangeViewBackgroundColor = register({ | export const actionChangeViewBackgroundColor = register({ | ||||||
|   name: "changeViewBackgroundColor", |   name: "changeViewBackgroundColor", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (_, appState, value) => { |   perform: (_, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       appState: { ...appState, ...value }, |       appState: { ...appState, ...value }, | ||||||
|       commitToHistory: !!value.viewBackgroundColor, |       commitToHistory: !!value.viewBackgroundColor, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData }) => { |   PanelComponent: ({ elements, appState, updateData }) => { | ||||||
|     return ( |     return ( | ||||||
|       <div style={{ position: "relative" }}> |       <div style={{ position: "relative" }}> | ||||||
|         <ColorPicker |         <ColorPicker | ||||||
| @@ -39,6 +41,8 @@ export const actionChangeViewBackgroundColor = register({ | |||||||
|             updateData({ openPopup: active ? "canvasColorPicker" : null }) |             updateData({ openPopup: active ? "canvasColorPicker" : null }) | ||||||
|           } |           } | ||||||
|           data-testid="canvas-background-picker" |           data-testid="canvas-background-picker" | ||||||
|  |           elements={elements} | ||||||
|  |           appState={appState} | ||||||
|         /> |         /> | ||||||
|       </div> |       </div> | ||||||
|     ); |     ); | ||||||
| @@ -47,6 +51,7 @@ export const actionChangeViewBackgroundColor = register({ | |||||||
|  |  | ||||||
| export const actionClearCanvas = register({ | export const actionClearCanvas = register({ | ||||||
|   name: "clearCanvas", |   name: "clearCanvas", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (elements, appState, _, app) => { |   perform: (elements, appState, _, app) => { | ||||||
|     app.imageCache.clear(); |     app.imageCache.clear(); | ||||||
|     return { |     return { | ||||||
| @@ -57,7 +62,6 @@ export const actionClearCanvas = register({ | |||||||
|         ...getDefaultAppState(), |         ...getDefaultAppState(), | ||||||
|         files: {}, |         files: {}, | ||||||
|         theme: appState.theme, |         theme: appState.theme, | ||||||
|         elementLocked: appState.elementLocked, |  | ||||||
|         penMode: appState.penMode, |         penMode: appState.penMode, | ||||||
|         penDetected: appState.penDetected, |         penDetected: appState.penDetected, | ||||||
|         exportBackground: appState.exportBackground, |         exportBackground: appState.exportBackground, | ||||||
| @@ -65,8 +69,10 @@ export const actionClearCanvas = register({ | |||||||
|         gridSize: appState.gridSize, |         gridSize: appState.gridSize, | ||||||
|         showStats: appState.showStats, |         showStats: appState.showStats, | ||||||
|         pasteDialog: appState.pasteDialog, |         pasteDialog: appState.pasteDialog, | ||||||
|         elementType: |         activeTool: | ||||||
|           appState.elementType === "image" ? "selection" : appState.elementType, |           appState.activeTool.type === "image" | ||||||
|  |             ? { ...appState.activeTool, type: "selection" } | ||||||
|  |             : appState.activeTool, | ||||||
|       }, |       }, | ||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
| @@ -77,6 +83,7 @@ export const actionClearCanvas = register({ | |||||||
|  |  | ||||||
| export const actionZoomIn = register({ | export const actionZoomIn = register({ | ||||||
|   name: "zoomIn", |   name: "zoomIn", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (_elements, appState, _, app) => { |   perform: (_elements, appState, _, app) => { | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
| @@ -112,6 +119,7 @@ export const actionZoomIn = register({ | |||||||
|  |  | ||||||
| export const actionZoomOut = register({ | export const actionZoomOut = register({ | ||||||
|   name: "zoomOut", |   name: "zoomOut", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (_elements, appState, _, app) => { |   perform: (_elements, appState, _, app) => { | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
| @@ -147,6 +155,7 @@ export const actionZoomOut = register({ | |||||||
|  |  | ||||||
| export const actionResetZoom = register({ | export const actionResetZoom = register({ | ||||||
|   name: "resetZoom", |   name: "resetZoom", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (_elements, appState, _, app) => { |   perform: (_elements, appState, _, app) => { | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
| @@ -245,6 +254,7 @@ const zoomToFitElements = ( | |||||||
|  |  | ||||||
| export const actionZoomToSelected = register({ | export const actionZoomToSelected = register({ | ||||||
|   name: "zoomToSelection", |   name: "zoomToSelection", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (elements, appState) => zoomToFitElements(elements, appState, true), |   perform: (elements, appState) => zoomToFitElements(elements, appState, true), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event.code === CODES.TWO && |     event.code === CODES.TWO && | ||||||
| @@ -255,6 +265,7 @@ export const actionZoomToSelected = register({ | |||||||
|  |  | ||||||
| export const actionZoomToFit = register({ | export const actionZoomToFit = register({ | ||||||
|   name: "zoomToFit", |   name: "zoomToFit", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (elements, appState) => zoomToFitElements(elements, appState, false), |   perform: (elements, appState) => zoomToFitElements(elements, appState, false), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event.code === CODES.ONE && |     event.code === CODES.ONE && | ||||||
| @@ -265,6 +276,7 @@ export const actionZoomToFit = register({ | |||||||
|  |  | ||||||
| export const actionToggleTheme = register({ | export const actionToggleTheme = register({ | ||||||
|   name: "toggleTheme", |   name: "toggleTheme", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (_, appState, value) => { |   perform: (_, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
| @@ -287,3 +299,49 @@ export const actionToggleTheme = register({ | |||||||
|   ), |   ), | ||||||
|   keyTest: (event) => event.altKey && event.shiftKey && event.code === CODES.D, |   keyTest: (event) => event.altKey && event.shiftKey && event.code === CODES.D, | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | export const actionErase = register({ | ||||||
|  |   name: "eraser", | ||||||
|  |   trackEvent: { category: "toolbar" }, | ||||||
|  |   perform: (elements, appState) => { | ||||||
|  |     let activeTool: AppState["activeTool"]; | ||||||
|  |  | ||||||
|  |     if (isEraserActive(appState)) { | ||||||
|  |       activeTool = updateActiveTool(appState, { | ||||||
|  |         ...(appState.activeTool.lastActiveToolBeforeEraser || { | ||||||
|  |           type: "selection", | ||||||
|  |         }), | ||||||
|  |         lastActiveToolBeforeEraser: null, | ||||||
|  |       }); | ||||||
|  |     } else { | ||||||
|  |       activeTool = updateActiveTool(appState, { | ||||||
|  |         type: "eraser", | ||||||
|  |         lastActiveToolBeforeEraser: appState.activeTool, | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return { | ||||||
|  |       appState: { | ||||||
|  |         ...appState, | ||||||
|  |         selectedElementIds: {}, | ||||||
|  |         selectedGroupIds: {}, | ||||||
|  |         activeTool, | ||||||
|  |       }, | ||||||
|  |       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,16 +1,23 @@ | |||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { copyToClipboard } from "../clipboard"; | import { | ||||||
|  |   copyTextToSystemClipboard, | ||||||
|  |   copyToClipboard, | ||||||
|  |   probablySupportsClipboardWriteText, | ||||||
|  | } from "../clipboard"; | ||||||
| import { actionDeleteSelected } from "./actionDeleteSelected"; | import { actionDeleteSelected } from "./actionDeleteSelected"; | ||||||
| import { getSelectedElements } from "../scene/selection"; | import { getSelectedElements } from "../scene/selection"; | ||||||
| import { exportCanvas } from "../data/index"; | import { exportCanvas } from "../data/index"; | ||||||
| import { getNonDeletedElements } from "../element"; | import { getNonDeletedElements, isTextElement } from "../element"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
|  |  | ||||||
| export const actionCopy = register({ | export const actionCopy = register({ | ||||||
|   name: "copy", |   name: "copy", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState, _, app) => { |   perform: (elements, appState, _, app) => { | ||||||
|     copyToClipboard(getNonDeletedElements(elements), appState, app.files); |     const selectedElements = getSelectedElements(elements, appState, true); | ||||||
|  |  | ||||||
|  |     copyToClipboard(selectedElements, appState, app.files); | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|       commitToHistory: false, |       commitToHistory: false, | ||||||
| @@ -23,6 +30,7 @@ export const actionCopy = register({ | |||||||
|  |  | ||||||
| export const actionCut = register({ | export const actionCut = register({ | ||||||
|   name: "cut", |   name: "cut", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState, data, app) => { |   perform: (elements, appState, data, app) => { | ||||||
|     actionCopy.perform(elements, appState, data, app); |     actionCopy.perform(elements, appState, data, app); | ||||||
|     return actionDeleteSelected.perform(elements, appState); |     return actionDeleteSelected.perform(elements, appState); | ||||||
| @@ -33,6 +41,7 @@ export const actionCut = register({ | |||||||
|  |  | ||||||
| export const actionCopyAsSvg = register({ | export const actionCopyAsSvg = register({ | ||||||
|   name: "copyAsSvg", |   name: "copyAsSvg", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: async (elements, appState, _data, app) => { |   perform: async (elements, appState, _data, app) => { | ||||||
|     if (!app.canvas) { |     if (!app.canvas) { | ||||||
|       return { |       return { | ||||||
| @@ -73,6 +82,7 @@ export const actionCopyAsSvg = register({ | |||||||
|  |  | ||||||
| export const actionCopyAsPng = register({ | export const actionCopyAsPng = register({ | ||||||
|   name: "copyAsPng", |   name: "copyAsPng", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: async (elements, appState, _data, app) => { |   perform: async (elements, appState, _data, app) => { | ||||||
|     if (!app.canvas) { |     if (!app.canvas) { | ||||||
|       return { |       return { | ||||||
| @@ -97,7 +107,8 @@ export const actionCopyAsPng = register({ | |||||||
|       return { |       return { | ||||||
|         appState: { |         appState: { | ||||||
|           ...appState, |           ...appState, | ||||||
|           toastMessage: t("toast.copyToClipboardAsPng", { |           toast: { | ||||||
|  |             message: t("toast.copyToClipboardAsPng", { | ||||||
|               exportSelection: selectedElements.length |               exportSelection: selectedElements.length | ||||||
|                 ? t("toast.selection") |                 ? t("toast.selection") | ||||||
|                 : t("toast.canvas"), |                 : t("toast.canvas"), | ||||||
| @@ -106,6 +117,7 @@ export const actionCopyAsPng = register({ | |||||||
|                 : t("buttons.lightMode"), |                 : t("buttons.lightMode"), | ||||||
|             }), |             }), | ||||||
|           }, |           }, | ||||||
|  |         }, | ||||||
|         commitToHistory: false, |         commitToHistory: false, | ||||||
|       }; |       }; | ||||||
|     } catch (error: any) { |     } catch (error: any) { | ||||||
| @@ -122,3 +134,35 @@ export const actionCopyAsPng = register({ | |||||||
|   contextItemLabel: "labels.copyAsPng", |   contextItemLabel: "labels.copyAsPng", | ||||||
|   keyTest: (event) => event.code === CODES.C && event.altKey && event.shiftKey, |   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", | ||||||
|  | }); | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import { getElementsInGroup } from "../groups"; | |||||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | import { LinearElementEditor } from "../element/linearElementEditor"; | ||||||
| import { fixBindingsAfterDeletion } from "../element/binding"; | import { fixBindingsAfterDeletion } from "../element/binding"; | ||||||
| import { isBoundToContainer } from "../element/typeChecks"; | import { isBoundToContainer } from "../element/typeChecks"; | ||||||
|  | import { updateActiveTool } from "../utils"; | ||||||
|  |  | ||||||
| const deleteSelectedElements = ( | const deleteSelectedElements = ( | ||||||
|   elements: readonly ExcalidrawElement[], |   elements: readonly ExcalidrawElement[], | ||||||
| @@ -58,6 +59,7 @@ const handleGroupEditingState = ( | |||||||
|  |  | ||||||
| export const actionDeleteSelected = register({ | export const actionDeleteSelected = register({ | ||||||
|   name: "deleteSelectedElements", |   name: "deleteSelectedElements", | ||||||
|  |   trackEvent: { category: "element", action: "delete" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     if (appState.editingLinearElement) { |     if (appState.editingLinearElement) { | ||||||
|       const { |       const { | ||||||
| @@ -133,7 +135,7 @@ export const actionDeleteSelected = register({ | |||||||
|       elements: nextElements, |       elements: nextElements, | ||||||
|       appState: { |       appState: { | ||||||
|         ...nextAppState, |         ...nextAppState, | ||||||
|         elementType: "selection", |         activeTool: updateActiveTool(appState, { type: "selection" }), | ||||||
|         multiElement: null, |         multiElement: null, | ||||||
|       }, |       }, | ||||||
|       commitToHistory: isSomeElementSelected( |       commitToHistory: isSomeElementSelected( | ||||||
|   | |||||||
| @@ -3,11 +3,11 @@ import { | |||||||
|   DistributeVerticallyIcon, |   DistributeVerticallyIcon, | ||||||
| } from "../components/icons"; | } from "../components/icons"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| import { distributeElements, Distribution } from "../disitrubte"; | import { distributeElements, Distribution } from "../distribute"; | ||||||
| import { getNonDeletedElements } from "../element"; | import { getNonDeletedElements } from "../element"; | ||||||
| import { ExcalidrawElement } from "../element/types"; | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { CODES } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
| import { arrayToMap, getShortcutKey } from "../utils"; | import { arrayToMap, getShortcutKey } from "../utils"; | ||||||
| @@ -39,6 +39,7 @@ const distributeSelectedElements = ( | |||||||
|  |  | ||||||
| export const distributeHorizontally = register({ | export const distributeHorizontally = register({ | ||||||
|   name: "distributeHorizontally", |   name: "distributeHorizontally", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -49,7 +50,8 @@ export const distributeHorizontally = register({ | |||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   keyTest: (event) => event.altKey && event.code === CODES.H, |   keyTest: (event) => | ||||||
|  |     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.H, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
| @@ -67,6 +69,7 @@ export const distributeHorizontally = register({ | |||||||
|  |  | ||||||
| export const distributeVertically = register({ | export const distributeVertically = register({ | ||||||
|   name: "distributeVertically", |   name: "distributeVertically", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
| @@ -77,7 +80,8 @@ export const distributeVertically = register({ | |||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   keyTest: (event) => event.altKey && event.code === CODES.V, |   keyTest: (event) => | ||||||
|  |     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.V, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ import { isBoundToContainer } from "../element/typeChecks"; | |||||||
|  |  | ||||||
| export const actionDuplicateSelection = register({ | export const actionDuplicateSelection = register({ | ||||||
|   name: "duplicateSelection", |   name: "duplicateSelection", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     // duplicate selected point(s) if editing a line |     // duplicate selected point(s) if editing a line | ||||||
|     if (appState.editingLinearElement) { |     if (appState.editingLinearElement) { | ||||||
| @@ -127,12 +128,15 @@ const duplicateElements = ( | |||||||
|       { |       { | ||||||
|         ...appState, |         ...appState, | ||||||
|         selectedGroupIds: {}, |         selectedGroupIds: {}, | ||||||
|         selectedElementIds: newElements.reduce((acc, element) => { |         selectedElementIds: newElements.reduce( | ||||||
|  |           (acc: Record<ExcalidrawElement["id"], true>, element) => { | ||||||
|             if (!isBoundToContainer(element)) { |             if (!isBoundToContainer(element)) { | ||||||
|               acc[element.id] = true; |               acc[element.id] = true; | ||||||
|             } |             } | ||||||
|             return acc; |             return acc; | ||||||
|         }, {} as any), |           }, | ||||||
|  |           {}, | ||||||
|  |         ), | ||||||
|       }, |       }, | ||||||
|       getNonDeletedElements(finalElements), |       getNonDeletedElements(finalElements), | ||||||
|     ), |     ), | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import { trackEvent } from "../analytics"; |  | ||||||
| import { load, questionCircle, saveAs } from "../components/icons"; | import { load, questionCircle, saveAs } from "../components/icons"; | ||||||
| import { ProjectName } from "../components/ProjectName"; | import { ProjectName } from "../components/ProjectName"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| @@ -8,7 +7,7 @@ import { DarkModeToggle } from "../components/DarkModeToggle"; | |||||||
| import { loadFromJSON, saveAsJSON } from "../data"; | import { loadFromJSON, saveAsJSON } from "../data"; | ||||||
| import { resaveAsImageWithScene } from "../data/resave"; | import { resaveAsImageWithScene } from "../data/resave"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "../components/App"; | import { useDevice } from "../components/App"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { CheckboxItem } from "../components/CheckboxItem"; | import { CheckboxItem } from "../components/CheckboxItem"; | ||||||
| @@ -23,8 +22,8 @@ import { Theme } from "../element/types"; | |||||||
|  |  | ||||||
| export const actionChangeProjectName = register({ | export const actionChangeProjectName = register({ | ||||||
|   name: "changeProjectName", |   name: "changeProjectName", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     trackEvent("change", "title"); |  | ||||||
|     return { appState: { ...appState, name: value }, commitToHistory: false }; |     return { appState: { ...appState, name: value }, commitToHistory: false }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData, appProps }) => ( |   PanelComponent: ({ appState, updateData, appProps }) => ( | ||||||
| @@ -41,6 +40,7 @@ export const actionChangeProjectName = register({ | |||||||
|  |  | ||||||
| export const actionChangeExportScale = register({ | export const actionChangeExportScale = register({ | ||||||
|   name: "changeExportScale", |   name: "changeExportScale", | ||||||
|  |   trackEvent: { category: "export", action: "scale" }, | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       appState: { ...appState, exportScale: value }, |       appState: { ...appState, exportScale: value }, | ||||||
| @@ -89,6 +89,7 @@ export const actionChangeExportScale = register({ | |||||||
|  |  | ||||||
| export const actionChangeExportBackground = register({ | export const actionChangeExportBackground = register({ | ||||||
|   name: "changeExportBackground", |   name: "changeExportBackground", | ||||||
|  |   trackEvent: { category: "export", action: "toggleBackground" }, | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       appState: { ...appState, exportBackground: value }, |       appState: { ...appState, exportBackground: value }, | ||||||
| @@ -107,6 +108,7 @@ export const actionChangeExportBackground = register({ | |||||||
|  |  | ||||||
| export const actionChangeExportEmbedScene = register({ | export const actionChangeExportEmbedScene = register({ | ||||||
|   name: "changeExportEmbedScene", |   name: "changeExportEmbedScene", | ||||||
|  |   trackEvent: { category: "export", action: "embedScene" }, | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       appState: { ...appState, exportEmbedScene: value }, |       appState: { ...appState, exportEmbedScene: value }, | ||||||
| @@ -128,6 +130,7 @@ export const actionChangeExportEmbedScene = register({ | |||||||
|  |  | ||||||
| export const actionSaveToActiveFile = register({ | export const actionSaveToActiveFile = register({ | ||||||
|   name: "saveToActiveFile", |   name: "saveToActiveFile", | ||||||
|  |   trackEvent: { category: "export" }, | ||||||
|   perform: async (elements, appState, value, app) => { |   perform: async (elements, appState, value, app) => { | ||||||
|     const fileHandleExists = !!appState.fileHandle; |     const fileHandleExists = !!appState.fileHandle; | ||||||
|  |  | ||||||
| @@ -141,13 +144,15 @@ export const actionSaveToActiveFile = register({ | |||||||
|         appState: { |         appState: { | ||||||
|           ...appState, |           ...appState, | ||||||
|           fileHandle, |           fileHandle, | ||||||
|           toastMessage: fileHandleExists |           toast: fileHandleExists | ||||||
|             ? fileHandle?.name |             ? { | ||||||
|  |                 message: fileHandle?.name | ||||||
|                   ? t("toast.fileSavedToFilename").replace( |                   ? t("toast.fileSavedToFilename").replace( | ||||||
|                       "{filename}", |                       "{filename}", | ||||||
|                       `"${fileHandle.name}"`, |                       `"${fileHandle.name}"`, | ||||||
|                     ) |                     ) | ||||||
|               : t("toast.fileSaved") |                   : t("toast.fileSaved"), | ||||||
|  |               } | ||||||
|             : null, |             : null, | ||||||
|         }, |         }, | ||||||
|       }; |       }; | ||||||
| @@ -172,6 +177,7 @@ export const actionSaveToActiveFile = register({ | |||||||
|  |  | ||||||
| export const actionSaveFileToDisk = register({ | export const actionSaveFileToDisk = register({ | ||||||
|   name: "saveFileToDisk", |   name: "saveFileToDisk", | ||||||
|  |   trackEvent: { category: "export" }, | ||||||
|   perform: async (elements, appState, value, app) => { |   perform: async (elements, appState, value, app) => { | ||||||
|     try { |     try { | ||||||
|       const { fileHandle } = await saveAsJSON( |       const { fileHandle } = await saveAsJSON( | ||||||
| @@ -200,7 +206,7 @@ export const actionSaveFileToDisk = register({ | |||||||
|       icon={saveAs} |       icon={saveAs} | ||||||
|       title={t("buttons.saveAs")} |       title={t("buttons.saveAs")} | ||||||
|       aria-label={t("buttons.saveAs")} |       aria-label={t("buttons.saveAs")} | ||||||
|       showAriaLabel={useIsMobile()} |       showAriaLabel={useDevice().isMobile} | ||||||
|       hidden={!nativeFileSystemSupported} |       hidden={!nativeFileSystemSupported} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       data-testid="save-as-button" |       data-testid="save-as-button" | ||||||
| @@ -210,6 +216,7 @@ export const actionSaveFileToDisk = register({ | |||||||
|  |  | ||||||
| export const actionLoadScene = register({ | export const actionLoadScene = register({ | ||||||
|   name: "loadScene", |   name: "loadScene", | ||||||
|  |   trackEvent: { category: "export" }, | ||||||
|   perform: async (elements, appState, _, app) => { |   perform: async (elements, appState, _, app) => { | ||||||
|     try { |     try { | ||||||
|       const { |       const { | ||||||
| @@ -243,7 +250,7 @@ export const actionLoadScene = register({ | |||||||
|       icon={load} |       icon={load} | ||||||
|       title={t("buttons.load")} |       title={t("buttons.load")} | ||||||
|       aria-label={t("buttons.load")} |       aria-label={t("buttons.load")} | ||||||
|       showAriaLabel={useIsMobile()} |       showAriaLabel={useDevice().isMobile} | ||||||
|       onClick={updateData} |       onClick={updateData} | ||||||
|       data-testid="load-button" |       data-testid="load-button" | ||||||
|     /> |     /> | ||||||
| @@ -252,6 +259,7 @@ export const actionLoadScene = register({ | |||||||
|  |  | ||||||
| export const actionExportWithDarkMode = register({ | export const actionExportWithDarkMode = register({ | ||||||
|   name: "exportWithDarkMode", |   name: "exportWithDarkMode", | ||||||
|  |   trackEvent: { category: "export", action: "toggleTheme" }, | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       appState: { ...appState, exportWithDarkMode: value }, |       appState: { ...appState, exportWithDarkMode: value }, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { isInvisiblySmallElement } from "../element"; | import { isInvisiblySmallElement } from "../element"; | ||||||
| import { resetCursor } from "../utils"; | import { updateActiveTool, resetCursor } from "../utils"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| import { done } from "../components/icons"; | import { done } from "../components/icons"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| @@ -14,10 +14,12 @@ import { | |||||||
|   bindOrUnbindLinearElement, |   bindOrUnbindLinearElement, | ||||||
| } from "../element/binding"; | } from "../element/binding"; | ||||||
| import { isBindingElement } from "../element/typeChecks"; | import { isBindingElement } from "../element/typeChecks"; | ||||||
|  | import { AppState } from "../types"; | ||||||
|  |  | ||||||
| export const actionFinalize = register({ | export const actionFinalize = register({ | ||||||
|   name: "finalize", |   name: "finalize", | ||||||
|   perform: (elements, appState, _, { canvas, focusContainer }) => { |   trackEvent: false, | ||||||
|  |   perform: (elements, appState, _, { canvas, focusContainer, scene }) => { | ||||||
|     if (appState.editingLinearElement) { |     if (appState.editingLinearElement) { | ||||||
|       const { elementId, startBindingElement, endBindingElement } = |       const { elementId, startBindingElement, endBindingElement } = | ||||||
|         appState.editingLinearElement; |         appState.editingLinearElement; | ||||||
| @@ -38,6 +40,7 @@ export const actionFinalize = register({ | |||||||
|               : undefined, |               : undefined, | ||||||
|           appState: { |           appState: { | ||||||
|             ...appState, |             ...appState, | ||||||
|  |             cursorButton: "up", | ||||||
|             editingLinearElement: null, |             editingLinearElement: null, | ||||||
|           }, |           }, | ||||||
|           commitToHistory: true, |           commitToHistory: true, | ||||||
| @@ -47,8 +50,12 @@ export const actionFinalize = register({ | |||||||
|  |  | ||||||
|     let newElements = elements; |     let newElements = elements; | ||||||
|  |  | ||||||
|     if (appState.pendingImageElement) { |     const pendingImageElement = | ||||||
|       mutateElement(appState.pendingImageElement, { isDeleted: true }, false); |       appState.pendingImageElementId && | ||||||
|  |       scene.getElement(appState.pendingImageElementId); | ||||||
|  |  | ||||||
|  |     if (pendingImageElement) { | ||||||
|  |       mutateElement(pendingImageElement, { isDeleted: true }, false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (window.document.activeElement instanceof HTMLElement) { |     if (window.document.activeElement instanceof HTMLElement) { | ||||||
| @@ -119,27 +126,47 @@ export const actionFinalize = register({ | |||||||
|         ); |         ); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (!appState.elementLocked && appState.elementType !== "freedraw") { |       if ( | ||||||
|  |         !appState.activeTool.locked && | ||||||
|  |         appState.activeTool.type !== "freedraw" | ||||||
|  |       ) { | ||||||
|         appState.selectedElementIds[multiPointElement.id] = true; |         appState.selectedElementIds[multiPointElement.id] = true; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if ( |     if ( | ||||||
|       (!appState.elementLocked && appState.elementType !== "freedraw") || |       (!appState.activeTool.locked && | ||||||
|  |         appState.activeTool.type !== "freedraw") || | ||||||
|       !multiPointElement |       !multiPointElement | ||||||
|     ) { |     ) { | ||||||
|       resetCursor(canvas); |       resetCursor(canvas); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     let activeTool: AppState["activeTool"]; | ||||||
|  |     if (appState.activeTool.type === "eraser") { | ||||||
|  |       activeTool = updateActiveTool(appState, { | ||||||
|  |         ...(appState.activeTool.lastActiveToolBeforeEraser || { | ||||||
|  |           type: "selection", | ||||||
|  |         }), | ||||||
|  |         lastActiveToolBeforeEraser: null, | ||||||
|  |       }); | ||||||
|  |     } else { | ||||||
|  |       activeTool = updateActiveTool(appState, { | ||||||
|  |         type: "selection", | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|       elements: newElements, |       elements: newElements, | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|         elementType: |         cursorButton: "up", | ||||||
|           (appState.elementLocked || appState.elementType === "freedraw") && |         activeTool: | ||||||
|  |           (appState.activeTool.locked || | ||||||
|  |             appState.activeTool.type === "freedraw") && | ||||||
|           multiPointElement |           multiPointElement | ||||||
|             ? appState.elementType |             ? appState.activeTool | ||||||
|             : "selection", |             : activeTool, | ||||||
|         draggingElement: null, |         draggingElement: null, | ||||||
|         multiElement: null, |         multiElement: null, | ||||||
|         editingElement: null, |         editingElement: null, | ||||||
| @@ -147,16 +174,16 @@ export const actionFinalize = register({ | |||||||
|         suggestedBindings: [], |         suggestedBindings: [], | ||||||
|         selectedElementIds: |         selectedElementIds: | ||||||
|           multiPointElement && |           multiPointElement && | ||||||
|           !appState.elementLocked && |           !appState.activeTool.locked && | ||||||
|           appState.elementType !== "freedraw" |           appState.activeTool.type !== "freedraw" | ||||||
|             ? { |             ? { | ||||||
|                 ...appState.selectedElementIds, |                 ...appState.selectedElementIds, | ||||||
|                 [multiPointElement.id]: true, |                 [multiPointElement.id]: true, | ||||||
|               } |               } | ||||||
|             : appState.selectedElementIds, |             : appState.selectedElementIds, | ||||||
|         pendingImageElement: null, |         pendingImageElementId: null, | ||||||
|       }, |       }, | ||||||
|       commitToHistory: appState.elementType === "freedraw", |       commitToHistory: appState.activeTool.type === "freedraw", | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   keyTest: (event, appState) => |   keyTest: (event, appState) => | ||||||
| @@ -165,7 +192,7 @@ export const actionFinalize = register({ | |||||||
|         (!appState.draggingElement && appState.multiElement === null))) || |         (!appState.draggingElement && appState.multiElement === null))) || | ||||||
|     ((event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) && |     ((event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) && | ||||||
|       appState.multiElement !== null), |       appState.multiElement !== null), | ||||||
|   PanelComponent: ({ appState, updateData }) => ( |   PanelComponent: ({ appState, updateData, data }) => ( | ||||||
|     <ToolButton |     <ToolButton | ||||||
|       type="button" |       type="button" | ||||||
|       icon={done} |       icon={done} | ||||||
| @@ -173,6 +200,7 @@ export const actionFinalize = register({ | |||||||
|       aria-label={t("buttons.done")} |       aria-label={t("buttons.done")} | ||||||
|       onClick={updateData} |       onClick={updateData} | ||||||
|       visible={appState.multiElement != null} |       visible={appState.multiElement != null} | ||||||
|  |       size={data?.size || "medium"} | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -35,6 +35,7 @@ const enableActionFlipVertical = ( | |||||||
|  |  | ||||||
| export const actionFlipHorizontal = register({ | export const actionFlipHorizontal = register({ | ||||||
|   name: "flipHorizontal", |   name: "flipHorizontal", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       elements: flipSelectedElements(elements, appState, "horizontal"), |       elements: flipSelectedElements(elements, appState, "horizontal"), | ||||||
| @@ -50,6 +51,7 @@ export const actionFlipHorizontal = register({ | |||||||
|  |  | ||||||
| export const actionFlipVertical = register({ | export const actionFlipVertical = register({ | ||||||
|   name: "flipVertical", |   name: "flipVertical", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       elements: flipSelectedElements(elements, appState, "vertical"), |       elements: flipSelectedElements(elements, appState, "vertical"), | ||||||
| @@ -155,7 +157,7 @@ const flipElement = ( | |||||||
|     // calculate new x-coord for transformation |     // calculate new x-coord for transformation | ||||||
|     newNCoordsX = usingNWHandle ? element.x + 2 * width : element.x - 2 * width; |     newNCoordsX = usingNWHandle ? element.x + 2 * width : element.x - 2 * width; | ||||||
|     resizeSingleElement( |     resizeSingleElement( | ||||||
|       element, |       new Map().set(element.id, element), | ||||||
|       true, |       true, | ||||||
|       element, |       element, | ||||||
|       usingNWHandle ? "nw" : "ne", |       usingNWHandle ? "nw" : "ne", | ||||||
|   | |||||||
| @@ -54,6 +54,7 @@ const enableActionGroup = ( | |||||||
|  |  | ||||||
| export const actionGroup = register({ | export const actionGroup = register({ | ||||||
|   name: "group", |   name: "group", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     const selectedElements = getSelectedElements( |     const selectedElements = getSelectedElements( | ||||||
|       getNonDeletedElements(elements), |       getNonDeletedElements(elements), | ||||||
| @@ -147,6 +148,7 @@ export const actionGroup = register({ | |||||||
|  |  | ||||||
| export const actionUngroup = register({ | export const actionUngroup = register({ | ||||||
|   name: "ungroup", |   name: "ungroup", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     const groupIds = getSelectedGroupIds(appState); |     const groupIds = getSelectedGroupIds(appState); | ||||||
|     if (groupIds.length === 0) { |     if (groupIds.length === 0) { | ||||||
|   | |||||||
| @@ -62,6 +62,7 @@ type ActionCreator = (history: History) => Action; | |||||||
|  |  | ||||||
| export const createUndoAction: ActionCreator = (history) => ({ | export const createUndoAction: ActionCreator = (history) => ({ | ||||||
|   name: "undo", |   name: "undo", | ||||||
|  |   trackEvent: { category: "history" }, | ||||||
|   perform: (elements, appState) => |   perform: (elements, appState) => | ||||||
|     writeData(elements, appState, () => history.undoOnce()), |     writeData(elements, appState, () => history.undoOnce()), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
| @@ -82,6 +83,7 @@ export const createUndoAction: ActionCreator = (history) => ({ | |||||||
|  |  | ||||||
| export const createRedoAction: ActionCreator = (history) => ({ | export const createRedoAction: ActionCreator = (history) => ({ | ||||||
|   name: "redo", |   name: "redo", | ||||||
|  |   trackEvent: { category: "history" }, | ||||||
|   perform: (elements, appState) => |   perform: (elements, appState) => | ||||||
|     writeData(elements, appState, () => history.redoOnce()), |     writeData(elements, appState, () => history.redoOnce()), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import { HelpIcon } from "../components/HelpIcon"; | |||||||
|  |  | ||||||
| export const actionToggleCanvasMenu = register({ | export const actionToggleCanvasMenu = register({ | ||||||
|   name: "toggleCanvasMenu", |   name: "toggleCanvasMenu", | ||||||
|  |   trackEvent: { category: "menu" }, | ||||||
|   perform: (_, appState) => ({ |   perform: (_, appState) => ({ | ||||||
|     appState: { |     appState: { | ||||||
|       ...appState, |       ...appState, | ||||||
| @@ -29,6 +30,7 @@ export const actionToggleCanvasMenu = register({ | |||||||
|  |  | ||||||
| export const actionToggleEditMenu = register({ | export const actionToggleEditMenu = register({ | ||||||
|   name: "toggleEditMenu", |   name: "toggleEditMenu", | ||||||
|  |   trackEvent: { category: "menu" }, | ||||||
|   perform: (_elements, appState) => ({ |   perform: (_elements, appState) => ({ | ||||||
|     appState: { |     appState: { | ||||||
|       ...appState, |       ...appState, | ||||||
| @@ -53,6 +55,7 @@ export const actionToggleEditMenu = register({ | |||||||
|  |  | ||||||
| export const actionFullScreen = register({ | export const actionFullScreen = register({ | ||||||
|   name: "toggleFullScreen", |   name: "toggleFullScreen", | ||||||
|  |   trackEvent: { category: "canvas", predicate: (appState) => !isFullScreen() }, | ||||||
|   perform: () => { |   perform: () => { | ||||||
|     if (!isFullScreen()) { |     if (!isFullScreen()) { | ||||||
|       allowFullScreen(); |       allowFullScreen(); | ||||||
| @@ -69,6 +72,7 @@ export const actionFullScreen = register({ | |||||||
|  |  | ||||||
| export const actionShortcuts = register({ | export const actionShortcuts = register({ | ||||||
|   name: "toggleShortcuts", |   name: "toggleShortcuts", | ||||||
|  |   trackEvent: { category: "menu", action: "toggleHelpDialog" }, | ||||||
|   perform: (_elements, appState, _, { focusContainer }) => { |   perform: (_elements, appState, _, { focusContainer }) => { | ||||||
|     if (appState.showHelpDialog) { |     if (appState.showHelpDialog) { | ||||||
|       focusContainer(); |       focusContainer(); | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { getClientColors, getClientInitials } from "../clients"; | import { getClientColors } from "../clients"; | ||||||
| import { Avatar } from "../components/Avatar"; | import { Avatar } from "../components/Avatar"; | ||||||
| import { centerScrollOn } from "../scene/scroll"; | import { centerScrollOn } from "../scene/scroll"; | ||||||
| import { Collaborator } from "../types"; | import { Collaborator } from "../types"; | ||||||
| @@ -6,6 +6,7 @@ import { register } from "./register"; | |||||||
|  |  | ||||||
| export const actionGoToCollaborator = register({ | export const actionGoToCollaborator = register({ | ||||||
|   name: "goToCollaborator", |   name: "goToCollaborator", | ||||||
|  |   trackEvent: { category: "collab" }, | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     const point = value as Collaborator["pointer"]; |     const point = value as Collaborator["pointer"]; | ||||||
|     if (!point) { |     if (!point) { | ||||||
| @@ -30,28 +31,18 @@ export const actionGoToCollaborator = register({ | |||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData, data }) => { |   PanelComponent: ({ appState, updateData, data }) => { | ||||||
|     const clientId: string | undefined = data?.id; |     const [clientId, collaborator] = data as [string, Collaborator]; | ||||||
|     if (!clientId) { |  | ||||||
|       return null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const collaborator = appState.collaborators.get(clientId); |  | ||||||
|  |  | ||||||
|     if (!collaborator) { |  | ||||||
|       return null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const { background, stroke } = getClientColors(clientId, appState); |     const { background, stroke } = getClientColors(clientId, appState); | ||||||
|     const shortName = getClientInitials(collaborator.username); |  | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|       <Avatar |       <Avatar | ||||||
|         color={background} |         color={background} | ||||||
|         border={stroke} |         border={stroke} | ||||||
|         onClick={() => updateData(collaborator.pointer)} |         onClick={() => updateData(collaborator.pointer)} | ||||||
|       > |         name={collaborator.username || ""} | ||||||
|         {shortName} |         src={collaborator.avatarUrl} | ||||||
|       </Avatar> |       /> | ||||||
|     ); |     ); | ||||||
|   }, |   }, | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -30,11 +30,15 @@ import { | |||||||
|   TextAlignCenterIcon, |   TextAlignCenterIcon, | ||||||
|   TextAlignLeftIcon, |   TextAlignLeftIcon, | ||||||
|   TextAlignRightIcon, |   TextAlignRightIcon, | ||||||
|  |   TextAlignTopIcon, | ||||||
|  |   TextAlignBottomIcon, | ||||||
|  |   TextAlignMiddleIcon, | ||||||
| } from "../components/icons"; | } from "../components/icons"; | ||||||
| import { | import { | ||||||
|   DEFAULT_FONT_FAMILY, |   DEFAULT_FONT_FAMILY, | ||||||
|   DEFAULT_FONT_SIZE, |   DEFAULT_FONT_SIZE, | ||||||
|   FONT_FAMILY, |   FONT_FAMILY, | ||||||
|  |   VERTICAL_ALIGN, | ||||||
| } from "../constants"; | } from "../constants"; | ||||||
| import { | import { | ||||||
|   getNonDeletedElements, |   getNonDeletedElements, | ||||||
| @@ -58,6 +62,7 @@ import { | |||||||
|   ExcalidrawTextElement, |   ExcalidrawTextElement, | ||||||
|   FontFamilyValues, |   FontFamilyValues, | ||||||
|   TextAlign, |   TextAlign, | ||||||
|  |   VerticalAlign, | ||||||
| } from "../element/types"; | } from "../element/types"; | ||||||
| import { getLanguage, t } from "../i18n"; | import { getLanguage, t } from "../i18n"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| @@ -161,11 +166,7 @@ const changeFontSize = ( | |||||||
|           let newElement: ExcalidrawTextElement = newElementWith(oldElement, { |           let newElement: ExcalidrawTextElement = newElementWith(oldElement, { | ||||||
|             fontSize: newFontSize, |             fontSize: newFontSize, | ||||||
|           }); |           }); | ||||||
|           redrawTextBoundingBox( |           redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||||
|             newElement, |  | ||||||
|             getContainerElement(oldElement), |  | ||||||
|             appState, |  | ||||||
|           ); |  | ||||||
|  |  | ||||||
|           newElement = offsetElementAfterFontResize(oldElement, newElement); |           newElement = offsetElementAfterFontResize(oldElement, newElement); | ||||||
|  |  | ||||||
| @@ -193,6 +194,7 @@ const changeFontSize = ( | |||||||
|  |  | ||||||
| export const actionChangeStrokeColor = register({ | export const actionChangeStrokeColor = register({ | ||||||
|   name: "changeStrokeColor", |   name: "changeStrokeColor", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       ...(value.currentItemStrokeColor && { |       ...(value.currentItemStrokeColor && { | ||||||
| @@ -233,6 +235,8 @@ export const actionChangeStrokeColor = register({ | |||||||
|         setActive={(active) => |         setActive={(active) => | ||||||
|           updateData({ openPopup: active ? "strokeColorPicker" : null }) |           updateData({ openPopup: active ? "strokeColorPicker" : null }) | ||||||
|         } |         } | ||||||
|  |         elements={elements} | ||||||
|  |         appState={appState} | ||||||
|       /> |       /> | ||||||
|     </> |     </> | ||||||
|   ), |   ), | ||||||
| @@ -240,6 +244,7 @@ export const actionChangeStrokeColor = register({ | |||||||
|  |  | ||||||
| export const actionChangeBackgroundColor = register({ | export const actionChangeBackgroundColor = register({ | ||||||
|   name: "changeBackgroundColor", |   name: "changeBackgroundColor", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       ...(value.currentItemBackgroundColor && { |       ...(value.currentItemBackgroundColor && { | ||||||
| @@ -273,6 +278,8 @@ export const actionChangeBackgroundColor = register({ | |||||||
|         setActive={(active) => |         setActive={(active) => | ||||||
|           updateData({ openPopup: active ? "backgroundColorPicker" : null }) |           updateData({ openPopup: active ? "backgroundColorPicker" : null }) | ||||||
|         } |         } | ||||||
|  |         elements={elements} | ||||||
|  |         appState={appState} | ||||||
|       /> |       /> | ||||||
|     </> |     </> | ||||||
|   ), |   ), | ||||||
| @@ -280,6 +287,7 @@ export const actionChangeBackgroundColor = register({ | |||||||
|  |  | ||||||
| export const actionChangeFillStyle = register({ | export const actionChangeFillStyle = register({ | ||||||
|   name: "changeFillStyle", |   name: "changeFillStyle", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
| @@ -329,6 +337,7 @@ export const actionChangeFillStyle = register({ | |||||||
|  |  | ||||||
| export const actionChangeStrokeWidth = register({ | export const actionChangeStrokeWidth = register({ | ||||||
|   name: "changeStrokeWidth", |   name: "changeStrokeWidth", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
| @@ -376,6 +385,7 @@ export const actionChangeStrokeWidth = register({ | |||||||
|  |  | ||||||
| export const actionChangeSloppiness = register({ | export const actionChangeSloppiness = register({ | ||||||
|   name: "changeSloppiness", |   name: "changeSloppiness", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
| @@ -424,6 +434,7 @@ export const actionChangeSloppiness = register({ | |||||||
|  |  | ||||||
| export const actionChangeStrokeStyle = register({ | export const actionChangeStrokeStyle = register({ | ||||||
|   name: "changeStrokeStyle", |   name: "changeStrokeStyle", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
| @@ -471,12 +482,17 @@ export const actionChangeStrokeStyle = register({ | |||||||
|  |  | ||||||
| export const actionChangeOpacity = register({ | export const actionChangeOpacity = register({ | ||||||
|   name: "changeOpacity", |   name: "changeOpacity", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty( | ||||||
|  |         elements, | ||||||
|  |         appState, | ||||||
|  |         (el) => | ||||||
|           newElementWith(el, { |           newElementWith(el, { | ||||||
|             opacity: value, |             opacity: value, | ||||||
|           }), |           }), | ||||||
|  |         true, | ||||||
|       ), |       ), | ||||||
|       appState: { ...appState, currentItemOpacity: value }, |       appState: { ...appState, currentItemOpacity: value }, | ||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
| @@ -491,20 +507,6 @@ export const actionChangeOpacity = register({ | |||||||
|         max="100" |         max="100" | ||||||
|         step="10" |         step="10" | ||||||
|         onChange={(event) => updateData(+event.target.value)} |         onChange={(event) => updateData(+event.target.value)} | ||||||
|         onWheel={(event) => { |  | ||||||
|           event.stopPropagation(); |  | ||||||
|           const target = event.target as HTMLInputElement; |  | ||||||
|           const STEP = 10; |  | ||||||
|           const MAX = 100; |  | ||||||
|           const MIN = 0; |  | ||||||
|           const value = +target.value; |  | ||||||
|  |  | ||||||
|           if (event.deltaY < 0 && value < MAX) { |  | ||||||
|             updateData(value + STEP); |  | ||||||
|           } else if (event.deltaY > 0 && value > MIN) { |  | ||||||
|             updateData(value - STEP); |  | ||||||
|           } |  | ||||||
|         }} |  | ||||||
|         value={ |         value={ | ||||||
|           getFormValue( |           getFormValue( | ||||||
|             elements, |             elements, | ||||||
| @@ -520,6 +522,7 @@ export const actionChangeOpacity = register({ | |||||||
|  |  | ||||||
| export const actionChangeFontSize = register({ | export const actionChangeFontSize = register({ | ||||||
|   name: "changeFontSize", |   name: "changeFontSize", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return changeFontSize(elements, appState, () => value, value); |     return changeFontSize(elements, appState, () => value, value); | ||||||
|   }, |   }, | ||||||
| @@ -577,6 +580,7 @@ export const actionChangeFontSize = register({ | |||||||
|  |  | ||||||
| export const actionDecreaseFontSize = register({ | export const actionDecreaseFontSize = register({ | ||||||
|   name: "decreaseFontSize", |   name: "decreaseFontSize", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return changeFontSize(elements, appState, (element) => |     return changeFontSize(elements, appState, (element) => | ||||||
|       Math.round( |       Math.round( | ||||||
| @@ -598,6 +602,7 @@ export const actionDecreaseFontSize = register({ | |||||||
|  |  | ||||||
| export const actionIncreaseFontSize = register({ | export const actionIncreaseFontSize = register({ | ||||||
|   name: "increaseFontSize", |   name: "increaseFontSize", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return changeFontSize(elements, appState, (element) => |     return changeFontSize(elements, appState, (element) => | ||||||
|       Math.round(element.fontSize * (1 + FONT_SIZE_RELATIVE_INCREASE_STEP)), |       Math.round(element.fontSize * (1 + FONT_SIZE_RELATIVE_INCREASE_STEP)), | ||||||
| @@ -615,6 +620,7 @@ export const actionIncreaseFontSize = register({ | |||||||
|  |  | ||||||
| export const actionChangeFontFamily = register({ | export const actionChangeFontFamily = register({ | ||||||
|   name: "changeFontFamily", |   name: "changeFontFamily", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty( |       elements: changeProperty( | ||||||
| @@ -628,11 +634,7 @@ export const actionChangeFontFamily = register({ | |||||||
|                 fontFamily: value, |                 fontFamily: value, | ||||||
|               }, |               }, | ||||||
|             ); |             ); | ||||||
|             redrawTextBoundingBox( |             redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||||
|               newElement, |  | ||||||
|               getContainerElement(oldElement), |  | ||||||
|               appState, |  | ||||||
|             ); |  | ||||||
|             return newElement; |             return newElement; | ||||||
|           } |           } | ||||||
|  |  | ||||||
| @@ -700,6 +702,7 @@ export const actionChangeFontFamily = register({ | |||||||
|  |  | ||||||
| export const actionChangeTextAlign = register({ | export const actionChangeTextAlign = register({ | ||||||
|   name: "changeTextAlign", |   name: "changeTextAlign", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty( |       elements: changeProperty( | ||||||
| @@ -709,15 +712,9 @@ export const actionChangeTextAlign = register({ | |||||||
|           if (isTextElement(oldElement)) { |           if (isTextElement(oldElement)) { | ||||||
|             const newElement: ExcalidrawTextElement = newElementWith( |             const newElement: ExcalidrawTextElement = newElementWith( | ||||||
|               oldElement, |               oldElement, | ||||||
|               { |               { textAlign: value }, | ||||||
|                 textAlign: value, |  | ||||||
|               }, |  | ||||||
|             ); |  | ||||||
|             redrawTextBoundingBox( |  | ||||||
|               newElement, |  | ||||||
|               getContainerElement(oldElement), |  | ||||||
|               appState, |  | ||||||
|             ); |             ); | ||||||
|  |             redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||||
|             return newElement; |             return newElement; | ||||||
|           } |           } | ||||||
|  |  | ||||||
| @@ -732,7 +729,8 @@ export const actionChangeTextAlign = register({ | |||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => { | ||||||
|  |     return ( | ||||||
|       <fieldset> |       <fieldset> | ||||||
|         <legend>{t("labels.textAlign")}</legend> |         <legend>{t("labels.textAlign")}</legend> | ||||||
|         <ButtonIconSelect<TextAlign | false> |         <ButtonIconSelect<TextAlign | false> | ||||||
| @@ -772,11 +770,80 @@ export const actionChangeTextAlign = register({ | |||||||
|           onChange={(value) => updateData(value)} |           onChange={(value) => updateData(value)} | ||||||
|         /> |         /> | ||||||
|       </fieldset> |       </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> | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionChangeSharpness = register({ | export const actionChangeSharpness = register({ | ||||||
|   name: "changeSharpness", |   name: "changeSharpness", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|     const targetElements = getTargetElements( |     const targetElements = getTargetElements( | ||||||
|       getNonDeletedElements(elements), |       getNonDeletedElements(elements), | ||||||
| @@ -784,10 +851,10 @@ export const actionChangeSharpness = register({ | |||||||
|     ); |     ); | ||||||
|     const shouldUpdateForNonLinearElements = targetElements.length |     const shouldUpdateForNonLinearElements = targetElements.length | ||||||
|       ? targetElements.every((el) => !isLinearElement(el)) |       ? targetElements.every((el) => !isLinearElement(el)) | ||||||
|       : !isLinearElementType(appState.elementType); |       : !isLinearElementType(appState.activeTool.type); | ||||||
|     const shouldUpdateForLinearElements = targetElements.length |     const shouldUpdateForLinearElements = targetElements.length | ||||||
|       ? targetElements.every(isLinearElement) |       ? targetElements.every(isLinearElement) | ||||||
|       : isLinearElementType(appState.elementType); |       : isLinearElementType(appState.activeTool.type); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -827,8 +894,8 @@ export const actionChangeSharpness = register({ | |||||||
|           elements, |           elements, | ||||||
|           appState, |           appState, | ||||||
|           (element) => element.strokeSharpness, |           (element) => element.strokeSharpness, | ||||||
|           (canChangeSharpness(appState.elementType) && |           (canChangeSharpness(appState.activeTool.type) && | ||||||
|             (isLinearElementType(appState.elementType) |             (isLinearElementType(appState.activeTool.type) | ||||||
|               ? appState.currentItemLinearStrokeSharpness |               ? appState.currentItemLinearStrokeSharpness | ||||||
|               : appState.currentItemStrokeSharpness)) || |               : appState.currentItemStrokeSharpness)) || | ||||||
|             null, |             null, | ||||||
| @@ -841,6 +908,7 @@ export const actionChangeSharpness = register({ | |||||||
|  |  | ||||||
| export const actionChangeArrowhead = register({ | export const actionChangeArrowhead = register({ | ||||||
|   name: "changeArrowhead", |   name: "changeArrowhead", | ||||||
|  |   trackEvent: false, | ||||||
|   perform: ( |   perform: ( | ||||||
|     elements, |     elements, | ||||||
|     appState, |     appState, | ||||||
|   | |||||||
| @@ -2,9 +2,11 @@ import { KEYS } from "../keys"; | |||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { selectGroupsForSelectedElements } from "../groups"; | import { selectGroupsForSelectedElements } from "../groups"; | ||||||
| import { getNonDeletedElements, isTextElement } from "../element"; | import { getNonDeletedElements, isTextElement } from "../element"; | ||||||
|  | import { ExcalidrawElement } from "../element/types"; | ||||||
|  |  | ||||||
| export const actionSelectAll = register({ | export const actionSelectAll = register({ | ||||||
|   name: "selectAll", |   name: "selectAll", | ||||||
|  |   trackEvent: { category: "canvas" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     if (appState.editingLinearElement) { |     if (appState.editingLinearElement) { | ||||||
|       return false; |       return false; | ||||||
| @@ -14,15 +16,19 @@ export const actionSelectAll = register({ | |||||||
|         { |         { | ||||||
|           ...appState, |           ...appState, | ||||||
|           editingGroupId: null, |           editingGroupId: null, | ||||||
|           selectedElementIds: elements.reduce((map, element) => { |           selectedElementIds: elements.reduce( | ||||||
|  |             (map: Record<ExcalidrawElement["id"], true>, element) => { | ||||||
|               if ( |               if ( | ||||||
|                 !element.isDeleted && |                 !element.isDeleted && | ||||||
|               !(isTextElement(element) && element.containerId) |                 !(isTextElement(element) && element.containerId) && | ||||||
|  |                 !element.locked | ||||||
|               ) { |               ) { | ||||||
|                 map[element.id] = true; |                 map[element.id] = true; | ||||||
|               } |               } | ||||||
|               return map; |               return map; | ||||||
|           }, {} as any), |             }, | ||||||
|  |             {}, | ||||||
|  |           ), | ||||||
|         }, |         }, | ||||||
|         getNonDeletedElements(elements), |         getNonDeletedElements(elements), | ||||||
|       ), |       ), | ||||||
|   | |||||||
							
								
								
									
										71
									
								
								src/actions/actionStyles.test.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/actions/actionStyles.test.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | |||||||
|  | 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)[0]; | ||||||
|  |     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); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -6,28 +6,37 @@ import { | |||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { mutateElement, newElementWith } from "../element/mutateElement"; | import { newElementWith } from "../element/mutateElement"; | ||||||
| import { | import { | ||||||
|   DEFAULT_FONT_SIZE, |   DEFAULT_FONT_SIZE, | ||||||
|   DEFAULT_FONT_FAMILY, |   DEFAULT_FONT_FAMILY, | ||||||
|   DEFAULT_TEXT_ALIGN, |   DEFAULT_TEXT_ALIGN, | ||||||
| } from "../constants"; | } from "../constants"; | ||||||
| import { getContainerElement } from "../element/textElement"; | import { getBoundTextElement } from "../element/textElement"; | ||||||
|  | import { hasBoundTextElement } from "../element/typeChecks"; | ||||||
|  | import { getSelectedElements } from "../scene"; | ||||||
|  |  | ||||||
| // `copiedStyles` is exported only for tests. | // `copiedStyles` is exported only for tests. | ||||||
| export let copiedStyles: string = "{}"; | export let copiedStyles: string = "{}"; | ||||||
|  |  | ||||||
| export const actionCopyStyles = register({ | export const actionCopyStyles = register({ | ||||||
|   name: "copyStyles", |   name: "copyStyles", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     const elementsCopied = []; | ||||||
|     const element = elements.find((el) => appState.selectedElementIds[el.id]); |     const element = elements.find((el) => appState.selectedElementIds[el.id]); | ||||||
|  |     elementsCopied.push(element); | ||||||
|  |     if (element && hasBoundTextElement(element)) { | ||||||
|  |       const boundTextElement = getBoundTextElement(element); | ||||||
|  |       elementsCopied.push(boundTextElement); | ||||||
|  |     } | ||||||
|     if (element) { |     if (element) { | ||||||
|       copiedStyles = JSON.stringify(element); |       copiedStyles = JSON.stringify(elementsCopied); | ||||||
|     } |     } | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|         toastMessage: t("toast.copyStyles"), |         toast: { message: t("toast.copyStyles") }, | ||||||
|       }, |       }, | ||||||
|       commitToHistory: false, |       commitToHistory: false, | ||||||
|     }; |     }; | ||||||
| @@ -39,36 +48,64 @@ export const actionCopyStyles = register({ | |||||||
|  |  | ||||||
| export const actionPasteStyles = register({ | export const actionPasteStyles = register({ | ||||||
|   name: "pasteStyles", |   name: "pasteStyles", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     const pastedElement = JSON.parse(copiedStyles); |     const elementsCopied = JSON.parse(copiedStyles); | ||||||
|  |     const pastedElement = elementsCopied[0]; | ||||||
|  |     const boundTextElement = elementsCopied[1]; | ||||||
|     if (!isExcalidrawElement(pastedElement)) { |     if (!isExcalidrawElement(pastedElement)) { | ||||||
|       return { elements, commitToHistory: false }; |       return { elements, commitToHistory: false }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const selectedElements = getSelectedElements(elements, appState, true); | ||||||
|  |     const selectedElementIds = selectedElements.map((element) => element.id); | ||||||
|     return { |     return { | ||||||
|       elements: elements.map((element) => { |       elements: elements.map((element) => { | ||||||
|         if (appState.selectedElementIds[element.id]) { |         if (selectedElementIds.includes(element.id)) { | ||||||
|           const newElement = newElementWith(element, { |           let elementStylesToCopyFrom = pastedElement; | ||||||
|             backgroundColor: pastedElement?.backgroundColor, |           if (isTextElement(element) && element.containerId) { | ||||||
|             strokeWidth: pastedElement?.strokeWidth, |             elementStylesToCopyFrom = boundTextElement; | ||||||
|             strokeColor: pastedElement?.strokeColor, |           } | ||||||
|             strokeStyle: pastedElement?.strokeStyle, |           if (!elementStylesToCopyFrom) { | ||||||
|             fillStyle: pastedElement?.fillStyle, |             return element; | ||||||
|             opacity: pastedElement?.opacity, |           } | ||||||
|             roughness: pastedElement?.roughness, |           let newElement = newElementWith(element, { | ||||||
|           }); |             backgroundColor: elementStylesToCopyFrom?.backgroundColor, | ||||||
|           if (isTextElement(newElement) && isTextElement(element)) { |             strokeWidth: elementStylesToCopyFrom?.strokeWidth, | ||||||
|             mutateElement(newElement, { |             strokeColor: elementStylesToCopyFrom?.strokeColor, | ||||||
|               fontSize: pastedElement?.fontSize || DEFAULT_FONT_SIZE, |             strokeStyle: elementStylesToCopyFrom?.strokeStyle, | ||||||
|               fontFamily: pastedElement?.fontFamily || DEFAULT_FONT_FAMILY, |             fillStyle: elementStylesToCopyFrom?.fillStyle, | ||||||
|               textAlign: pastedElement?.textAlign || DEFAULT_TEXT_ALIGN, |             opacity: elementStylesToCopyFrom?.opacity, | ||||||
|  |             roughness: elementStylesToCopyFrom?.roughness, | ||||||
|           }); |           }); | ||||||
|  |  | ||||||
|             redrawTextBoundingBox( |           if (isTextElement(newElement)) { | ||||||
|               element, |             newElement = newElementWith(newElement, { | ||||||
|               getContainerElement(element), |               fontSize: elementStylesToCopyFrom?.fontSize || DEFAULT_FONT_SIZE, | ||||||
|               appState, |               fontFamily: | ||||||
|             ); |                 elementStylesToCopyFrom?.fontFamily || DEFAULT_FONT_FAMILY, | ||||||
|  |               textAlign: | ||||||
|  |                 elementStylesToCopyFrom?.textAlign || DEFAULT_TEXT_ALIGN, | ||||||
|  |             }); | ||||||
|  |             let container = null; | ||||||
|  |             if (newElement.containerId) { | ||||||
|  |               container = | ||||||
|  |                 selectedElements.find( | ||||||
|  |                   (element) => | ||||||
|  |                     isTextElement(newElement) && | ||||||
|  |                     element.id === newElement.containerId, | ||||||
|  |                 ) || null; | ||||||
|             } |             } | ||||||
|  |             redrawTextBoundingBox(newElement, container); | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           if (newElement.type === "arrow") { | ||||||
|  |             newElement = newElementWith(newElement, { | ||||||
|  |               startArrowhead: elementStylesToCopyFrom.startArrowhead, | ||||||
|  |               endArrowhead: elementStylesToCopyFrom.endArrowhead, | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |  | ||||||
|           return newElement; |           return newElement; | ||||||
|         } |         } | ||||||
|         return element; |         return element; | ||||||
|   | |||||||
| @@ -2,12 +2,14 @@ import { CODES, KEYS } from "../keys"; | |||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { GRID_SIZE } from "../constants"; | import { GRID_SIZE } from "../constants"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
| import { trackEvent } from "../analytics"; |  | ||||||
|  |  | ||||||
| export const actionToggleGridMode = register({ | export const actionToggleGridMode = register({ | ||||||
|   name: "gridMode", |   name: "gridMode", | ||||||
|  |   trackEvent: { | ||||||
|  |     category: "canvas", | ||||||
|  |     predicate: (appState) => !appState.gridSize, | ||||||
|  |   }, | ||||||
|   perform(elements, appState) { |   perform(elements, appState) { | ||||||
|     trackEvent("view", "mode", "grid"); |  | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|   | |||||||
							
								
								
									
										63
									
								
								src/actions/actionToggleLock.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								src/actions/actionToggleLock.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | |||||||
|  | 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,6 +3,7 @@ import { CODES, KEYS } from "../keys"; | |||||||
|  |  | ||||||
| export const actionToggleStats = register({ | export const actionToggleStats = register({ | ||||||
|   name: "stats", |   name: "stats", | ||||||
|  |   trackEvent: { category: "menu" }, | ||||||
|   perform(elements, appState) { |   perform(elements, appState) { | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|   | |||||||
| @@ -1,11 +1,13 @@ | |||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { trackEvent } from "../analytics"; |  | ||||||
|  |  | ||||||
| export const actionToggleViewMode = register({ | export const actionToggleViewMode = register({ | ||||||
|   name: "viewMode", |   name: "viewMode", | ||||||
|  |   trackEvent: { | ||||||
|  |     category: "canvas", | ||||||
|  |     predicate: (appState) => !appState.viewModeEnabled, | ||||||
|  |   }, | ||||||
|   perform(elements, appState) { |   perform(elements, appState) { | ||||||
|     trackEvent("view", "mode", "view"); |  | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|   | |||||||
| @@ -1,12 +1,13 @@ | |||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { trackEvent } from "../analytics"; |  | ||||||
|  |  | ||||||
| export const actionToggleZenMode = register({ | export const actionToggleZenMode = register({ | ||||||
|   name: "zenMode", |   name: "zenMode", | ||||||
|  |   trackEvent: { | ||||||
|  |     category: "canvas", | ||||||
|  |     predicate: (appState) => !appState.zenModeEnabled, | ||||||
|  |   }, | ||||||
|   perform(elements, appState) { |   perform(elements, appState) { | ||||||
|     trackEvent("view", "mode", "zen"); |  | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|   | |||||||
| @@ -1,44 +0,0 @@ | |||||||
| import { getNonDeletedElements } from "../element"; |  | ||||||
| import { mutateElement } from "../element/mutateElement"; |  | ||||||
| import { getBoundTextElement, measureText } from "../element/textElement"; |  | ||||||
| import { 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", |  | ||||||
|   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, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -18,6 +18,7 @@ import { | |||||||
|  |  | ||||||
| export const actionSendBackward = register({ | export const actionSendBackward = register({ | ||||||
|   name: "sendBackward", |   name: "sendBackward", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       elements: moveOneLeft(elements, appState), |       elements: moveOneLeft(elements, appState), | ||||||
| @@ -45,6 +46,7 @@ export const actionSendBackward = register({ | |||||||
|  |  | ||||||
| export const actionBringForward = register({ | export const actionBringForward = register({ | ||||||
|   name: "bringForward", |   name: "bringForward", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       elements: moveOneRight(elements, appState), |       elements: moveOneRight(elements, appState), | ||||||
| @@ -72,6 +74,7 @@ export const actionBringForward = register({ | |||||||
|  |  | ||||||
| export const actionSendToBack = register({ | export const actionSendToBack = register({ | ||||||
|   name: "sendToBack", |   name: "sendToBack", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       elements: moveAllLeft(elements, appState), |       elements: moveAllLeft(elements, appState), | ||||||
| @@ -106,6 +109,8 @@ export const actionSendToBack = register({ | |||||||
|  |  | ||||||
| export const actionBringToFront = register({ | export const actionBringToFront = register({ | ||||||
|   name: "bringToFront", |   name: "bringToFront", | ||||||
|  |   trackEvent: { category: "element" }, | ||||||
|  |  | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|     return { |     return { | ||||||
|       elements: moveAllRight(elements, appState), |       elements: moveAllRight(elements, appState), | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ export { | |||||||
|   actionChangeFontSize, |   actionChangeFontSize, | ||||||
|   actionChangeFontFamily, |   actionChangeFontFamily, | ||||||
|   actionChangeTextAlign, |   actionChangeTextAlign, | ||||||
|  |   actionChangeVerticalAlign, | ||||||
| } from "./actionProperties"; | } from "./actionProperties"; | ||||||
|  |  | ||||||
| export { | export { | ||||||
| @@ -74,11 +75,13 @@ export { | |||||||
|   actionCut, |   actionCut, | ||||||
|   actionCopyAsPng, |   actionCopyAsPng, | ||||||
|   actionCopyAsSvg, |   actionCopyAsSvg, | ||||||
|  |   copyText, | ||||||
| } from "./actionClipboard"; | } from "./actionClipboard"; | ||||||
|  |  | ||||||
| export { actionToggleGridMode } from "./actionToggleGridMode"; | export { actionToggleGridMode } from "./actionToggleGridMode"; | ||||||
| export { actionToggleZenMode } from "./actionToggleZenMode"; | export { actionToggleZenMode } from "./actionToggleZenMode"; | ||||||
|  |  | ||||||
| export { actionToggleStats } from "./actionToggleStats"; | export { actionToggleStats } from "./actionToggleStats"; | ||||||
| export { actionUnbindText } from "./actionUnbindText"; | export { actionUnbindText, actionBindText } from "./actionBoundText"; | ||||||
| export { actionLink } from "../element/Hyperlink"; | export { actionLink } from "../element/Hyperlink"; | ||||||
|  | export { actionToggleLock } from "./actionToggleLock"; | ||||||
|   | |||||||
| @@ -1,18 +1,47 @@ | |||||||
| import React from "react"; | import React from "react"; | ||||||
| import { | import { | ||||||
|   Action, |   Action, | ||||||
|   ActionsManagerInterface, |  | ||||||
|   UpdaterFn, |   UpdaterFn, | ||||||
|   ActionName, |   ActionName, | ||||||
|   ActionResult, |   ActionResult, | ||||||
|   PanelComponentProps, |   PanelComponentProps, | ||||||
|  |   ActionSource, | ||||||
| } from "./types"; | } from "./types"; | ||||||
| import { ExcalidrawElement } from "../element/types"; | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { AppClassProperties, AppState } from "../types"; | import { AppClassProperties, AppState } from "../types"; | ||||||
| import { MODES } from "../constants"; | import { MODES } from "../constants"; | ||||||
|  | import { trackEvent } from "../analytics"; | ||||||
|  |  | ||||||
| export class ActionManager implements ActionsManagerInterface { | const trackAction = ( | ||||||
|   actions = {} as ActionsManagerInterface["actions"]; |   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.device.isMobile ? "mobile" : "desktop"})`, | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       console.error("error while logging action:", error); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export class ActionManager { | ||||||
|  |   actions = {} as Record<ActionName, Action>; | ||||||
|  |  | ||||||
|   updater: (actionResult: ActionResult | Promise<ActionResult>) => void; |   updater: (actionResult: ActionResult | Promise<ActionResult>) => void; | ||||||
|  |  | ||||||
| @@ -65,9 +94,15 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|           ), |           ), | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
|     if (data.length === 0) { |     if (data.length !== 1) { | ||||||
|  |       if (data.length > 1) { | ||||||
|  |         console.warn("Canceling as multiple actions match this shortcut", data); | ||||||
|  |       } | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const action = data[0]; | ||||||
|  |  | ||||||
|     const { viewModeEnabled } = this.getAppState(); |     const { viewModeEnabled } = this.getAppState(); | ||||||
|     if (viewModeEnabled) { |     if (viewModeEnabled) { | ||||||
|       if (!Object.values(MODES).includes(data[0].name)) { |       if (!Object.values(MODES).includes(data[0].name)) { | ||||||
| @@ -75,27 +110,26 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const elements = this.getElementsIncludingDeleted(); | ||||||
|  |     const appState = this.getAppState(); | ||||||
|  |     const value = null; | ||||||
|  |  | ||||||
|  |     trackAction(action, "keyboard", appState, elements, this.app, null); | ||||||
|  |  | ||||||
|     event.preventDefault(); |     event.preventDefault(); | ||||||
|     this.updater( |     event.stopPropagation(); | ||||||
|       data[0].perform( |     this.updater(data[0].perform(elements, appState, value, this.app)); | ||||||
|         this.getElementsIncludingDeleted(), |  | ||||||
|         this.getAppState(), |  | ||||||
|         null, |  | ||||||
|         this.app, |  | ||||||
|       ), |  | ||||||
|     ); |  | ||||||
|     return true; |     return true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   executeAction(action: Action) { |   executeAction(action: Action, source: ActionSource = "api") { | ||||||
|     this.updater( |     const elements = this.getElementsIncludingDeleted(); | ||||||
|       action.perform( |     const appState = this.getAppState(); | ||||||
|         this.getElementsIncludingDeleted(), |     const value = null; | ||||||
|         this.getAppState(), |  | ||||||
|         null, |     trackAction(action, source, appState, elements, this.app, value); | ||||||
|         this.app, |  | ||||||
|       ), |     this.updater(action.perform(elements, appState, value, this.app)); | ||||||
|     ); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -113,7 +147,11 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|     ) { |     ) { | ||||||
|       const action = this.actions[name]; |       const action = this.actions[name]; | ||||||
|       const PanelComponent = action.PanelComponent!; |       const PanelComponent = action.PanelComponent!; | ||||||
|  |       const elements = this.getElementsIncludingDeleted(); | ||||||
|  |       const appState = this.getAppState(); | ||||||
|       const updateData = (formState?: any) => { |       const updateData = (formState?: any) => { | ||||||
|  |         trackAction(action, "ui", appState, elements, this.app, formState); | ||||||
|  |  | ||||||
|         this.updater( |         this.updater( | ||||||
|           action.perform( |           action.perform( | ||||||
|             this.getElementsIncludingDeleted(), |             this.getElementsIncludingDeleted(), | ||||||
|   | |||||||
| @@ -1,8 +1,10 @@ | |||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { isDarwin } from "../keys"; | import { isDarwin } from "../keys"; | ||||||
| import { getShortcutKey } from "../utils"; | import { getShortcutKey } from "../utils"; | ||||||
|  | import { ActionName } from "./types"; | ||||||
|  |  | ||||||
| export type ShortcutName = | export type ShortcutName = SubtypeOf< | ||||||
|  |   ActionName, | ||||||
|   | "cut" |   | "cut" | ||||||
|   | "copy" |   | "copy" | ||||||
|   | "paste" |   | "paste" | ||||||
| @@ -26,7 +28,9 @@ export type ShortcutName = | |||||||
|   | "viewMode" |   | "viewMode" | ||||||
|   | "flipHorizontal" |   | "flipHorizontal" | ||||||
|   | "flipVertical" |   | "flipVertical" | ||||||
|   | "link"; |   | "hyperlink" | ||||||
|  |   | "toggleLock" | ||||||
|  | >; | ||||||
|  |  | ||||||
| const shortcutMap: Record<ShortcutName, string[]> = { | const shortcutMap: Record<ShortcutName, string[]> = { | ||||||
|   cut: [getShortcutKey("CtrlOrCmd+X")], |   cut: [getShortcutKey("CtrlOrCmd+X")], | ||||||
| @@ -63,11 +67,12 @@ const shortcutMap: Record<ShortcutName, string[]> = { | |||||||
|   flipHorizontal: [getShortcutKey("Shift+H")], |   flipHorizontal: [getShortcutKey("Shift+H")], | ||||||
|   flipVertical: [getShortcutKey("Shift+V")], |   flipVertical: [getShortcutKey("Shift+V")], | ||||||
|   viewMode: [getShortcutKey("Alt+R")], |   viewMode: [getShortcutKey("Alt+R")], | ||||||
|   link: [getShortcutKey("CtrlOrCmd+K")], |   hyperlink: [getShortcutKey("CtrlOrCmd+K")], | ||||||
|  |   toggleLock: [getShortcutKey("CtrlOrCmd+Shift+L")], | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const getShortcutFromShortcutName = (name: ShortcutName) => { | export const getShortcutFromShortcutName = (name: ShortcutName) => { | ||||||
|   const shortcuts = shortcutMap[name]; |   const shortcuts = shortcutMap[name]; | ||||||
|   // if multiple shortcuts availiable, take the first one |   // if multiple shortcuts available, take the first one | ||||||
|   return shortcuts && shortcuts.length > 0 ? shortcuts[0] : ""; |   return shortcuts && shortcuts.length > 0 ? shortcuts[0] : ""; | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -6,7 +6,8 @@ import { | |||||||
|   ExcalidrawProps, |   ExcalidrawProps, | ||||||
|   BinaryFiles, |   BinaryFiles, | ||||||
| } from "../types"; | } from "../types"; | ||||||
| import { ToolButtonSize } from "../components/ToolButton"; |  | ||||||
|  | export type ActionSource = "ui" | "keyboard" | "contextMenu" | "api"; | ||||||
|  |  | ||||||
| /** if false, the action should be prevented */ | /** if false, the action should be prevented */ | ||||||
| export type ActionResult = | export type ActionResult = | ||||||
| @@ -39,6 +40,7 @@ export type ActionName = | |||||||
|   | "paste" |   | "paste" | ||||||
|   | "copyAsPng" |   | "copyAsPng" | ||||||
|   | "copyAsSvg" |   | "copyAsSvg" | ||||||
|  |   | "copyText" | ||||||
|   | "sendBackward" |   | "sendBackward" | ||||||
|   | "bringForward" |   | "bringForward" | ||||||
|   | "sendToBack" |   | "sendToBack" | ||||||
| @@ -82,6 +84,7 @@ export type ActionName = | |||||||
|   | "zoomToSelection" |   | "zoomToSelection" | ||||||
|   | "changeFontFamily" |   | "changeFontFamily" | ||||||
|   | "changeTextAlign" |   | "changeTextAlign" | ||||||
|  |   | "changeVerticalAlign" | ||||||
|   | "toggleFullScreen" |   | "toggleFullScreen" | ||||||
|   | "toggleShortcuts" |   | "toggleShortcuts" | ||||||
|   | "group" |   | "group" | ||||||
| @@ -105,14 +108,17 @@ export type ActionName = | |||||||
|   | "increaseFontSize" |   | "increaseFontSize" | ||||||
|   | "decreaseFontSize" |   | "decreaseFontSize" | ||||||
|   | "unbindText" |   | "unbindText" | ||||||
|   | "link"; |   | "hyperlink" | ||||||
|  |   | "eraser" | ||||||
|  |   | "bindText" | ||||||
|  |   | "toggleLock"; | ||||||
|  |  | ||||||
| export type PanelComponentProps = { | export type PanelComponentProps = { | ||||||
|   elements: readonly ExcalidrawElement[]; |   elements: readonly ExcalidrawElement[]; | ||||||
|   appState: AppState; |   appState: AppState; | ||||||
|   updateData: (formData?: any) => void; |   updateData: (formData?: any) => void; | ||||||
|   appProps: ExcalidrawProps; |   appProps: ExcalidrawProps; | ||||||
|   data?: Partial<{ id: string; size: ToolButtonSize }>; |   data?: Record<string, any>; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export interface Action { | export interface Action { | ||||||
| @@ -136,12 +142,23 @@ export interface Action { | |||||||
|     appState: AppState, |     appState: AppState, | ||||||
|   ) => boolean; |   ) => boolean; | ||||||
|   checked?: (appState: Readonly<AppState>) => boolean; |   checked?: (appState: Readonly<AppState>) => boolean; | ||||||
| } |   trackEvent: | ||||||
|  |     | false | ||||||
| export interface ActionsManagerInterface { |     | { | ||||||
|   actions: Record<ActionName, Action>; |         category: | ||||||
|   registerAction: (action: Action) => void; |           | "toolbar" | ||||||
|   handleKeyDown: (event: React.KeyboardEvent | KeyboardEvent) => boolean; |           | "element" | ||||||
|   renderAction: (name: ActionName) => React.ReactElement | null; |           | "canvas" | ||||||
|   executeAction: (action: Action) => void; |           | "export" | ||||||
|  |           | "history" | ||||||
|  |           | "menu" | ||||||
|  |           | "collab" | ||||||
|  |           | "hyperlink"; | ||||||
|  |         action?: string; | ||||||
|  |         predicate?: ( | ||||||
|  |           appState: Readonly<AppState>, | ||||||
|  |           elements: readonly ExcalidrawElement[], | ||||||
|  |           value: any, | ||||||
|  |         ) => boolean; | ||||||
|  |       }; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,16 +3,20 @@ export const trackEvent = | |||||||
|   process.env?.REACT_APP_GOOGLE_ANALYTICS_ID && |   process.env?.REACT_APP_GOOGLE_ANALYTICS_ID && | ||||||
|   typeof window !== "undefined" && |   typeof window !== "undefined" && | ||||||
|   window.gtag |   window.gtag | ||||||
|     ? (category: string, name: string, label?: string, value?: number) => { |     ? (category: string, action: string, label?: string, value?: number) => { | ||||||
|         window.gtag("event", name, { |         try { | ||||||
|  |           window.gtag("event", action, { | ||||||
|             event_category: category, |             event_category: category, | ||||||
|             event_label: label, |             event_label: label, | ||||||
|             value, |             value, | ||||||
|           }); |           }); | ||||||
|  |         } catch (error) { | ||||||
|  |           console.error("error logging to ga", error); | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|     : typeof process !== "undefined" && process.env?.JEST_WORKER_ID |     : typeof process !== "undefined" && process.env?.JEST_WORKER_ID | ||||||
|     ? (category: string, name: string, label?: string, value?: number) => {} |     ? (category: string, action: string, label?: string, value?: number) => {} | ||||||
|     : (category: string, name: string, label?: string, value?: number) => { |     : (category: string, action: string, label?: string, value?: number) => { | ||||||
|         // Uncomment the next line to track locally |         // Uncomment the next line to track locally | ||||||
|         // console.info("Track Event", category, name, label, value); |         // console.log("Track Event", { category, action, label, value }); | ||||||
|       }; |       }; | ||||||
|   | |||||||
| @@ -41,8 +41,12 @@ export const getDefaultAppState = (): Omit< | |||||||
|     editingElement: null, |     editingElement: null, | ||||||
|     editingGroupId: null, |     editingGroupId: null, | ||||||
|     editingLinearElement: null, |     editingLinearElement: null, | ||||||
|     elementLocked: false, |     activeTool: { | ||||||
|     elementType: "selection", |       type: "selection", | ||||||
|  |       customType: null, | ||||||
|  |       locked: false, | ||||||
|  |       lastActiveToolBeforeEraser: null, | ||||||
|  |     }, | ||||||
|     penMode: false, |     penMode: false, | ||||||
|     penDetected: false, |     penDetected: false, | ||||||
|     errorMessage: null, |     errorMessage: null, | ||||||
| @@ -54,6 +58,7 @@ export const getDefaultAppState = (): Omit< | |||||||
|     gridSize: null, |     gridSize: null, | ||||||
|     isBindingEnabled: true, |     isBindingEnabled: true, | ||||||
|     isLibraryOpen: false, |     isLibraryOpen: false, | ||||||
|  |     isLibraryMenuDocked: false, | ||||||
|     isLoading: false, |     isLoading: false, | ||||||
|     isResizing: false, |     isResizing: false, | ||||||
|     isRotating: false, |     isRotating: false, | ||||||
| @@ -76,14 +81,14 @@ export const getDefaultAppState = (): Omit< | |||||||
|     showStats: false, |     showStats: false, | ||||||
|     startBoundElement: null, |     startBoundElement: null, | ||||||
|     suggestedBindings: [], |     suggestedBindings: [], | ||||||
|     toastMessage: null, |     toast: null, | ||||||
|     viewBackgroundColor: oc.white, |     viewBackgroundColor: oc.white, | ||||||
|     zenModeEnabled: false, |     zenModeEnabled: false, | ||||||
|     zoom: { |     zoom: { | ||||||
|       value: 1 as NormalizedZoomValue, |       value: 1 as NormalizedZoomValue, | ||||||
|     }, |     }, | ||||||
|     viewModeEnabled: false, |     viewModeEnabled: false, | ||||||
|     pendingImageElement: null, |     pendingImageElementId: null, | ||||||
|     showHyperlinkPopup: false, |     showHyperlinkPopup: false, | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| @@ -130,10 +135,9 @@ const APP_STATE_STORAGE_CONF = (< | |||||||
|   editingElement: { browser: false, export: false, server: false }, |   editingElement: { browser: false, export: false, server: false }, | ||||||
|   editingGroupId: { browser: true, export: false, server: false }, |   editingGroupId: { browser: true, export: false, server: false }, | ||||||
|   editingLinearElement: { browser: false, export: false, server: false }, |   editingLinearElement: { browser: false, export: false, server: false }, | ||||||
|   elementLocked: { browser: true, export: false, server: false }, |   activeTool: { browser: true, export: false, server: false }, | ||||||
|   elementType: { browser: true, export: false, server: false }, |   penMode: { browser: true, export: false, server: false }, | ||||||
|   penMode: { browser: false, export: false, server: false }, |   penDetected: { browser: true, export: false, server: false }, | ||||||
|   penDetected: { browser: false, export: false, server: false }, |  | ||||||
|   errorMessage: { browser: false, export: false, server: false }, |   errorMessage: { browser: false, export: false, server: false }, | ||||||
|   exportBackground: { browser: true, export: false, server: false }, |   exportBackground: { browser: true, export: false, server: false }, | ||||||
|   exportEmbedScene: { browser: true, export: false, server: false }, |   exportEmbedScene: { browser: true, export: false, server: false }, | ||||||
| @@ -143,7 +147,8 @@ const APP_STATE_STORAGE_CONF = (< | |||||||
|   gridSize: { browser: true, export: true, server: true }, |   gridSize: { browser: true, export: true, server: true }, | ||||||
|   height: { browser: false, export: false, server: false }, |   height: { browser: false, export: false, server: false }, | ||||||
|   isBindingEnabled: { browser: false, export: false, server: false }, |   isBindingEnabled: { browser: false, export: false, server: false }, | ||||||
|   isLibraryOpen: { browser: false, export: false, server: false }, |   isLibraryOpen: { browser: true, export: false, server: false }, | ||||||
|  |   isLibraryMenuDocked: { browser: true, export: false, server: false }, | ||||||
|   isLoading: { browser: false, export: false, server: false }, |   isLoading: { browser: false, export: false, server: false }, | ||||||
|   isResizing: { browser: false, export: false, server: false }, |   isResizing: { browser: false, export: false, server: false }, | ||||||
|   isRotating: { browser: false, export: false, server: false }, |   isRotating: { browser: false, export: false, server: false }, | ||||||
| @@ -168,13 +173,13 @@ const APP_STATE_STORAGE_CONF = (< | |||||||
|   showStats: { browser: true, export: false, server: false }, |   showStats: { browser: true, export: false, server: false }, | ||||||
|   startBoundElement: { browser: false, export: false, server: false }, |   startBoundElement: { browser: false, export: false, server: false }, | ||||||
|   suggestedBindings: { browser: false, export: false, server: false }, |   suggestedBindings: { browser: false, export: false, server: false }, | ||||||
|   toastMessage: { browser: false, export: false, server: false }, |   toast: { browser: false, export: false, server: false }, | ||||||
|   viewBackgroundColor: { browser: true, export: true, server: true }, |   viewBackgroundColor: { browser: true, export: true, server: true }, | ||||||
|   width: { browser: false, export: false, server: false }, |   width: { browser: false, export: false, server: false }, | ||||||
|   zenModeEnabled: { browser: true, export: false, server: false }, |   zenModeEnabled: { browser: true, export: false, server: false }, | ||||||
|   zoom: { browser: true, export: false, server: false }, |   zoom: { browser: true, export: false, server: false }, | ||||||
|   viewModeEnabled: { browser: false, export: false, server: false }, |   viewModeEnabled: { browser: false, export: false, server: false }, | ||||||
|   pendingImageElement: { browser: false, export: false, server: false }, |   pendingImageElementId: { browser: false, export: false, server: false }, | ||||||
|   showHyperlinkPopup: { browser: false, export: false, server: false }, |   showHyperlinkPopup: { browser: false, export: false, server: false }, | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -213,3 +218,9 @@ export const cleanAppStateForExport = (appState: Partial<AppState>) => { | |||||||
| export const clearAppStateForDatabase = (appState: Partial<AppState>) => { | export const clearAppStateForDatabase = (appState: Partial<AppState>) => { | ||||||
|   return _clearAppStateForStorage(appState, "server"); |   return _clearAppStateForStorage(appState, "server"); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export const isEraserActive = ({ | ||||||
|  |   activeTool, | ||||||
|  | }: { | ||||||
|  |   activeTool: AppState["activeTool"]; | ||||||
|  | }) => activeTool.type === "eraser"; | ||||||
|   | |||||||
							
								
								
									
										121
									
								
								src/charts.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								src/charts.test.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | |||||||
|  | import { | ||||||
|  |   Spreadsheet, | ||||||
|  |   tryParseCells, | ||||||
|  |   tryParseNumber, | ||||||
|  |   VALID_SPREADSHEET, | ||||||
|  | } from "./charts"; | ||||||
|  |  | ||||||
|  | describe("charts", () => { | ||||||
|  |   describe("tryParseNumber", () => { | ||||||
|  |     it.each<[string, number]>([ | ||||||
|  |       ["1", 1], | ||||||
|  |       ["0", 0], | ||||||
|  |       ["-1", -1], | ||||||
|  |       ["0.1", 0.1], | ||||||
|  |       [".1", 0.1], | ||||||
|  |       ["1.", 1], | ||||||
|  |       ["424.", 424], | ||||||
|  |       ["$1", 1], | ||||||
|  |       ["-.1", -0.1], | ||||||
|  |       ["-$1", -1], | ||||||
|  |       ["$-1", -1], | ||||||
|  |     ])("should correctly identify %s as numbers", (given, expected) => { | ||||||
|  |       expect(tryParseNumber(given)).toEqual(expected); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it.each<[string]>([["a"], ["$"], ["$a"], ["-$a"]])( | ||||||
|  |       "should correctly identify %s as not a number", | ||||||
|  |       (given) => { | ||||||
|  |         expect(tryParseNumber(given)).toBeNull(); | ||||||
|  |       }, | ||||||
|  |     ); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   describe("tryParseCells", () => { | ||||||
|  |     it("Successfully parses a spreadsheet", () => { | ||||||
|  |       const spreadsheet = [ | ||||||
|  |         ["time", "value"], | ||||||
|  |         ["01:00", "61"], | ||||||
|  |         ["02:00", "-60"], | ||||||
|  |         ["03:00", "85"], | ||||||
|  |         ["04:00", "-67"], | ||||||
|  |         ["05:00", "54"], | ||||||
|  |         ["06:00", "95"], | ||||||
|  |       ]; | ||||||
|  |  | ||||||
|  |       const result = tryParseCells(spreadsheet); | ||||||
|  |  | ||||||
|  |       expect(result.type).toBe(VALID_SPREADSHEET); | ||||||
|  |  | ||||||
|  |       const { title, labels, values } = ( | ||||||
|  |         result as { type: typeof VALID_SPREADSHEET; spreadsheet: Spreadsheet } | ||||||
|  |       ).spreadsheet; | ||||||
|  |  | ||||||
|  |       expect(title).toEqual("value"); | ||||||
|  |       expect(labels).toEqual([ | ||||||
|  |         "01:00", | ||||||
|  |         "02:00", | ||||||
|  |         "03:00", | ||||||
|  |         "04:00", | ||||||
|  |         "05:00", | ||||||
|  |         "06:00", | ||||||
|  |       ]); | ||||||
|  |       expect(values).toEqual([61, -60, 85, -67, 54, 95]); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it("Uses the second column as the label if it is not a number", () => { | ||||||
|  |       const spreadsheet = [ | ||||||
|  |         ["time", "value"], | ||||||
|  |         ["01:00", "61"], | ||||||
|  |         ["02:00", "-60"], | ||||||
|  |         ["03:00", "85"], | ||||||
|  |         ["04:00", "-67"], | ||||||
|  |         ["05:00", "54"], | ||||||
|  |         ["06:00", "95"], | ||||||
|  |       ]; | ||||||
|  |  | ||||||
|  |       const result = tryParseCells(spreadsheet); | ||||||
|  |  | ||||||
|  |       expect(result.type).toBe(VALID_SPREADSHEET); | ||||||
|  |  | ||||||
|  |       const { title, labels, values } = ( | ||||||
|  |         result as { type: typeof VALID_SPREADSHEET; spreadsheet: Spreadsheet } | ||||||
|  |       ).spreadsheet; | ||||||
|  |  | ||||||
|  |       expect(title).toEqual("value"); | ||||||
|  |       expect(labels).toEqual([ | ||||||
|  |         "01:00", | ||||||
|  |         "02:00", | ||||||
|  |         "03:00", | ||||||
|  |         "04:00", | ||||||
|  |         "05:00", | ||||||
|  |         "06:00", | ||||||
|  |       ]); | ||||||
|  |       expect(values).toEqual([61, -60, 85, -67, 54, 95]); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it("treats the first column as labels if both columns are numbers", () => { | ||||||
|  |       const spreadsheet = [ | ||||||
|  |         ["time", "value"], | ||||||
|  |         ["01", "61"], | ||||||
|  |         ["02", "-60"], | ||||||
|  |         ["03", "85"], | ||||||
|  |         ["04", "-67"], | ||||||
|  |         ["05", "54"], | ||||||
|  |         ["06", "95"], | ||||||
|  |       ]; | ||||||
|  |  | ||||||
|  |       const result = tryParseCells(spreadsheet); | ||||||
|  |  | ||||||
|  |       expect(result.type).toBe(VALID_SPREADSHEET); | ||||||
|  |  | ||||||
|  |       const { title, labels, values } = ( | ||||||
|  |         result as { type: typeof VALID_SPREADSHEET; spreadsheet: Spreadsheet } | ||||||
|  |       ).spreadsheet; | ||||||
|  |  | ||||||
|  |       expect(title).toEqual("value"); | ||||||
|  |       expect(labels).toEqual(["01", "02", "03", "04", "05", "06"]); | ||||||
|  |       expect(values).toEqual([61, -60, 85, -67, 54, 95]); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -1,5 +1,10 @@ | |||||||
| import colors from "./colors"; | import colors from "./colors"; | ||||||
| import { DEFAULT_FONT_FAMILY, DEFAULT_FONT_SIZE, ENV } from "./constants"; | import { | ||||||
|  |   DEFAULT_FONT_FAMILY, | ||||||
|  |   DEFAULT_FONT_SIZE, | ||||||
|  |   ENV, | ||||||
|  |   VERTICAL_ALIGN, | ||||||
|  | } from "./constants"; | ||||||
| import { newElement, newLinearElement, newTextElement } from "./element"; | import { newElement, newLinearElement, newTextElement } from "./element"; | ||||||
| import { NonDeletedExcalidrawElement } from "./element/types"; | import { NonDeletedExcalidrawElement } from "./element/types"; | ||||||
| import { randomId } from "./random"; | import { randomId } from "./random"; | ||||||
| @@ -24,18 +29,24 @@ type ParseSpreadsheetResult = | |||||||
|   | { type: typeof NOT_SPREADSHEET; reason: string } |   | { type: typeof NOT_SPREADSHEET; reason: string } | ||||||
|   | { type: typeof VALID_SPREADSHEET; spreadsheet: Spreadsheet }; |   | { type: typeof VALID_SPREADSHEET; spreadsheet: Spreadsheet }; | ||||||
|  |  | ||||||
| const tryParseNumber = (s: string): number | null => { | /** | ||||||
|   const match = /^[$€£¥₩]?([0-9,]+(\.[0-9]+)?)$/.exec(s); |  * @private exported for testing | ||||||
|  |  */ | ||||||
|  | export const tryParseNumber = (s: string): number | null => { | ||||||
|  |   const match = /^([-+]?)[$€£¥₩]?([-+]?)([\d.,]+)[%]?$/.exec(s); | ||||||
|   if (!match) { |   if (!match) { | ||||||
|     return null; |     return null; | ||||||
|   } |   } | ||||||
|   return parseFloat(match[1].replace(/,/g, "")); |   return parseFloat(`${(match[1] || match[2]) + match[3]}`.replace(/,/g, "")); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const isNumericColumn = (lines: string[][], columnIndex: number) => | const isNumericColumn = (lines: string[][], columnIndex: number) => | ||||||
|   lines.slice(1).every((line) => tryParseNumber(line[columnIndex]) !== null); |   lines.slice(1).every((line) => tryParseNumber(line[columnIndex]) !== null); | ||||||
|  |  | ||||||
| const tryParseCells = (cells: string[][]): ParseSpreadsheetResult => { | /** | ||||||
|  |  * @private exported for testing | ||||||
|  |  */ | ||||||
|  | export const tryParseCells = (cells: string[][]): ParseSpreadsheetResult => { | ||||||
|   const numCols = cells[0].length; |   const numCols = cells[0].length; | ||||||
|  |  | ||||||
|   if (numCols > 2) { |   if (numCols > 2) { | ||||||
| @@ -66,13 +77,16 @@ const tryParseCells = (cells: string[][]): ParseSpreadsheetResult => { | |||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   const valueColumnIndex = isNumericColumn(cells, 0) ? 0 : 1; |   const labelColumnNumeric = isNumericColumn(cells, 0); | ||||||
|  |   const valueColumnNumeric = isNumericColumn(cells, 1); | ||||||
|  |  | ||||||
|   if (!isNumericColumn(cells, valueColumnIndex)) { |   if (!labelColumnNumeric && !valueColumnNumeric) { | ||||||
|     return { type: NOT_SPREADSHEET, reason: "Value is not numeric" }; |     return { type: NOT_SPREADSHEET, reason: "Value is not numeric" }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   const labelColumnIndex = (valueColumnIndex + 1) % 2; |   const [labelColumnIndex, valueColumnIndex] = valueColumnNumeric | ||||||
|  |     ? [0, 1] | ||||||
|  |     : [1, 0]; | ||||||
|   const hasHeader = tryParseNumber(cells[0][valueColumnIndex]) === null; |   const hasHeader = tryParseNumber(cells[0][valueColumnIndex]) === null; | ||||||
|   const rows = hasHeader ? cells.slice(1) : cells; |   const rows = hasHeader ? cells.slice(1) : cells; | ||||||
|  |  | ||||||
| @@ -103,7 +117,7 @@ const transposeCells = (cells: string[][]) => { | |||||||
| }; | }; | ||||||
|  |  | ||||||
| export const tryParseSpreadsheet = (text: string): ParseSpreadsheetResult => { | export const tryParseSpreadsheet = (text: string): ParseSpreadsheetResult => { | ||||||
|   // Copy/paste from excel, spreadhseets, tsv, csv. |   // Copy/paste from excel, spreadsheets, tsv, csv. | ||||||
|   // For now we only accept 2 columns with an optional header |   // For now we only accept 2 columns with an optional header | ||||||
|  |  | ||||||
|   // Check for tab separated values |   // Check for tab separated values | ||||||
| @@ -161,7 +175,8 @@ const commonProps = { | |||||||
|   strokeSharpness: "sharp", |   strokeSharpness: "sharp", | ||||||
|   strokeStyle: "solid", |   strokeStyle: "solid", | ||||||
|   strokeWidth: 1, |   strokeWidth: 1, | ||||||
|   verticalAlign: "middle", |   verticalAlign: VERTICAL_ALIGN.MIDDLE, | ||||||
|  |   locked: false, | ||||||
| } as const; | } as const; | ||||||
|  |  | ||||||
| const getChartDimentions = (spreadsheet: Spreadsheet) => { | const getChartDimentions = (spreadsheet: Spreadsheet) => { | ||||||
|   | |||||||
| @@ -2,16 +2,16 @@ import { | |||||||
|   ExcalidrawElement, |   ExcalidrawElement, | ||||||
|   NonDeletedExcalidrawElement, |   NonDeletedExcalidrawElement, | ||||||
| } from "./element/types"; | } from "./element/types"; | ||||||
| import { getSelectedElements } from "./scene"; |  | ||||||
| import { AppState, BinaryFiles } from "./types"; | import { AppState, BinaryFiles } from "./types"; | ||||||
| import { SVG_EXPORT_TAG } from "./scene/export"; | import { SVG_EXPORT_TAG } from "./scene/export"; | ||||||
| import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts"; | import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts"; | ||||||
| import { EXPORT_DATA_TYPES, MIME_TYPES } from "./constants"; | import { EXPORT_DATA_TYPES, MIME_TYPES } from "./constants"; | ||||||
| import { isInitializedImageElement } from "./element/typeChecks"; | import { isInitializedImageElement } from "./element/typeChecks"; | ||||||
|  | import { isPromiseLike } from "./utils"; | ||||||
|  |  | ||||||
| type ElementsClipboard = { | type ElementsClipboard = { | ||||||
|   type: typeof EXPORT_DATA_TYPES.excalidrawClipboard; |   type: typeof EXPORT_DATA_TYPES.excalidrawClipboard; | ||||||
|   elements: ExcalidrawElement[]; |   elements: readonly NonDeletedExcalidrawElement[]; | ||||||
|   files: BinaryFiles | undefined; |   files: BinaryFiles | undefined; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -56,19 +56,20 @@ const clipboardContainsElements = ( | |||||||
| export const copyToClipboard = async ( | export const copyToClipboard = async ( | ||||||
|   elements: readonly NonDeletedExcalidrawElement[], |   elements: readonly NonDeletedExcalidrawElement[], | ||||||
|   appState: AppState, |   appState: AppState, | ||||||
|   files: BinaryFiles, |   files: BinaryFiles | null, | ||||||
| ) => { | ) => { | ||||||
|   // select binded text elements when copying |   // select binded text elements when copying | ||||||
|   const selectedElements = getSelectedElements(elements, appState, true); |  | ||||||
|   const contents: ElementsClipboard = { |   const contents: ElementsClipboard = { | ||||||
|     type: EXPORT_DATA_TYPES.excalidrawClipboard, |     type: EXPORT_DATA_TYPES.excalidrawClipboard, | ||||||
|     elements: selectedElements, |     elements, | ||||||
|     files: selectedElements.reduce((acc, element) => { |     files: files | ||||||
|  |       ? elements.reduce((acc, element) => { | ||||||
|           if (isInitializedImageElement(element) && files[element.fileId]) { |           if (isInitializedImageElement(element) && files[element.fileId]) { | ||||||
|             acc[element.fileId] = files[element.fileId]; |             acc[element.fileId] = files[element.fileId]; | ||||||
|           } |           } | ||||||
|           return acc; |           return acc; | ||||||
|     }, {} as BinaryFiles), |         }, {} as BinaryFiles) | ||||||
|  |       : undefined, | ||||||
|   }; |   }; | ||||||
|   const json = JSON.stringify(contents); |   const json = JSON.stringify(contents); | ||||||
|   CLIPBOARD = json; |   CLIPBOARD = json; | ||||||
| @@ -124,7 +125,7 @@ const getSystemClipboard = async ( | |||||||
| }; | }; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Attemps to parse clipboard. Prefers system clipboard. |  * Attempts to parse clipboard. Prefers system clipboard. | ||||||
|  */ |  */ | ||||||
| export const parseClipboard = async ( | export const parseClipboard = async ( | ||||||
|   event: ClipboardEvent | null, |   event: ClipboardEvent | null, | ||||||
| @@ -166,10 +167,35 @@ export const parseClipboard = async ( | |||||||
|   } |   } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const copyBlobToClipboardAsPng = async (blob: Blob) => { | export const copyBlobToClipboardAsPng = async (blob: Blob | Promise<Blob>) => { | ||||||
|   await navigator.clipboard.write([ |   let promise; | ||||||
|     new window.ClipboardItem({ [MIME_TYPES.png]: blob }), |   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 copyTextToSystemClipboard = async (text: string | null) => { | export const copyTextToSystemClipboard = async (text: string | null) => { | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import { ActionManager } from "../actions/manager"; | |||||||
| import { getNonDeletedElements } from "../element"; | import { getNonDeletedElements } from "../element"; | ||||||
| import { ExcalidrawElement, PointerType } from "../element/types"; | import { ExcalidrawElement, PointerType } from "../element/types"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "../components/App"; | import { useDevice } from "../components/App"; | ||||||
| import { | import { | ||||||
|   canChangeSharpness, |   canChangeSharpness, | ||||||
|   canHaveArrowheads, |   canHaveArrowheads, | ||||||
| @@ -15,40 +15,59 @@ import { | |||||||
| } from "../scene"; | } from "../scene"; | ||||||
| import { SHAPES } from "../shapes"; | import { SHAPES } from "../shapes"; | ||||||
| import { AppState, Zoom } from "../types"; | import { AppState, Zoom } from "../types"; | ||||||
| import { capitalizeString, isTransparent, setCursorForShape } from "../utils"; | import { | ||||||
|  |   capitalizeString, | ||||||
|  |   isTransparent, | ||||||
|  |   updateActiveTool, | ||||||
|  |   setCursorForShape, | ||||||
|  | } from "../utils"; | ||||||
| import Stack from "./Stack"; | import Stack from "./Stack"; | ||||||
| import { ToolButton } from "./ToolButton"; | import { ToolButton } from "./ToolButton"; | ||||||
| import { hasStrokeColor } from "../scene/comparisons"; | import { hasStrokeColor } from "../scene/comparisons"; | ||||||
|  | import { trackEvent } from "../analytics"; | ||||||
|  | import { hasBoundTextElement, isBoundToContainer } from "../element/typeChecks"; | ||||||
|  |  | ||||||
| export const SelectedShapeActions = ({ | export const SelectedShapeActions = ({ | ||||||
|   appState, |   appState, | ||||||
|   elements, |   elements, | ||||||
|   renderAction, |   renderAction, | ||||||
|   elementType, |   activeTool, | ||||||
| }: { | }: { | ||||||
|   appState: AppState; |   appState: AppState; | ||||||
|   elements: readonly ExcalidrawElement[]; |   elements: readonly ExcalidrawElement[]; | ||||||
|   renderAction: ActionManager["renderAction"]; |   renderAction: ActionManager["renderAction"]; | ||||||
|   elementType: ExcalidrawElement["type"]; |   activeTool: AppState["activeTool"]["type"]; | ||||||
| }) => { | }) => { | ||||||
|   const targetElements = getTargetElements( |   const targetElements = getTargetElements( | ||||||
|     getNonDeletedElements(elements), |     getNonDeletedElements(elements), | ||||||
|     appState, |     appState, | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|  |   let isSingleElementBoundContainer = false; | ||||||
|  |   if ( | ||||||
|  |     targetElements.length === 2 && | ||||||
|  |     (hasBoundTextElement(targetElements[0]) || | ||||||
|  |       hasBoundTextElement(targetElements[1])) | ||||||
|  |   ) { | ||||||
|  |     isSingleElementBoundContainer = true; | ||||||
|  |   } | ||||||
|   const isEditing = Boolean(appState.editingElement); |   const isEditing = Boolean(appState.editingElement); | ||||||
|   const isMobile = useIsMobile(); |   const device = useDevice(); | ||||||
|   const isRTL = document.documentElement.getAttribute("dir") === "rtl"; |   const isRTL = document.documentElement.getAttribute("dir") === "rtl"; | ||||||
|  |  | ||||||
|   const showFillIcons = |   const showFillIcons = | ||||||
|     hasBackground(elementType) || |     hasBackground(activeTool) || | ||||||
|     targetElements.some( |     targetElements.some( | ||||||
|       (element) => |       (element) => | ||||||
|         hasBackground(element.type) && !isTransparent(element.backgroundColor), |         hasBackground(element.type) && !isTransparent(element.backgroundColor), | ||||||
|     ); |     ); | ||||||
|   const showChangeBackgroundIcons = |   const showChangeBackgroundIcons = | ||||||
|     hasBackground(elementType) || |     hasBackground(activeTool) || | ||||||
|     targetElements.some((element) => hasBackground(element.type)); |     targetElements.some((element) => hasBackground(element.type)); | ||||||
|  |  | ||||||
|  |   const showLinkIcon = | ||||||
|  |     targetElements.length === 1 || isSingleElementBoundContainer; | ||||||
|  |  | ||||||
|   let commonSelectedType: string | null = targetElements[0]?.type || null; |   let commonSelectedType: string | null = targetElements[0]?.type || null; | ||||||
|  |  | ||||||
|   for (const element of targetElements) { |   for (const element of targetElements) { | ||||||
| @@ -60,23 +79,23 @@ export const SelectedShapeActions = ({ | |||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="panelColumn"> |     <div className="panelColumn"> | ||||||
|       {((hasStrokeColor(elementType) && |       {((hasStrokeColor(activeTool) && | ||||||
|         elementType !== "image" && |         activeTool !== "image" && | ||||||
|         commonSelectedType !== "image") || |         commonSelectedType !== "image") || | ||||||
|         targetElements.some((element) => hasStrokeColor(element.type))) && |         targetElements.some((element) => hasStrokeColor(element.type))) && | ||||||
|         renderAction("changeStrokeColor")} |         renderAction("changeStrokeColor")} | ||||||
|       {showChangeBackgroundIcons && renderAction("changeBackgroundColor")} |       {showChangeBackgroundIcons && renderAction("changeBackgroundColor")} | ||||||
|       {showFillIcons && renderAction("changeFillStyle")} |       {showFillIcons && renderAction("changeFillStyle")} | ||||||
|  |  | ||||||
|       {(hasStrokeWidth(elementType) || |       {(hasStrokeWidth(activeTool) || | ||||||
|         targetElements.some((element) => hasStrokeWidth(element.type))) && |         targetElements.some((element) => hasStrokeWidth(element.type))) && | ||||||
|         renderAction("changeStrokeWidth")} |         renderAction("changeStrokeWidth")} | ||||||
|  |  | ||||||
|       {(elementType === "freedraw" || |       {(activeTool === "freedraw" || | ||||||
|         targetElements.some((element) => element.type === "freedraw")) && |         targetElements.some((element) => element.type === "freedraw")) && | ||||||
|         renderAction("changeStrokeShape")} |         renderAction("changeStrokeShape")} | ||||||
|  |  | ||||||
|       {(hasStrokeStyle(elementType) || |       {(hasStrokeStyle(activeTool) || | ||||||
|         targetElements.some((element) => hasStrokeStyle(element.type))) && ( |         targetElements.some((element) => hasStrokeStyle(element.type))) && ( | ||||||
|         <> |         <> | ||||||
|           {renderAction("changeStrokeStyle")} |           {renderAction("changeStrokeStyle")} | ||||||
| @@ -84,12 +103,12 @@ export const SelectedShapeActions = ({ | |||||||
|         </> |         </> | ||||||
|       )} |       )} | ||||||
|  |  | ||||||
|       {(canChangeSharpness(elementType) || |       {(canChangeSharpness(activeTool) || | ||||||
|         targetElements.some((element) => canChangeSharpness(element.type))) && ( |         targetElements.some((element) => canChangeSharpness(element.type))) && ( | ||||||
|         <>{renderAction("changeSharpness")}</> |         <>{renderAction("changeSharpness")}</> | ||||||
|       )} |       )} | ||||||
|  |  | ||||||
|       {(hasText(elementType) || |       {(hasText(activeTool) || | ||||||
|         targetElements.some((element) => hasText(element.type))) && ( |         targetElements.some((element) => hasText(element.type))) && ( | ||||||
|         <> |         <> | ||||||
|           {renderAction("changeFontSize")} |           {renderAction("changeFontSize")} | ||||||
| @@ -100,7 +119,11 @@ export const SelectedShapeActions = ({ | |||||||
|         </> |         </> | ||||||
|       )} |       )} | ||||||
|  |  | ||||||
|       {(canHaveArrowheads(elementType) || |       {targetElements.some( | ||||||
|  |         (element) => | ||||||
|  |           hasBoundTextElement(element) || isBoundToContainer(element), | ||||||
|  |       ) && renderAction("changeVerticalAlign")} | ||||||
|  |       {(canHaveArrowheads(activeTool) || | ||||||
|         targetElements.some((element) => canHaveArrowheads(element.type))) && ( |         targetElements.some((element) => canHaveArrowheads(element.type))) && ( | ||||||
|         <>{renderAction("changeArrowhead")}</> |         <>{renderAction("changeArrowhead")}</> | ||||||
|       )} |       )} | ||||||
| @@ -117,7 +140,7 @@ export const SelectedShapeActions = ({ | |||||||
|         </div> |         </div> | ||||||
|       </fieldset> |       </fieldset> | ||||||
|  |  | ||||||
|       {targetElements.length > 1 && ( |       {targetElements.length > 1 && !isSingleElementBoundContainer && ( | ||||||
|         <fieldset> |         <fieldset> | ||||||
|           <legend>{t("labels.align")}</legend> |           <legend>{t("labels.align")}</legend> | ||||||
|           <div className="buttonList"> |           <div className="buttonList"> | ||||||
| @@ -150,15 +173,15 @@ export const SelectedShapeActions = ({ | |||||||
|           </div> |           </div> | ||||||
|         </fieldset> |         </fieldset> | ||||||
|       )} |       )} | ||||||
|       {!isMobile && !isEditing && targetElements.length > 0 && ( |       {!isEditing && targetElements.length > 0 && ( | ||||||
|         <fieldset> |         <fieldset> | ||||||
|           <legend>{t("labels.actions")}</legend> |           <legend>{t("labels.actions")}</legend> | ||||||
|           <div className="buttonList"> |           <div className="buttonList"> | ||||||
|             {renderAction("duplicateSelection")} |             {!device.isMobile && renderAction("duplicateSelection")} | ||||||
|             {renderAction("deleteSelectedElements")} |             {!device.isMobile && renderAction("deleteSelectedElements")} | ||||||
|             {renderAction("group")} |             {renderAction("group")} | ||||||
|             {renderAction("ungroup")} |             {renderAction("ungroup")} | ||||||
|             {targetElements.length === 1 && renderAction("link")} |             {showLinkIcon && renderAction("hyperlink")} | ||||||
|           </div> |           </div> | ||||||
|         </fieldset> |         </fieldset> | ||||||
|       )} |       )} | ||||||
| @@ -168,14 +191,16 @@ export const SelectedShapeActions = ({ | |||||||
|  |  | ||||||
| export const ShapesSwitcher = ({ | export const ShapesSwitcher = ({ | ||||||
|   canvas, |   canvas, | ||||||
|   elementType, |   activeTool, | ||||||
|   setAppState, |   setAppState, | ||||||
|   onImageAction, |   onImageAction, | ||||||
|  |   appState, | ||||||
| }: { | }: { | ||||||
|   canvas: HTMLCanvasElement | null; |   canvas: HTMLCanvasElement | null; | ||||||
|   elementType: ExcalidrawElement["type"]; |   activeTool: AppState["activeTool"]; | ||||||
|   setAppState: React.Component<any, AppState>["setState"]; |   setAppState: React.Component<any, AppState>["setState"]; | ||||||
|   onImageAction: (data: { pointerType: PointerType | null }) => void; |   onImageAction: (data: { pointerType: PointerType | null }) => void; | ||||||
|  |   appState: AppState; | ||||||
| }) => ( | }) => ( | ||||||
|   <> |   <> | ||||||
|     {SHAPES.map(({ value, icon, key }, index) => { |     {SHAPES.map(({ value, icon, key }, index) => { | ||||||
| @@ -190,20 +215,37 @@ export const ShapesSwitcher = ({ | |||||||
|           key={value} |           key={value} | ||||||
|           type="radio" |           type="radio" | ||||||
|           icon={icon} |           icon={icon} | ||||||
|           checked={elementType === value} |           checked={activeTool.type === value} | ||||||
|           name="editor-current-shape" |           name="editor-current-shape" | ||||||
|           title={`${capitalizeString(label)} — ${shortcut}`} |           title={`${capitalizeString(label)} — ${shortcut}`} | ||||||
|           keyBindingLabel={`${index + 1}`} |           keyBindingLabel={`${index + 1}`} | ||||||
|           aria-label={capitalizeString(label)} |           aria-label={capitalizeString(label)} | ||||||
|           aria-keyshortcuts={shortcut} |           aria-keyshortcuts={shortcut} | ||||||
|           data-testid={value} |           data-testid={value} | ||||||
|           onChange={({ pointerType }) => { |           onPointerDown={({ pointerType }) => { | ||||||
|  |             if (!appState.penDetected && pointerType === "pen") { | ||||||
|               setAppState({ |               setAppState({ | ||||||
|               elementType: value, |                 penDetected: true, | ||||||
|  |                 penMode: true, | ||||||
|  |               }); | ||||||
|  |             } | ||||||
|  |           }} | ||||||
|  |           onChange={({ pointerType }) => { | ||||||
|  |             if (appState.activeTool.type !== value) { | ||||||
|  |               trackEvent("toolbar", value, "ui"); | ||||||
|  |             } | ||||||
|  |             const nextActiveTool = updateActiveTool(appState, { | ||||||
|  |               type: value, | ||||||
|  |             }); | ||||||
|  |             setAppState({ | ||||||
|  |               activeTool: nextActiveTool, | ||||||
|               multiElement: null, |               multiElement: null, | ||||||
|               selectedElementIds: {}, |               selectedElementIds: {}, | ||||||
|             }); |             }); | ||||||
|             setCursorForShape(canvas, value); |             setCursorForShape(canvas, { | ||||||
|  |               ...appState, | ||||||
|  |               activeTool: nextActiveTool, | ||||||
|  |             }); | ||||||
|             if (value === "image") { |             if (value === "image") { | ||||||
|               onImageAction({ pointerType }); |               onImageAction({ pointerType }); | ||||||
|             } |             } | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -12,5 +12,11 @@ | |||||||
|     cursor: pointer; |     cursor: pointer; | ||||||
|     font-size: 0.8rem; |     font-size: 0.8rem; | ||||||
|     font-weight: 500; |     font-weight: 500; | ||||||
|  |  | ||||||
|  |     &-img { | ||||||
|  |       width: 100%; | ||||||
|  |       height: 100%; | ||||||
|  |       border-radius: 100%; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,20 +1,36 @@ | |||||||
| import "./Avatar.scss"; | import "./Avatar.scss"; | ||||||
|  |  | ||||||
| import React from "react"; | import React, { useState } from "react"; | ||||||
|  | import { getClientInitials } from "../clients"; | ||||||
|  |  | ||||||
| type AvatarProps = { | type AvatarProps = { | ||||||
|   children: string; |  | ||||||
|   onClick: (e: React.MouseEvent<HTMLDivElement, MouseEvent>) => void; |   onClick: (e: React.MouseEvent<HTMLDivElement, MouseEvent>) => void; | ||||||
|   color: string; |   color: string; | ||||||
|   border: string; |   border: string; | ||||||
|  |   name: string; | ||||||
|  |   src?: string; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const Avatar = ({ children, color, border, onClick }: AvatarProps) => ( | export const Avatar = ({ color, border, onClick, name, src }: AvatarProps) => { | ||||||
|   <div |   const shortName = getClientInitials(name); | ||||||
|     className="Avatar" |   const [error, setError] = useState(false); | ||||||
|     style={{ background: color, border: `1px solid ${border}` }} |   const loadImg = !error && src; | ||||||
|     onClick={onClick} |   const style = loadImg | ||||||
|   > |     ? undefined | ||||||
|     {children} |     : { background: color, border: `1px solid ${border}` }; | ||||||
|  |   return ( | ||||||
|  |     <div className="Avatar" style={style} onClick={onClick}> | ||||||
|  |       {loadImg ? ( | ||||||
|  |         <img | ||||||
|  |           className="Avatar-img" | ||||||
|  |           src={src} | ||||||
|  |           alt={shortName} | ||||||
|  |           referrerPolicy="no-referrer" | ||||||
|  |           onError={() => setError(true)} | ||||||
|  |         /> | ||||||
|  |       ) : ( | ||||||
|  |         shortName | ||||||
|  |       )} | ||||||
|     </div> |     </div> | ||||||
| ); |   ); | ||||||
|  | }; | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { useState } from "react"; | import { useState } from "react"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "./App"; | import { useDevice } from "./App"; | ||||||
| import { trash } from "./icons"; | import { trash } from "./icons"; | ||||||
| import { ToolButton } from "./ToolButton"; | import { ToolButton } from "./ToolButton"; | ||||||
|  |  | ||||||
| @@ -19,7 +19,7 @@ const ClearCanvas = ({ onConfirm }: { onConfirm: () => void }) => { | |||||||
|         icon={trash} |         icon={trash} | ||||||
|         title={t("buttons.clearReset")} |         title={t("buttons.clearReset")} | ||||||
|         aria-label={t("buttons.clearReset")} |         aria-label={t("buttons.clearReset")} | ||||||
|         showAriaLabel={useIsMobile()} |         showAriaLabel={useDevice().isMobile} | ||||||
|         onClick={toggleDialog} |         onClick={toggleDialog} | ||||||
|         data-testid="clear-canvas-button" |         data-testid="clear-canvas-button" | ||||||
|       /> |       /> | ||||||
|   | |||||||
| @@ -18,13 +18,15 @@ | |||||||
|       left: -5px; |       left: -5px; | ||||||
|     } |     } | ||||||
|     min-width: 1em; |     min-width: 1em; | ||||||
|  |     min-height: 1em; | ||||||
|  |     line-height: 1; | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     bottom: -5px; |     bottom: -5px; | ||||||
|     padding: 3px; |     padding: 3px; | ||||||
|     border-radius: 50%; |     border-radius: 50%; | ||||||
|     background-color: $oc-green-6; |     background-color: $oc-green-6; | ||||||
|     color: $oc-white; |     color: $oc-white; | ||||||
|     font-size: 0.7em; |     font-size: 0.6em; | ||||||
|     font-family: var(--ui-font); |     font-family: "Cascadia"; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| import clsx from "clsx"; | import clsx from "clsx"; | ||||||
| import { ToolButton } from "./ToolButton"; | import { ToolButton } from "./ToolButton"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "../components/App"; | import { useDevice } from "../components/App"; | ||||||
| import { users } from "./icons"; | import { users } from "./icons"; | ||||||
|  |  | ||||||
| import "./CollabButton.scss"; | import "./CollabButton.scss"; | ||||||
| @@ -26,9 +26,9 @@ const CollabButton = ({ | |||||||
|         type="button" |         type="button" | ||||||
|         title={t("labels.liveCollaboration")} |         title={t("labels.liveCollaboration")} | ||||||
|         aria-label={t("labels.liveCollaboration")} |         aria-label={t("labels.liveCollaboration")} | ||||||
|         showAriaLabel={useIsMobile()} |         showAriaLabel={useDevice().isMobile} | ||||||
|       > |       > | ||||||
|         {collaboratorCount > 0 && ( |         {isCollaborating && ( | ||||||
|           <div className="CollabButton-collaborators">{collaboratorCount}</div> |           <div className="CollabButton-collaborators">{collaboratorCount}</div> | ||||||
|         )} |         )} | ||||||
|       </ToolButton> |       </ToolButton> | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ | |||||||
|     top: -11px; |     top: -11px; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .color-picker-content { |   .color-picker-content--default { | ||||||
|     padding: 0.5rem; |     padding: 0.5rem; | ||||||
|     display: grid; |     display: grid; | ||||||
|     grid-template-columns: repeat(5, auto); |     grid-template-columns: repeat(5, auto); | ||||||
| @@ -59,6 +59,26 @@ | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   .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 { |   .color-picker-content .color-input-container { | ||||||
|     grid-column: 1 / span 5; |     grid-column: 1 / span 5; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -7,6 +7,53 @@ import { isArrowKey, KEYS } from "../keys"; | |||||||
| import { t, getLanguage } from "../i18n"; | import { t, getLanguage } from "../i18n"; | ||||||
| import { isWritableElement } from "../utils"; | import { isWritableElement } from "../utils"; | ||||||
| import colors from "../colors"; | 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 isValidColor = (color: string) => { | ||||||
|   const style = new Option().style; |   const style = new Option().style; | ||||||
| @@ -35,6 +82,7 @@ const keyBindings = [ | |||||||
|   ["1", "2", "3", "4", "5"], |   ["1", "2", "3", "4", "5"], | ||||||
|   ["q", "w", "e", "r", "t"], |   ["q", "w", "e", "r", "t"], | ||||||
|   ["a", "s", "d", "f", "g"], |   ["a", "s", "d", "f", "g"], | ||||||
|  |   ["z", "x", "c", "v", "b"], | ||||||
| ].flat(); | ].flat(); | ||||||
|  |  | ||||||
| const Picker = ({ | const Picker = ({ | ||||||
| @@ -45,6 +93,7 @@ const Picker = ({ | |||||||
|   label, |   label, | ||||||
|   showInput = true, |   showInput = true, | ||||||
|   type, |   type, | ||||||
|  |   elements, | ||||||
| }: { | }: { | ||||||
|   colors: string[]; |   colors: string[]; | ||||||
|   color: string | null; |   color: string | null; | ||||||
| @@ -53,12 +102,20 @@ const Picker = ({ | |||||||
|   label: string; |   label: string; | ||||||
|   showInput: boolean; |   showInput: boolean; | ||||||
|   type: "canvasBackground" | "elementBackground" | "elementStroke"; |   type: "canvasBackground" | "elementBackground" | "elementStroke"; | ||||||
|  |   elements: readonly ExcalidrawElement[]; | ||||||
| }) => { | }) => { | ||||||
|   const firstItem = React.useRef<HTMLButtonElement>(); |   const firstItem = React.useRef<HTMLButtonElement>(); | ||||||
|   const activeItem = React.useRef<HTMLButtonElement>(); |   const activeItem = React.useRef<HTMLButtonElement>(); | ||||||
|   const gallery = React.useRef<HTMLDivElement>(); |   const gallery = React.useRef<HTMLDivElement>(); | ||||||
|   const colorInput = React.useRef<HTMLInputElement>(); |   const colorInput = React.useRef<HTMLInputElement>(); | ||||||
|  |  | ||||||
|  |   const [customColors] = React.useState(() => { | ||||||
|  |     if (type === "canvasBackground") { | ||||||
|  |       return []; | ||||||
|  |     } | ||||||
|  |     return getCustomColors(elements, type); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|   React.useEffect(() => { |   React.useEffect(() => { | ||||||
|     // After the component is first mounted focus on first input |     // After the component is first mounted focus on first input | ||||||
|     if (activeItem.current) { |     if (activeItem.current) { | ||||||
| @@ -71,52 +128,119 @@ const Picker = ({ | |||||||
|   }, []); |   }, []); | ||||||
|  |  | ||||||
|   const handleKeyDown = (event: React.KeyboardEvent) => { |   const handleKeyDown = (event: React.KeyboardEvent) => { | ||||||
|     if (event.key === KEYS.TAB) { |     let handled = false; | ||||||
|       const { activeElement } = document; |     if (isArrowKey(event.key)) { | ||||||
|       if (event.shiftKey) { |       handled = true; | ||||||
|         if (activeElement === firstItem.current) { |  | ||||||
|           colorInput.current?.focus(); |  | ||||||
|           event.preventDefault(); |  | ||||||
|         } |  | ||||||
|       } else if (activeElement === colorInput.current) { |  | ||||||
|         firstItem.current?.focus(); |  | ||||||
|         event.preventDefault(); |  | ||||||
|       } |  | ||||||
|     } else if (isArrowKey(event.key)) { |  | ||||||
|       const { activeElement } = document; |       const { activeElement } = document; | ||||||
|       const isRTL = getLanguage().rtl; |       const isRTL = getLanguage().rtl; | ||||||
|       const index = Array.prototype.indexOf.call( |       let isCustom = false; | ||||||
|         gallery!.current!.children, |       let index = Array.prototype.indexOf.call( | ||||||
|  |         gallery.current!.querySelector(".color-picker-content--default") | ||||||
|  |           ?.children, | ||||||
|  |         activeElement, | ||||||
|  |       ); | ||||||
|  |       if (index === -1) { | ||||||
|  |         index = Array.prototype.indexOf.call( | ||||||
|  |           gallery.current!.querySelector(".color-picker-content--canvas-colors") | ||||||
|  |             ?.children, | ||||||
|           activeElement, |           activeElement, | ||||||
|         ); |         ); | ||||||
|         if (index !== -1) { |         if (index !== -1) { | ||||||
|         const length = gallery!.current!.children.length - (showInput ? 1 : 0); |           isCustom = true; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       const parentElement = isCustom | ||||||
|  |         ? gallery.current?.querySelector(".color-picker-content--canvas-colors") | ||||||
|  |         : gallery.current?.querySelector(".color-picker-content--default"); | ||||||
|  |  | ||||||
|  |       if (parentElement && index !== -1) { | ||||||
|  |         const length = parentElement.children.length - (showInput ? 1 : 0); | ||||||
|         const nextIndex = |         const nextIndex = | ||||||
|           event.key === (isRTL ? KEYS.ARROW_LEFT : KEYS.ARROW_RIGHT) |           event.key === (isRTL ? KEYS.ARROW_LEFT : KEYS.ARROW_RIGHT) | ||||||
|             ? (index + 1) % length |             ? (index + 1) % length | ||||||
|             : event.key === (isRTL ? KEYS.ARROW_RIGHT : KEYS.ARROW_LEFT) |             : event.key === (isRTL ? KEYS.ARROW_RIGHT : KEYS.ARROW_LEFT) | ||||||
|             ? (length + index - 1) % length |             ? (length + index - 1) % length | ||||||
|             : event.key === KEYS.ARROW_DOWN |             : !isCustom && event.key === KEYS.ARROW_DOWN | ||||||
|             ? (index + 5) % length |             ? (index + 5) % length | ||||||
|             : event.key === KEYS.ARROW_UP |             : !isCustom && event.key === KEYS.ARROW_UP | ||||||
|             ? (length + index - 5) % length |             ? (length + index - 5) % length | ||||||
|             : index; |             : index; | ||||||
|         (gallery!.current!.children![nextIndex] as any).focus(); |         (parentElement.children[nextIndex] as HTMLElement | undefined)?.focus(); | ||||||
|       } |       } | ||||||
|       event.preventDefault(); |       event.preventDefault(); | ||||||
|     } else if ( |     } else if ( | ||||||
|       keyBindings.includes(event.key.toLowerCase()) && |       keyBindings.includes(event.key.toLowerCase()) && | ||||||
|  |       !event[KEYS.CTRL_OR_CMD] && | ||||||
|  |       !event.altKey && | ||||||
|       !isWritableElement(event.target) |       !isWritableElement(event.target) | ||||||
|     ) { |     ) { | ||||||
|  |       handled = true; | ||||||
|       const index = keyBindings.indexOf(event.key.toLowerCase()); |       const index = keyBindings.indexOf(event.key.toLowerCase()); | ||||||
|       (gallery!.current!.children![index] as any).focus(); |       const isCustom = index >= MAX_DEFAULT_COLORS; | ||||||
|  |       const parentElement = isCustom | ||||||
|  |         ? gallery?.current?.querySelector( | ||||||
|  |             ".color-picker-content--canvas-colors", | ||||||
|  |           ) | ||||||
|  |         : gallery?.current?.querySelector(".color-picker-content--default"); | ||||||
|  |       const actualIndex = isCustom ? index - MAX_DEFAULT_COLORS : index; | ||||||
|  |       ( | ||||||
|  |         parentElement?.children[actualIndex] as HTMLElement | undefined | ||||||
|  |       )?.focus(); | ||||||
|  |  | ||||||
|       event.preventDefault(); |       event.preventDefault(); | ||||||
|     } else if (event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) { |     } else if (event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) { | ||||||
|  |       handled = true; | ||||||
|       event.preventDefault(); |       event.preventDefault(); | ||||||
|       onClose(); |       onClose(); | ||||||
|     } |     } | ||||||
|  |     if (handled) { | ||||||
|       event.nativeEvent.stopImmediatePropagation(); |       event.nativeEvent.stopImmediatePropagation(); | ||||||
|       event.stopPropagation(); |       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 ( |   return ( | ||||||
| @@ -136,43 +260,23 @@ const Picker = ({ | |||||||
|             gallery.current = el; |             gallery.current = el; | ||||||
|           } |           } | ||||||
|         }} |         }} | ||||||
|         tabIndex={0} |         // to allow focusing by clicking but not by tabbing | ||||||
|  |         tabIndex={-1} | ||||||
|       > |       > | ||||||
|         {colors.map((_color, i) => { |         <div className="color-picker-content--default"> | ||||||
|           const _colorWithoutHash = _color.replace("#", ""); |           {renderColors(colors)} | ||||||
|           return ( |         </div> | ||||||
|             <button |         {!!customColors.length && ( | ||||||
|               className="color-picker-swatch" |           <div className="color-picker-content--canvas"> | ||||||
|               onClick={(event) => { |             <span className="color-picker-content--canvas-title"> | ||||||
|                 (event.currentTarget as HTMLButtonElement).focus(); |               {t("labels.canvasColors")} | ||||||
|                 onChange(_color); |             </span> | ||||||
|               }} |             <div className="color-picker-content--canvas-colors"> | ||||||
|               title={`${t(`colors.${_colorWithoutHash}`)}${ |               {renderColors(customColors, true)} | ||||||
|                 !isTransparent(_color) ? ` (${_color})` : "" |             </div> | ||||||
|               } — ${keyBindings[i].toUpperCase()}`} |           </div> | ||||||
|               aria-label={t(`colors.${_colorWithoutHash}`)} |         )} | ||||||
|               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); |  | ||||||
|               }} |  | ||||||
|             > |  | ||||||
|               {isTransparent(_color) ? ( |  | ||||||
|                 <div className="color-picker-transparent"></div> |  | ||||||
|               ) : undefined} |  | ||||||
|               <span className="color-picker-keybinding">{keyBindings[i]}</span> |  | ||||||
|             </button> |  | ||||||
|           ); |  | ||||||
|         })} |  | ||||||
|         {showInput && ( |         {showInput && ( | ||||||
|           <ColorInput |           <ColorInput | ||||||
|             color={color} |             color={color} | ||||||
| @@ -246,6 +350,8 @@ export const ColorPicker = ({ | |||||||
|   label, |   label, | ||||||
|   isActive, |   isActive, | ||||||
|   setActive, |   setActive, | ||||||
|  |   elements, | ||||||
|  |   appState, | ||||||
| }: { | }: { | ||||||
|   type: "canvasBackground" | "elementBackground" | "elementStroke"; |   type: "canvasBackground" | "elementBackground" | "elementStroke"; | ||||||
|   color: string | null; |   color: string | null; | ||||||
| @@ -253,6 +359,8 @@ export const ColorPicker = ({ | |||||||
|   label: string; |   label: string; | ||||||
|   isActive: boolean; |   isActive: boolean; | ||||||
|   setActive: (active: boolean) => void; |   setActive: (active: boolean) => void; | ||||||
|  |   elements: readonly ExcalidrawElement[]; | ||||||
|  |   appState: AppState; | ||||||
| }) => { | }) => { | ||||||
|   const pickerButton = React.useRef<HTMLButtonElement>(null); |   const pickerButton = React.useRef<HTMLButtonElement>(null); | ||||||
|  |  | ||||||
| @@ -294,6 +402,7 @@ export const ColorPicker = ({ | |||||||
|               label={label} |               label={label} | ||||||
|               showInput={false} |               showInput={false} | ||||||
|               type={type} |               type={type} | ||||||
|  |               elements={elements} | ||||||
|             /> |             /> | ||||||
|           </Popover> |           </Popover> | ||||||
|         ) : null} |         ) : null} | ||||||
|   | |||||||
| @@ -70,7 +70,9 @@ const ContextMenu = ({ | |||||||
|                   dangerous: actionName === "deleteSelectedElements", |                   dangerous: actionName === "deleteSelectedElements", | ||||||
|                   checkmark: option.checked?.(appState), |                   checkmark: option.checked?.(appState), | ||||||
|                 })} |                 })} | ||||||
|                 onClick={() => actionManager.executeAction(option)} |                 onClick={() => | ||||||
|  |                   actionManager.executeAction(option, "contextMenu") | ||||||
|  |                 } | ||||||
|               > |               > | ||||||
|                 <div className="context-menu-option__label">{label}</div> |                 <div className="context-menu-option__label">{label}</div> | ||||||
|                 <kbd className="context-menu-option__shortcut"> |                 <kbd className="context-menu-option__shortcut"> | ||||||
|   | |||||||
| @@ -2,13 +2,14 @@ import clsx from "clsx"; | |||||||
| import React, { useEffect, useState } from "react"; | import React, { useEffect, useState } from "react"; | ||||||
| import { useCallbackRefState } from "../hooks/useCallbackRefState"; | import { useCallbackRefState } from "../hooks/useCallbackRefState"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useExcalidrawContainer, useIsMobile } from "../components/App"; | import { useExcalidrawContainer, useDevice } from "../components/App"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import "./Dialog.scss"; | import "./Dialog.scss"; | ||||||
| import { back, close } from "./icons"; | import { back, close } from "./icons"; | ||||||
| import { Island } from "./Island"; | import { Island } from "./Island"; | ||||||
| import { Modal } from "./Modal"; | import { Modal } from "./Modal"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
|  | import { queryFocusableElements } from "../utils"; | ||||||
|  |  | ||||||
| export interface DialogProps { | export interface DialogProps { | ||||||
|   children: React.ReactNode; |   children: React.ReactNode; | ||||||
| @@ -64,14 +65,6 @@ export const Dialog = (props: DialogProps) => { | |||||||
|     return () => islandNode.removeEventListener("keydown", handleKeyDown); |     return () => islandNode.removeEventListener("keydown", handleKeyDown); | ||||||
|   }, [islandNode, props.autofocus]); |   }, [islandNode, props.autofocus]); | ||||||
|  |  | ||||||
|   const queryFocusableElements = (node: HTMLElement) => { |  | ||||||
|     const focusableElements = node.querySelectorAll<HTMLElement>( |  | ||||||
|       "button, a, input, select, textarea, div[tabindex]", |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     return focusableElements ? Array.from(focusableElements) : []; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   const onClose = () => { |   const onClose = () => { | ||||||
|     (lastActiveElement as HTMLElement).focus(); |     (lastActiveElement as HTMLElement).focus(); | ||||||
|     props.onCloseRequest(); |     props.onCloseRequest(); | ||||||
| @@ -94,7 +87,7 @@ export const Dialog = (props: DialogProps) => { | |||||||
|             onClick={onClose} |             onClick={onClose} | ||||||
|             aria-label={t("buttons.close")} |             aria-label={t("buttons.close")} | ||||||
|           > |           > | ||||||
|             {useIsMobile() ? back : close} |             {useDevice().isMobile ? back : close} | ||||||
|           </button> |           </button> | ||||||
|         </h2> |         </h2> | ||||||
|         <div className="Dialog__content">{props.children}</div> |         <div className="Dialog__content">{props.children}</div> | ||||||
|   | |||||||
| @@ -139,7 +139,7 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | |||||||
|         <Section title={t("helpDialog.shortcuts")}> |         <Section title={t("helpDialog.shortcuts")}> | ||||||
|           <Columns> |           <Columns> | ||||||
|             <Column> |             <Column> | ||||||
|               <ShortcutIsland caption={t("helpDialog.shapes")}> |               <ShortcutIsland caption={t("helpDialog.tools")}> | ||||||
|                 <Shortcut |                 <Shortcut | ||||||
|                   label={t("toolBar.selection")} |                   label={t("toolBar.selection")} | ||||||
|                   shortcuts={["V", "1"]} |                   shortcuts={["V", "1"]} | ||||||
| @@ -149,7 +149,7 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | |||||||
|                   shortcuts={["R", "2"]} |                   shortcuts={["R", "2"]} | ||||||
|                 /> |                 /> | ||||||
|                 <Shortcut label={t("toolBar.diamond")} shortcuts={["D", "3"]} /> |                 <Shortcut label={t("toolBar.diamond")} shortcuts={["D", "3"]} /> | ||||||
|                 <Shortcut label={t("toolBar.ellipse")} shortcuts={["E", "4"]} /> |                 <Shortcut label={t("toolBar.ellipse")} shortcuts={["O", "4"]} /> | ||||||
|                 <Shortcut label={t("toolBar.arrow")} shortcuts={["A", "5"]} /> |                 <Shortcut label={t("toolBar.arrow")} shortcuts={["A", "5"]} /> | ||||||
|                 <Shortcut label={t("toolBar.line")} shortcuts={["P", "6"]} /> |                 <Shortcut label={t("toolBar.line")} shortcuts={["P", "6"]} /> | ||||||
|                 <Shortcut |                 <Shortcut | ||||||
| @@ -159,6 +159,10 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | |||||||
|                 <Shortcut label={t("toolBar.text")} shortcuts={["T", "8"]} /> |                 <Shortcut label={t("toolBar.text")} shortcuts={["T", "8"]} /> | ||||||
|                 <Shortcut label={t("toolBar.image")} shortcuts={["9"]} /> |                 <Shortcut label={t("toolBar.image")} shortcuts={["9"]} /> | ||||||
|                 <Shortcut label={t("toolBar.library")} shortcuts={["0"]} /> |                 <Shortcut label={t("toolBar.library")} shortcuts={["0"]} /> | ||||||
|  |                 <Shortcut | ||||||
|  |                   label={t("toolBar.eraser")} | ||||||
|  |                   shortcuts={[getShortcutKey("E")]} | ||||||
|  |                 /> | ||||||
|                 <Shortcut |                 <Shortcut | ||||||
|                   label={t("helpDialog.editSelectedShape")} |                   label={t("helpDialog.editSelectedShape")} | ||||||
|                   shortcuts={[ |                   shortcuts={[ | ||||||
| @@ -359,6 +363,10 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | |||||||
|                     getShortcutKey(`Alt+${t("helpDialog.drag")}`), |                     getShortcutKey(`Alt+${t("helpDialog.drag")}`), | ||||||
|                   ]} |                   ]} | ||||||
|                 /> |                 /> | ||||||
|  |                 <Shortcut | ||||||
|  |                   label={t("helpDialog.toggleElementLock")} | ||||||
|  |                   shortcuts={[getShortcutKey("CtrlOrCmd+Shift+L")]} | ||||||
|  |                 /> | ||||||
|                 <Shortcut |                 <Shortcut | ||||||
|                   label={t("buttons.undo")} |                   label={t("buttons.undo")} | ||||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+Z")]} |                   shortcuts={[getShortcutKey("CtrlOrCmd+Z")]} | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ import { | |||||||
|   isTextElement, |   isTextElement, | ||||||
| } from "../element/typeChecks"; | } from "../element/typeChecks"; | ||||||
| import { getShortcutKey } from "../utils"; | import { getShortcutKey } from "../utils"; | ||||||
|  | import { isEraserActive } from "../appState"; | ||||||
|  |  | ||||||
| interface HintViewerProps { | interface HintViewerProps { | ||||||
|   appState: AppState; |   appState: AppState; | ||||||
| @@ -19,25 +20,32 @@ interface HintViewerProps { | |||||||
| } | } | ||||||
|  |  | ||||||
| const getHints = ({ appState, elements, isMobile }: HintViewerProps) => { | const getHints = ({ appState, elements, isMobile }: HintViewerProps) => { | ||||||
|   const { elementType, isResizing, isRotating, lastPointerDownWith } = appState; |   const { activeTool, isResizing, isRotating, lastPointerDownWith } = appState; | ||||||
|   const multiMode = appState.multiElement !== null; |   const multiMode = appState.multiElement !== null; | ||||||
|  |  | ||||||
|   if (elementType === "arrow" || elementType === "line") { |   if (appState.isLibraryOpen) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (isEraserActive(appState)) { | ||||||
|  |     return t("hints.eraserRevert"); | ||||||
|  |   } | ||||||
|  |   if (activeTool.type === "arrow" || activeTool.type === "line") { | ||||||
|     if (!multiMode) { |     if (!multiMode) { | ||||||
|       return t("hints.linearElement"); |       return t("hints.linearElement"); | ||||||
|     } |     } | ||||||
|     return t("hints.linearElementMulti"); |     return t("hints.linearElementMulti"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (elementType === "freedraw") { |   if (activeTool.type === "freedraw") { | ||||||
|     return t("hints.freeDraw"); |     return t("hints.freeDraw"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (elementType === "text") { |   if (activeTool.type === "text") { | ||||||
|     return t("hints.text"); |     return t("hints.text"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (appState.elementType === "image" && appState.pendingImageElement) { |   if (appState.activeTool.type === "image" && appState.pendingImageElementId) { | ||||||
|     return t("hints.placeImage"); |     return t("hints.placeImage"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -69,7 +77,7 @@ const getHints = ({ appState, elements, isMobile }: HintViewerProps) => { | |||||||
|     return t("hints.text_editing"); |     return t("hints.text_editing"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (elementType === "selection") { |   if (activeTool.type === "selection") { | ||||||
|     if ( |     if ( | ||||||
|       appState.draggingElement?.type === "selection" && |       appState.draggingElement?.type === "selection" && | ||||||
|       !appState.editingElement && |       !appState.editingElement && | ||||||
|   | |||||||
| @@ -1,12 +1,11 @@ | |||||||
| import React, { useEffect, useRef, useState } from "react"; | import React, { useEffect, useRef, useState } from "react"; | ||||||
| import { render, unmountComponentAtNode } from "react-dom"; | import { render, unmountComponentAtNode } from "react-dom"; | ||||||
| import { ActionsManagerInterface } from "../actions/types"; |  | ||||||
| import { probablySupportsClipboardBlob } from "../clipboard"; | import { probablySupportsClipboardBlob } from "../clipboard"; | ||||||
| import { canvasToBlob } from "../data/blob"; | import { canvasToBlob } from "../data/blob"; | ||||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | import { NonDeletedExcalidrawElement } from "../element/types"; | ||||||
| import { CanvasError } from "../errors"; | import { CanvasError } from "../errors"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "./App"; | import { useDevice } from "./App"; | ||||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||||
| import { exportToCanvas } from "../scene/export"; | import { exportToCanvas } from "../scene/export"; | ||||||
| import { AppState, BinaryFiles } from "../types"; | import { AppState, BinaryFiles } from "../types"; | ||||||
| @@ -19,6 +18,7 @@ import OpenColor from "open-color"; | |||||||
| import { CheckboxItem } from "./CheckboxItem"; | import { CheckboxItem } from "./CheckboxItem"; | ||||||
| import { DEFAULT_EXPORT_PADDING } from "../constants"; | import { DEFAULT_EXPORT_PADDING } from "../constants"; | ||||||
| import { nativeFileSystemSupported } from "../data/filesystem"; | import { nativeFileSystemSupported } from "../data/filesystem"; | ||||||
|  | import { ActionManager } from "../actions/manager"; | ||||||
|  |  | ||||||
| const supportsContextFilters = | const supportsContextFilters = | ||||||
|   "filter" in document.createElement("canvas").getContext("2d")!; |   "filter" in document.createElement("canvas").getContext("2d")!; | ||||||
| @@ -90,7 +90,7 @@ const ImageExportModal = ({ | |||||||
|   elements: readonly NonDeletedExcalidrawElement[]; |   elements: readonly NonDeletedExcalidrawElement[]; | ||||||
|   files: BinaryFiles; |   files: BinaryFiles; | ||||||
|   exportPadding?: number; |   exportPadding?: number; | ||||||
|   actionManager: ActionsManagerInterface; |   actionManager: ActionManager; | ||||||
|   onExportToPng: ExportCB; |   onExportToPng: ExportCB; | ||||||
|   onExportToSvg: ExportCB; |   onExportToSvg: ExportCB; | ||||||
|   onExportToClipboard: ExportCB; |   onExportToClipboard: ExportCB; | ||||||
| @@ -170,7 +170,9 @@ const ImageExportModal = ({ | |||||||
|         <Stack.Row gap={2}> |         <Stack.Row gap={2}> | ||||||
|           {actionManager.renderAction("changeExportScale")} |           {actionManager.renderAction("changeExportScale")} | ||||||
|         </Stack.Row> |         </Stack.Row> | ||||||
|         <p style={{ marginLeft: "1em", userSelect: "none" }}>Scale</p> |         <p style={{ marginLeft: "1em", userSelect: "none" }}> | ||||||
|  |           {t("buttons.scale")} | ||||||
|  |         </p> | ||||||
|       </div> |       </div> | ||||||
|       <div |       <div | ||||||
|         style={{ |         style={{ | ||||||
| @@ -229,7 +231,7 @@ export const ImageExportDialog = ({ | |||||||
|   elements: readonly NonDeletedExcalidrawElement[]; |   elements: readonly NonDeletedExcalidrawElement[]; | ||||||
|   files: BinaryFiles; |   files: BinaryFiles; | ||||||
|   exportPadding?: number; |   exportPadding?: number; | ||||||
|   actionManager: ActionsManagerInterface; |   actionManager: ActionManager; | ||||||
|   onExportToPng: ExportCB; |   onExportToPng: ExportCB; | ||||||
|   onExportToSvg: ExportCB; |   onExportToSvg: ExportCB; | ||||||
|   onExportToClipboard: ExportCB; |   onExportToClipboard: ExportCB; | ||||||
| @@ -250,7 +252,7 @@ export const ImageExportDialog = ({ | |||||||
|         icon={exportImage} |         icon={exportImage} | ||||||
|         type="button" |         type="button" | ||||||
|         aria-label={t("buttons.exportImage")} |         aria-label={t("buttons.exportImage")} | ||||||
|         showAriaLabel={useIsMobile()} |         showAriaLabel={useDevice().isMobile} | ||||||
|         title={t("buttons.exportImage")} |         title={t("buttons.exportImage")} | ||||||
|       /> |       /> | ||||||
|       {modalIsShown && ( |       {modalIsShown && ( | ||||||
|   | |||||||
| @@ -14,11 +14,11 @@ export const InitializeApp = (props: Props) => { | |||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const updateLang = async () => { |     const updateLang = async () => { | ||||||
|       await setLanguage(currentLang); |       await setLanguage(currentLang); | ||||||
|  |       setLoading(false); | ||||||
|     }; |     }; | ||||||
|     const currentLang = |     const currentLang = | ||||||
|       languages.find((lang) => lang.code === props.langCode) || defaultLang; |       languages.find((lang) => lang.code === props.langCode) || defaultLang; | ||||||
|     updateLang(); |     updateLang(); | ||||||
|     setLoading(false); |  | ||||||
|   }, [props.langCode]); |   }, [props.langCode]); | ||||||
|  |  | ||||||
|   return loading ? <LoadingMessage /> : props.children; |   return loading ? <LoadingMessage /> : props.children; | ||||||
|   | |||||||
| @@ -1,8 +1,7 @@ | |||||||
| import React, { useState } from "react"; | import React, { useState } from "react"; | ||||||
| import { ActionsManagerInterface } from "../actions/types"; |  | ||||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | import { NonDeletedExcalidrawElement } from "../element/types"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "./App"; | import { useDevice } from "./App"; | ||||||
| import { AppState, ExportOpts, BinaryFiles } from "../types"; | import { AppState, ExportOpts, BinaryFiles } from "../types"; | ||||||
| import { Dialog } from "./Dialog"; | import { Dialog } from "./Dialog"; | ||||||
| import { exportFile, exportToFileIcon, link } from "./icons"; | import { exportFile, exportToFileIcon, link } from "./icons"; | ||||||
| @@ -12,6 +11,9 @@ import { Card } from "./Card"; | |||||||
|  |  | ||||||
| import "./ExportDialog.scss"; | import "./ExportDialog.scss"; | ||||||
| import { nativeFileSystemSupported } from "../data/filesystem"; | import { nativeFileSystemSupported } from "../data/filesystem"; | ||||||
|  | import { trackEvent } from "../analytics"; | ||||||
|  | import { ActionManager } from "../actions/manager"; | ||||||
|  | import { getFrame } from "../utils"; | ||||||
|  |  | ||||||
| export type ExportCB = ( | export type ExportCB = ( | ||||||
|   elements: readonly NonDeletedExcalidrawElement[], |   elements: readonly NonDeletedExcalidrawElement[], | ||||||
| @@ -29,7 +31,7 @@ const JSONExportModal = ({ | |||||||
|   appState: AppState; |   appState: AppState; | ||||||
|   files: BinaryFiles; |   files: BinaryFiles; | ||||||
|   elements: readonly NonDeletedExcalidrawElement[]; |   elements: readonly NonDeletedExcalidrawElement[]; | ||||||
|   actionManager: ActionsManagerInterface; |   actionManager: ActionManager; | ||||||
|   onCloseRequest: () => void; |   onCloseRequest: () => void; | ||||||
|   exportOpts: ExportOpts; |   exportOpts: ExportOpts; | ||||||
|   canvas: HTMLCanvasElement | null; |   canvas: HTMLCanvasElement | null; | ||||||
| @@ -54,7 +56,7 @@ const JSONExportModal = ({ | |||||||
|               aria-label={t("exportDialog.disk_button")} |               aria-label={t("exportDialog.disk_button")} | ||||||
|               showAriaLabel={true} |               showAriaLabel={true} | ||||||
|               onClick={() => { |               onClick={() => { | ||||||
|                 actionManager.executeAction(actionSaveFileToDisk); |                 actionManager.executeAction(actionSaveFileToDisk, "ui"); | ||||||
|               }} |               }} | ||||||
|             /> |             /> | ||||||
|           </Card> |           </Card> | ||||||
| @@ -70,9 +72,10 @@ const JSONExportModal = ({ | |||||||
|               title={t("exportDialog.link_button")} |               title={t("exportDialog.link_button")} | ||||||
|               aria-label={t("exportDialog.link_button")} |               aria-label={t("exportDialog.link_button")} | ||||||
|               showAriaLabel={true} |               showAriaLabel={true} | ||||||
|               onClick={() => |               onClick={() => { | ||||||
|                 onExportToBackend(elements, appState, files, canvas) |                 onExportToBackend(elements, appState, files, canvas); | ||||||
|               } |                 trackEvent("export", "link", `ui (${getFrame()})`); | ||||||
|  |               }} | ||||||
|             /> |             /> | ||||||
|           </Card> |           </Card> | ||||||
|         )} |         )} | ||||||
| @@ -94,7 +97,7 @@ export const JSONExportDialog = ({ | |||||||
|   elements: readonly NonDeletedExcalidrawElement[]; |   elements: readonly NonDeletedExcalidrawElement[]; | ||||||
|   appState: AppState; |   appState: AppState; | ||||||
|   files: BinaryFiles; |   files: BinaryFiles; | ||||||
|   actionManager: ActionsManagerInterface; |   actionManager: ActionManager; | ||||||
|   exportOpts: ExportOpts; |   exportOpts: ExportOpts; | ||||||
|   canvas: HTMLCanvasElement | null; |   canvas: HTMLCanvasElement | null; | ||||||
| }) => { | }) => { | ||||||
| @@ -114,7 +117,7 @@ export const JSONExportDialog = ({ | |||||||
|         icon={exportFile} |         icon={exportFile} | ||||||
|         type="button" |         type="button" | ||||||
|         aria-label={t("buttons.export")} |         aria-label={t("buttons.export")} | ||||||
|         showAriaLabel={useIsMobile()} |         showAriaLabel={useDevice().isMobile} | ||||||
|         title={t("buttons.export")} |         title={t("buttons.export")} | ||||||
|       /> |       /> | ||||||
|       {modalIsShown && ( |       {modalIsShown && ( | ||||||
|   | |||||||
| @@ -1,9 +1,63 @@ | |||||||
| @import "open-color/open-color"; | @import "open-color/open-color"; | ||||||
|  | @import "../css/variables.module"; | ||||||
|  |  | ||||||
|  | .layer-ui__sidebar { | ||||||
|  |   position: absolute; | ||||||
|  |   top: var(--sat); | ||||||
|  |   bottom: var(--sab); | ||||||
|  |   right: var(--sar); | ||||||
|  |   z-index: 5; | ||||||
|  |  | ||||||
|  |   box-shadow: var(--shadow-island); | ||||||
|  |   overflow: hidden; | ||||||
|  |   border-radius: var(--border-radius-lg); | ||||||
|  |   margin: var(--space-factor); | ||||||
|  |   width: calc(#{$right-sidebar-width} - var(--space-factor) * 2); | ||||||
|  |  | ||||||
|  |   .Island { | ||||||
|  |     box-shadow: none; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .ToolIcon__icon { | ||||||
|  |     border-radius: var(--border-radius-md); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .ToolIcon__icon__close { | ||||||
|  |     .Modal__close { | ||||||
|  |       width: calc(var(--space-factor) * 7); | ||||||
|  |       height: calc(var(--space-factor) * 7); | ||||||
|  |       display: flex; | ||||||
|  |       justify-content: center; | ||||||
|  |       align-items: center; | ||||||
|  |       color: var(--color-text); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .Island { | ||||||
|  |     --padding: 0; | ||||||
|  |     background-color: var(--island-bg-color); | ||||||
|  |     border-radius: var(--border-radius-lg); | ||||||
|  |     padding: calc(var(--padding) * var(--space-factor)); | ||||||
|  |     position: relative; | ||||||
|  |     transition: box-shadow 0.5s ease-in-out; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| .excalidraw { | .excalidraw { | ||||||
|  |   .layer-ui__wrapper.animate { | ||||||
|  |     transition: width 0.1s ease-in-out; | ||||||
|  |   } | ||||||
|   .layer-ui__wrapper { |   .layer-ui__wrapper { | ||||||
|  |     // when the rightside sidebar is docked, we need to resize the UI by its | ||||||
|  |     // width, making the nested UI content shift to the left. To do this, | ||||||
|  |     // we need the UI container to actually have dimensions set, but | ||||||
|  |     // then we also need to disable pointer events else the canvas below | ||||||
|  |     // wouldn't be interactive. | ||||||
|  |     position: absolute; | ||||||
|  |     width: 100%; | ||||||
|  |     height: 100%; | ||||||
|  |     pointer-events: none; | ||||||
|     z-index: var(--zIndex-layerUI); |     z-index: var(--zIndex-layerUI); | ||||||
|  |  | ||||||
|     &__top-right { |     &__top-right { | ||||||
|       display: flex; |       display: flex; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,12 +1,11 @@ | |||||||
| import clsx from "clsx"; | import clsx from "clsx"; | ||||||
| import React, { useCallback } from "react"; | import React, { useCallback } from "react"; | ||||||
| import { ActionManager } from "../actions/manager"; | import { ActionManager } from "../actions/manager"; | ||||||
| import { CLASSES } from "../constants"; | import { CLASSES, LIBRARY_SIDEBAR_WIDTH } from "../constants"; | ||||||
| import { exportCanvas } from "../data"; | import { exportCanvas } from "../data"; | ||||||
| import { isTextElement, showSelectedShapeActions } from "../element"; | import { isTextElement, showSelectedShapeActions } from "../element"; | ||||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | import { NonDeletedExcalidrawElement } from "../element/types"; | ||||||
| import { Language, t } from "../i18n"; | import { Language, t } from "../i18n"; | ||||||
| import { useIsMobile } from "../components/App"; |  | ||||||
| import { calculateScrollCenter, getSelectedElements } from "../scene"; | import { calculateScrollCenter, getSelectedElements } from "../scene"; | ||||||
| import { ExportType } from "../scene/types"; | import { ExportType } from "../scene/types"; | ||||||
| import { AppProps, AppState, ExcalidrawProps, BinaryFiles } from "../types"; | import { AppProps, AppState, ExcalidrawProps, BinaryFiles } from "../types"; | ||||||
| @@ -26,9 +25,8 @@ import { PasteChartDialog } from "./PasteChartDialog"; | |||||||
| import { Section } from "./Section"; | import { Section } from "./Section"; | ||||||
| import { HelpDialog } from "./HelpDialog"; | import { HelpDialog } from "./HelpDialog"; | ||||||
| import Stack from "./Stack"; | import Stack from "./Stack"; | ||||||
| import { Tooltip } from "./Tooltip"; |  | ||||||
| import { UserList } from "./UserList"; | import { UserList } from "./UserList"; | ||||||
| import Library from "../data/library"; | import Library, { distributeLibraryItemsOnSquareGrid } from "../data/library"; | ||||||
| import { JSONExportDialog } from "./JSONExportDialog"; | import { JSONExportDialog } from "./JSONExportDialog"; | ||||||
| import { LibraryButton } from "./LibraryButton"; | import { LibraryButton } from "./LibraryButton"; | ||||||
| import { isImageFileHandle } from "../data/blob"; | import { isImageFileHandle } from "../data/blob"; | ||||||
| @@ -37,6 +35,11 @@ import { LibraryMenu } from "./LibraryMenu"; | |||||||
| import "./LayerUI.scss"; | import "./LayerUI.scss"; | ||||||
| import "./Toolbar.scss"; | import "./Toolbar.scss"; | ||||||
| import { PenModeButton } from "./PenModeButton"; | import { PenModeButton } from "./PenModeButton"; | ||||||
|  | import { trackEvent } from "../analytics"; | ||||||
|  | import { useDevice } from "../components/App"; | ||||||
|  | import { Stats } from "./Stats"; | ||||||
|  | import { actionToggleStats } from "../actions/actionToggleStats"; | ||||||
|  | import { actionToggleZenMode } from "../actions"; | ||||||
|  |  | ||||||
| interface LayerUIProps { | interface LayerUIProps { | ||||||
|   actionManager: ActionManager; |   actionManager: ActionManager; | ||||||
| @@ -49,18 +52,13 @@ interface LayerUIProps { | |||||||
|   onLockToggle: () => void; |   onLockToggle: () => void; | ||||||
|   onPenModeToggle: () => void; |   onPenModeToggle: () => void; | ||||||
|   onInsertElements: (elements: readonly NonDeletedExcalidrawElement[]) => void; |   onInsertElements: (elements: readonly NonDeletedExcalidrawElement[]) => void; | ||||||
|   zenModeEnabled: boolean; |  | ||||||
|   showExitZenModeBtn: boolean; |   showExitZenModeBtn: boolean; | ||||||
|   showThemeBtn: boolean; |   showThemeBtn: boolean; | ||||||
|   toggleZenMode: () => void; |  | ||||||
|   langCode: Language["code"]; |   langCode: Language["code"]; | ||||||
|   isCollaborating: boolean; |   isCollaborating: boolean; | ||||||
|   renderTopRightUI?: ( |   renderTopRightUI?: ExcalidrawProps["renderTopRightUI"]; | ||||||
|     isMobile: boolean, |   renderCustomFooter?: ExcalidrawProps["renderFooter"]; | ||||||
|     appState: AppState, |   renderCustomStats?: ExcalidrawProps["renderCustomStats"]; | ||||||
|   ) => JSX.Element | null; |  | ||||||
|   renderCustomFooter?: (isMobile: boolean, appState: AppState) => JSX.Element; |  | ||||||
|   viewModeEnabled: boolean; |  | ||||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; |   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||||
|   UIOptions: AppProps["UIOptions"]; |   UIOptions: AppProps["UIOptions"]; | ||||||
|   focusContainer: () => void; |   focusContainer: () => void; | ||||||
| @@ -68,7 +66,6 @@ interface LayerUIProps { | |||||||
|   id: string; |   id: string; | ||||||
|   onImageAction: (data: { insertOnCanvasDirectly: boolean }) => void; |   onImageAction: (data: { insertOnCanvasDirectly: boolean }) => void; | ||||||
| } | } | ||||||
|  |  | ||||||
| const LayerUI = ({ | const LayerUI = ({ | ||||||
|   actionManager, |   actionManager, | ||||||
|   appState, |   appState, | ||||||
| @@ -80,14 +77,12 @@ const LayerUI = ({ | |||||||
|   onLockToggle, |   onLockToggle, | ||||||
|   onPenModeToggle, |   onPenModeToggle, | ||||||
|   onInsertElements, |   onInsertElements, | ||||||
|   zenModeEnabled, |  | ||||||
|   showExitZenModeBtn, |   showExitZenModeBtn, | ||||||
|   showThemeBtn, |   showThemeBtn, | ||||||
|   toggleZenMode, |  | ||||||
|   isCollaborating, |   isCollaborating, | ||||||
|   renderTopRightUI, |   renderTopRightUI, | ||||||
|   renderCustomFooter, |   renderCustomFooter, | ||||||
|   viewModeEnabled, |   renderCustomStats, | ||||||
|   libraryReturnUrl, |   libraryReturnUrl, | ||||||
|   UIOptions, |   UIOptions, | ||||||
|   focusContainer, |   focusContainer, | ||||||
| @@ -95,7 +90,7 @@ const LayerUI = ({ | |||||||
|   id, |   id, | ||||||
|   onImageAction, |   onImageAction, | ||||||
| }: LayerUIProps) => { | }: LayerUIProps) => { | ||||||
|   const isMobile = useIsMobile(); |   const device = useDevice(); | ||||||
|  |  | ||||||
|   const renderJSONExportDialog = () => { |   const renderJSONExportDialog = () => { | ||||||
|     if (!UIOptions.canvasActions.export) { |     if (!UIOptions.canvasActions.export) { | ||||||
| @@ -122,6 +117,7 @@ const LayerUI = ({ | |||||||
|     const createExporter = |     const createExporter = | ||||||
|       (type: ExportType): ExportCB => |       (type: ExportType): ExportCB => | ||||||
|       async (exportedElements) => { |       async (exportedElements) => { | ||||||
|  |         trackEvent("export", type, "ui"); | ||||||
|         const fileHandle = await exportCanvas( |         const fileHandle = await exportCanvas( | ||||||
|           type, |           type, | ||||||
|           exportedElements, |           exportedElements, | ||||||
| @@ -170,7 +166,7 @@ const LayerUI = ({ | |||||||
|       <Section |       <Section | ||||||
|         heading="canvasActions" |         heading="canvasActions" | ||||||
|         className={clsx("zen-mode-transition", { |         className={clsx("zen-mode-transition", { | ||||||
|           "transition-left": zenModeEnabled, |           "transition-left": appState.zenModeEnabled, | ||||||
|         })} |         })} | ||||||
|       > |       > | ||||||
|         {/* the zIndex ensures this menu has higher stacking order, |         {/* the zIndex ensures this menu has higher stacking order, | ||||||
| @@ -191,7 +187,7 @@ const LayerUI = ({ | |||||||
|     <Section |     <Section | ||||||
|       heading="canvasActions" |       heading="canvasActions" | ||||||
|       className={clsx("zen-mode-transition", { |       className={clsx("zen-mode-transition", { | ||||||
|         "transition-left": zenModeEnabled, |         "transition-left": appState.zenModeEnabled, | ||||||
|       })} |       })} | ||||||
|     > |     > | ||||||
|       {/* the zIndex ensures this menu has higher stacking order, |       {/* the zIndex ensures this menu has higher stacking order, | ||||||
| @@ -231,14 +227,14 @@ const LayerUI = ({ | |||||||
|     <Section |     <Section | ||||||
|       heading="selectedShapeActions" |       heading="selectedShapeActions" | ||||||
|       className={clsx("zen-mode-transition", { |       className={clsx("zen-mode-transition", { | ||||||
|         "transition-left": zenModeEnabled, |         "transition-left": appState.zenModeEnabled, | ||||||
|       })} |       })} | ||||||
|     > |     > | ||||||
|       <Island |       <Island | ||||||
|         className={CLASSES.SHAPE_ACTIONS_MENU} |         className={CLASSES.SHAPE_ACTIONS_MENU} | ||||||
|         padding={2} |         padding={2} | ||||||
|         style={{ |         style={{ | ||||||
|           // we want to make sure this doesn't overflow so substracting 200 |           // we want to make sure this doesn't overflow so subtracting 200 | ||||||
|           // which is approximately height of zoom footer and top left menu items with some buffer |           // 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 |           // if active file name is displayed, subtracting 248 to account for its height | ||||||
|           maxHeight: `${appState.height - (appState.fileHandle ? 248 : 200)}px`, |           maxHeight: `${appState.height - (appState.fileHandle ? 248 : 200)}px`, | ||||||
| @@ -248,7 +244,7 @@ const LayerUI = ({ | |||||||
|           appState={appState} |           appState={appState} | ||||||
|           elements={elements} |           elements={elements} | ||||||
|           renderAction={actionManager.renderAction} |           renderAction={actionManager.renderAction} | ||||||
|           elementType={appState.elementType} |           activeTool={appState.activeTool.type} | ||||||
|         /> |         /> | ||||||
|       </Island> |       </Island> | ||||||
|     </Section> |     </Section> | ||||||
| @@ -275,7 +271,9 @@ const LayerUI = ({ | |||||||
|     <LibraryMenu |     <LibraryMenu | ||||||
|       pendingElements={getSelectedElements(elements, appState, true)} |       pendingElements={getSelectedElements(elements, appState, true)} | ||||||
|       onClose={closeLibrary} |       onClose={closeLibrary} | ||||||
|       onInsertShape={onInsertElements} |       onInsertLibraryItems={(libraryItems) => { | ||||||
|  |         onInsertElements(distributeLibraryItemsOnSquareGrid(libraryItems)); | ||||||
|  |       }} | ||||||
|       onAddToLibrary={deselectItems} |       onAddToLibrary={deselectItems} | ||||||
|       setAppState={setAppState} |       setAppState={setAppState} | ||||||
|       libraryReturnUrl={libraryReturnUrl} |       libraryReturnUrl={libraryReturnUrl} | ||||||
| @@ -299,52 +297,55 @@ const LayerUI = ({ | |||||||
|         <div className="App-menu App-menu_top"> |         <div className="App-menu App-menu_top"> | ||||||
|           <Stack.Col |           <Stack.Col | ||||||
|             gap={4} |             gap={4} | ||||||
|             className={clsx({ "disable-pointerEvents": zenModeEnabled })} |             className={clsx({ | ||||||
|  |               "disable-pointerEvents": appState.zenModeEnabled, | ||||||
|  |             })} | ||||||
|           > |           > | ||||||
|             {viewModeEnabled |             {appState.viewModeEnabled | ||||||
|               ? renderViewModeCanvasActions() |               ? renderViewModeCanvasActions() | ||||||
|               : renderCanvasActions()} |               : renderCanvasActions()} | ||||||
|             {shouldRenderSelectedShapeActions && renderSelectedShapeActions()} |             {shouldRenderSelectedShapeActions && renderSelectedShapeActions()} | ||||||
|           </Stack.Col> |           </Stack.Col> | ||||||
|           {!viewModeEnabled && ( |           {!appState.viewModeEnabled && ( | ||||||
|             <Section heading="shapes"> |             <Section heading="shapes"> | ||||||
|               {(heading) => ( |               {(heading) => ( | ||||||
|                 <Stack.Col gap={4} align="start"> |                 <Stack.Col gap={4} align="start"> | ||||||
|                   <Stack.Row |                   <Stack.Row | ||||||
|                     gap={1} |                     gap={1} | ||||||
|                     className={clsx("App-toolbar-container", { |                     className={clsx("App-toolbar-container", { | ||||||
|                       "zen-mode": zenModeEnabled, |                       "zen-mode": appState.zenModeEnabled, | ||||||
|                     })} |                     })} | ||||||
|                   > |                   > | ||||||
|                     <PenModeButton |                     <PenModeButton | ||||||
|                       zenModeEnabled={zenModeEnabled} |                       zenModeEnabled={appState.zenModeEnabled} | ||||||
|                       checked={appState.penMode} |                       checked={appState.penMode} | ||||||
|                       onChange={onPenModeToggle} |                       onChange={onPenModeToggle} | ||||||
|                       title={t("toolBar.penMode")} |                       title={t("toolBar.penMode")} | ||||||
|                       penDetected={appState.penDetected} |                       penDetected={appState.penDetected} | ||||||
|                     /> |                     /> | ||||||
|                     <LockButton |                     <LockButton | ||||||
|                       zenModeEnabled={zenModeEnabled} |                       zenModeEnabled={appState.zenModeEnabled} | ||||||
|                       checked={appState.elementLocked} |                       checked={appState.activeTool.locked} | ||||||
|                       onChange={onLockToggle} |                       onChange={() => onLockToggle()} | ||||||
|                       title={t("toolBar.lock")} |                       title={t("toolBar.lock")} | ||||||
|                     /> |                     /> | ||||||
|                     <Island |                     <Island | ||||||
|                       padding={1} |                       padding={1} | ||||||
|                       className={clsx("App-toolbar", { |                       className={clsx("App-toolbar", { | ||||||
|                         "zen-mode": zenModeEnabled, |                         "zen-mode": appState.zenModeEnabled, | ||||||
|                       })} |                       })} | ||||||
|                     > |                     > | ||||||
|                       <HintViewer |                       <HintViewer | ||||||
|                         appState={appState} |                         appState={appState} | ||||||
|                         elements={elements} |                         elements={elements} | ||||||
|                         isMobile={isMobile} |                         isMobile={device.isMobile} | ||||||
|                       /> |                       /> | ||||||
|                       {heading} |                       {heading} | ||||||
|                       <Stack.Row gap={1}> |                       <Stack.Row gap={1}> | ||||||
|                         <ShapesSwitcher |                         <ShapesSwitcher | ||||||
|  |                           appState={appState} | ||||||
|                           canvas={canvas} |                           canvas={canvas} | ||||||
|                           elementType={appState.elementType} |                           activeTool={appState.activeTool} | ||||||
|                           setAppState={setAppState} |                           setAppState={setAppState} | ||||||
|                           onImageAction={({ pointerType }) => { |                           onImageAction={({ pointerType }) => { | ||||||
|                             onImageAction({ |                             onImageAction({ | ||||||
| @@ -359,7 +360,6 @@ const LayerUI = ({ | |||||||
|                       setAppState={setAppState} |                       setAppState={setAppState} | ||||||
|                     /> |                     /> | ||||||
|                   </Stack.Row> |                   </Stack.Row> | ||||||
|                   {libraryMenu} |  | ||||||
|                 </Stack.Col> |                 </Stack.Col> | ||||||
|               )} |               )} | ||||||
|             </Section> |             </Section> | ||||||
| @@ -368,27 +368,15 @@ const LayerUI = ({ | |||||||
|             className={clsx( |             className={clsx( | ||||||
|               "layer-ui__wrapper__top-right zen-mode-transition", |               "layer-ui__wrapper__top-right zen-mode-transition", | ||||||
|               { |               { | ||||||
|                 "transition-right": zenModeEnabled, |                 "transition-right": appState.zenModeEnabled, | ||||||
|               }, |               }, | ||||||
|             )} |             )} | ||||||
|           > |           > | ||||||
|             <UserList> |             <UserList | ||||||
|               {appState.collaborators.size > 0 && |               collaborators={appState.collaborators} | ||||||
|                 Array.from(appState.collaborators) |               actionManager={actionManager} | ||||||
|                   // Collaborator is either not initialized or is actually the current user. |             /> | ||||||
|                   .filter(([_, client]) => Object.keys(client).length !== 0) |             {renderTopRightUI?.(device.isMobile, appState)} | ||||||
|                   .map(([clientId, client]) => ( |  | ||||||
|                     <Tooltip |  | ||||||
|                       label={client.username || "Unknown user"} |  | ||||||
|                       key={clientId} |  | ||||||
|                     > |  | ||||||
|                       {actionManager.renderAction("goToCollaborator", { |  | ||||||
|                         id: clientId, |  | ||||||
|                       })} |  | ||||||
|                     </Tooltip> |  | ||||||
|                   ))} |  | ||||||
|             </UserList> |  | ||||||
|             {renderTopRightUI?.(isMobile, appState)} |  | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|       </FixedSideContainer> |       </FixedSideContainer> | ||||||
| @@ -405,7 +393,8 @@ const LayerUI = ({ | |||||||
|           className={clsx( |           className={clsx( | ||||||
|             "layer-ui__wrapper__footer-left zen-mode-transition", |             "layer-ui__wrapper__footer-left zen-mode-transition", | ||||||
|             { |             { | ||||||
|               "layer-ui__wrapper__footer-left--transition-left": zenModeEnabled, |               "layer-ui__wrapper__footer-left--transition-left": | ||||||
|  |                 appState.zenModeEnabled, | ||||||
|             }, |             }, | ||||||
|           )} |           )} | ||||||
|         > |         > | ||||||
| @@ -417,16 +406,39 @@ const LayerUI = ({ | |||||||
|                   zoom={appState.zoom} |                   zoom={appState.zoom} | ||||||
|                 /> |                 /> | ||||||
|               </Island> |               </Island> | ||||||
|               {!viewModeEnabled && ( |               {!appState.viewModeEnabled && ( | ||||||
|  |                 <> | ||||||
|                   <div |                   <div | ||||||
|                     className={clsx("undo-redo-buttons zen-mode-transition", { |                     className={clsx("undo-redo-buttons zen-mode-transition", { | ||||||
|                       "layer-ui__wrapper__footer-left--transition-bottom": |                       "layer-ui__wrapper__footer-left--transition-bottom": | ||||||
|                       zenModeEnabled, |                         appState.zenModeEnabled, | ||||||
|                     })} |                     })} | ||||||
|                   > |                   > | ||||||
|                     {actionManager.renderAction("undo", { size: "small" })} |                     {actionManager.renderAction("undo", { size: "small" })} | ||||||
|                     {actionManager.renderAction("redo", { size: "small" })} |                     {actionManager.renderAction("redo", { size: "small" })} | ||||||
|                   </div> |                   </div> | ||||||
|  |  | ||||||
|  |                   <div | ||||||
|  |                     className={clsx("eraser-buttons zen-mode-transition", { | ||||||
|  |                       "layer-ui__wrapper__footer-left--transition-left": | ||||||
|  |                         appState.zenModeEnabled, | ||||||
|  |                     })} | ||||||
|  |                   > | ||||||
|  |                     {actionManager.renderAction("eraser", { size: "small" })} | ||||||
|  |                   </div> | ||||||
|  |                 </> | ||||||
|  |               )} | ||||||
|  |               {!appState.viewModeEnabled && | ||||||
|  |                 appState.multiElement && | ||||||
|  |                 device.isTouchScreen && ( | ||||||
|  |                   <div | ||||||
|  |                     className={clsx("finalize-button zen-mode-transition", { | ||||||
|  |                       "layer-ui__wrapper__footer-left--transition-left": | ||||||
|  |                         appState.zenModeEnabled, | ||||||
|  |                     })} | ||||||
|  |                   > | ||||||
|  |                     {actionManager.renderAction("finalize", { size: "small" })} | ||||||
|  |                   </div> | ||||||
|                 )} |                 )} | ||||||
|             </Section> |             </Section> | ||||||
|           </Stack.Col> |           </Stack.Col> | ||||||
| @@ -436,7 +448,7 @@ const LayerUI = ({ | |||||||
|             "layer-ui__wrapper__footer-center zen-mode-transition", |             "layer-ui__wrapper__footer-center zen-mode-transition", | ||||||
|             { |             { | ||||||
|               "layer-ui__wrapper__footer-left--transition-bottom": |               "layer-ui__wrapper__footer-left--transition-bottom": | ||||||
|                 zenModeEnabled, |                 appState.zenModeEnabled, | ||||||
|             }, |             }, | ||||||
|           )} |           )} | ||||||
|         > |         > | ||||||
| @@ -446,7 +458,7 @@ const LayerUI = ({ | |||||||
|           className={clsx( |           className={clsx( | ||||||
|             "layer-ui__wrapper__footer-right zen-mode-transition", |             "layer-ui__wrapper__footer-right zen-mode-transition", | ||||||
|             { |             { | ||||||
|               "transition-right disable-pointerEvents": zenModeEnabled, |               "transition-right disable-pointerEvents": appState.zenModeEnabled, | ||||||
|             }, |             }, | ||||||
|           )} |           )} | ||||||
|         > |         > | ||||||
| @@ -456,7 +468,7 @@ const LayerUI = ({ | |||||||
|           className={clsx("disable-zen-mode", { |           className={clsx("disable-zen-mode", { | ||||||
|             "disable-zen-mode--visible": showExitZenModeBtn, |             "disable-zen-mode--visible": showExitZenModeBtn, | ||||||
|           })} |           })} | ||||||
|           onClick={toggleZenMode} |           onClick={() => actionManager.executeAction(actionToggleZenMode)} | ||||||
|         > |         > | ||||||
|           {t("buttons.exitZenMode")} |           {t("buttons.exitZenMode")} | ||||||
|         </button> |         </button> | ||||||
| @@ -466,7 +478,7 @@ const LayerUI = ({ | |||||||
|  |  | ||||||
|   const dialogs = ( |   const dialogs = ( | ||||||
|     <> |     <> | ||||||
|       {appState.isLoading && <LoadingMessage />} |       {appState.isLoading && <LoadingMessage delay={250} />} | ||||||
|       {appState.errorMessage && ( |       {appState.errorMessage && ( | ||||||
|         <ErrorDialog |         <ErrorDialog | ||||||
|           message={appState.errorMessage} |           message={appState.errorMessage} | ||||||
| @@ -495,7 +507,24 @@ const LayerUI = ({ | |||||||
|     </> |     </> | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   return isMobile ? ( |   const renderStats = () => { | ||||||
|  |     if (!appState.showStats) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |     return ( | ||||||
|  |       <Stats | ||||||
|  |         appState={appState} | ||||||
|  |         setAppState={setAppState} | ||||||
|  |         elements={elements} | ||||||
|  |         onClose={() => { | ||||||
|  |           actionManager.executeAction(actionToggleStats); | ||||||
|  |         }} | ||||||
|  |         renderCustomStats={renderCustomStats} | ||||||
|  |       /> | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return device.isMobile ? ( | ||||||
|     <> |     <> | ||||||
|       {dialogs} |       {dialogs} | ||||||
|       <MobileMenu |       <MobileMenu | ||||||
| @@ -507,29 +536,39 @@ const LayerUI = ({ | |||||||
|         renderImageExportDialog={renderImageExportDialog} |         renderImageExportDialog={renderImageExportDialog} | ||||||
|         setAppState={setAppState} |         setAppState={setAppState} | ||||||
|         onCollabButtonClick={onCollabButtonClick} |         onCollabButtonClick={onCollabButtonClick} | ||||||
|         onLockToggle={onLockToggle} |         onLockToggle={() => onLockToggle()} | ||||||
|         onPenModeToggle={onPenModeToggle} |         onPenModeToggle={onPenModeToggle} | ||||||
|         canvas={canvas} |         canvas={canvas} | ||||||
|         isCollaborating={isCollaborating} |         isCollaborating={isCollaborating} | ||||||
|         renderCustomFooter={renderCustomFooter} |         renderCustomFooter={renderCustomFooter} | ||||||
|         viewModeEnabled={viewModeEnabled} |  | ||||||
|         showThemeBtn={showThemeBtn} |         showThemeBtn={showThemeBtn} | ||||||
|         onImageAction={onImageAction} |         onImageAction={onImageAction} | ||||||
|         renderTopRightUI={renderTopRightUI} |         renderTopRightUI={renderTopRightUI} | ||||||
|  |         renderStats={renderStats} | ||||||
|       /> |       /> | ||||||
|     </> |     </> | ||||||
|   ) : ( |   ) : ( | ||||||
|  |     <> | ||||||
|       <div |       <div | ||||||
|         className={clsx("layer-ui__wrapper", { |         className={clsx("layer-ui__wrapper", { | ||||||
|           "disable-pointerEvents": |           "disable-pointerEvents": | ||||||
|             appState.draggingElement || |             appState.draggingElement || | ||||||
|             appState.resizingElement || |             appState.resizingElement || | ||||||
|           (appState.editingElement && !isTextElement(appState.editingElement)), |             (appState.editingElement && | ||||||
|  |               !isTextElement(appState.editingElement)), | ||||||
|         })} |         })} | ||||||
|  |         style={ | ||||||
|  |           appState.isLibraryOpen && | ||||||
|  |           appState.isLibraryMenuDocked && | ||||||
|  |           device.canDeviceFitSidebar | ||||||
|  |             ? { width: `calc(100% - ${LIBRARY_SIDEBAR_WIDTH}px)` } | ||||||
|  |             : {} | ||||||
|  |         } | ||||||
|       > |       > | ||||||
|         {dialogs} |         {dialogs} | ||||||
|         {renderFixedSideContainer()} |         {renderFixedSideContainer()} | ||||||
|         {renderBottomAppMenu()} |         {renderBottomAppMenu()} | ||||||
|  |         {renderStats()} | ||||||
|         {appState.scrolledOutside && ( |         {appState.scrolledOutside && ( | ||||||
|           <button |           <button | ||||||
|             className="scroll-back-to-content" |             className="scroll-back-to-content" | ||||||
| @@ -543,6 +582,10 @@ const LayerUI = ({ | |||||||
|           </button> |           </button> | ||||||
|         )} |         )} | ||||||
|       </div> |       </div> | ||||||
|  |       {appState.isLibraryOpen && ( | ||||||
|  |         <div className="layer-ui__sidebar">{libraryMenu}</div> | ||||||
|  |       )} | ||||||
|  |     </> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,6 +3,8 @@ import clsx from "clsx"; | |||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
| import { capitalizeString } from "../utils"; | import { capitalizeString } from "../utils"; | ||||||
|  | import { trackEvent } from "../analytics"; | ||||||
|  | import { useDevice } from "./App"; | ||||||
|  |  | ||||||
| const LIBRARY_ICON = ( | const LIBRARY_ICON = ( | ||||||
|   <svg viewBox="0 0 576 512"> |   <svg viewBox="0 0 576 512"> | ||||||
| @@ -18,6 +20,7 @@ export const LibraryButton: React.FC<{ | |||||||
|   setAppState: React.Component<any, AppState>["setState"]; |   setAppState: React.Component<any, AppState>["setState"]; | ||||||
|   isMobile?: boolean; |   isMobile?: boolean; | ||||||
| }> = ({ appState, setAppState, isMobile }) => { | }> = ({ appState, setAppState, isMobile }) => { | ||||||
|  |   const device = useDevice(); | ||||||
|   return ( |   return ( | ||||||
|     <label |     <label | ||||||
|       className={clsx( |       className={clsx( | ||||||
| @@ -34,7 +37,19 @@ export const LibraryButton: React.FC<{ | |||||||
|         type="checkbox" |         type="checkbox" | ||||||
|         name="editor-library" |         name="editor-library" | ||||||
|         onChange={(event) => { |         onChange={(event) => { | ||||||
|           setAppState({ isLibraryOpen: event.target.checked }); |           document | ||||||
|  |             .querySelector(".layer-ui__wrapper") | ||||||
|  |             ?.classList.remove("animate"); | ||||||
|  |           const nextState = event.target.checked; | ||||||
|  |           setAppState({ isLibraryOpen: nextState }); | ||||||
|  |           // track only openings | ||||||
|  |           if (nextState) { | ||||||
|  |             trackEvent( | ||||||
|  |               "library", | ||||||
|  |               "toggleLibrary (open)", | ||||||
|  |               `toolbar (${device.isMobile ? "mobile" : "desktop"})`, | ||||||
|  |             ); | ||||||
|  |           } | ||||||
|         }} |         }} | ||||||
|         checked={appState.isLibraryOpen} |         checked={appState.isLibraryOpen} | ||||||
|         aria-label={capitalizeString(t("toolBar.library"))} |         aria-label={capitalizeString(t("toolBar.library"))} | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ | |||||||
|  |  | ||||||
| .excalidraw { | .excalidraw { | ||||||
|   .layer-ui__library { |   .layer-ui__library { | ||||||
|     margin: auto; |  | ||||||
|     display: flex; |     display: flex; | ||||||
|     align-items: center; |     align-items: center; | ||||||
|     justify-content: center; |     justify-content: center; | ||||||
| @@ -11,25 +10,41 @@ | |||||||
|       display: flex; |       display: flex; | ||||||
|       align-items: center; |       align-items: center; | ||||||
|       width: 100%; |       width: 100%; | ||||||
|       margin: 2px 0; |       margin: 2px 0 15px 0; | ||||||
|  |       .Spinner { | ||||||
|  |         margin-right: 1rem; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       button { |       button { | ||||||
|         // 2px from the left to account for focus border of left-most button |         // 2px from the left to account for focus border of left-most button | ||||||
|         margin: 0 2px; |         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__sidebar { | ||||||
|  |     .layer-ui__library { | ||||||
|  |       padding: 0; | ||||||
|  |       height: 100%; | ||||||
|  |     } | ||||||
|  |     .library-menu-items-container { | ||||||
|  |       height: 100%; | ||||||
|  |       width: 100%; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .layer-ui__library-message { |   .layer-ui__library-message { | ||||||
|     padding: 10px 20px; |     padding: 2em 4em; | ||||||
|     max-width: 200px; |     min-width: 200px; | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     align-items: center; | ||||||
|  |     .Spinner { | ||||||
|  |       margin-bottom: 1em; | ||||||
|  |     } | ||||||
|  |     span { | ||||||
|  |       font-size: 0.8em; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .publish-library-success { |   .publish-library-success { | ||||||
| @@ -52,4 +67,38 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   .library-menu-browse-button { | ||||||
|  |     width: 80%; | ||||||
|  |     min-height: 22px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     margin-top: 1rem; | ||||||
|  |     padding: 10px; | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: center; | ||||||
|  |     overflow: hidden; | ||||||
|  |     position: relative; | ||||||
|  |  | ||||||
|  |     border-radius: var(--border-radius-lg); | ||||||
|  |     background-color: var(--color-primary); | ||||||
|  |     color: $oc-white; | ||||||
|  |     text-align: center; | ||||||
|  |     white-space: nowrap; | ||||||
|  |     text-decoration: none !important; | ||||||
|  |     &:hover { | ||||||
|  |       background-color: var(--color-primary-darker); | ||||||
|  |     } | ||||||
|  |     &:active { | ||||||
|  |       background-color: var(--color-primary-darkest); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .library-menu-browse-button--mobile { | ||||||
|  |     min-height: 22px; | ||||||
|  |     margin-left: auto; | ||||||
|  |     a { | ||||||
|  |       padding-right: 0; | ||||||
|  |     } | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,5 +1,12 @@ | |||||||
| import { useRef, useState, useEffect, useCallback, RefObject } from "react"; | import { | ||||||
| import Library from "../data/library"; |   useRef, | ||||||
|  |   useState, | ||||||
|  |   useEffect, | ||||||
|  |   useCallback, | ||||||
|  |   RefObject, | ||||||
|  |   forwardRef, | ||||||
|  | } from "react"; | ||||||
|  | import Library, { libraryItemsAtom } from "../data/library"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { randomId } from "../random"; | import { randomId } from "../random"; | ||||||
| import { | import { | ||||||
| @@ -18,7 +25,11 @@ import "./LibraryMenu.scss"; | |||||||
| import LibraryMenuItems from "./LibraryMenuItems"; | import LibraryMenuItems from "./LibraryMenuItems"; | ||||||
| import { EVENT } from "../constants"; | import { EVENT } from "../constants"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { arrayToMap } from "../utils"; | import { trackEvent } from "../analytics"; | ||||||
|  | import { useAtom } from "jotai"; | ||||||
|  | import { jotaiScope } from "../jotai"; | ||||||
|  | import Spinner from "./Spinner"; | ||||||
|  | import { useDevice } from "./App"; | ||||||
|  |  | ||||||
| const useOnClickOutside = ( | const useOnClickOutside = ( | ||||||
|   ref: RefObject<HTMLElement>, |   ref: RefObject<HTMLElement>, | ||||||
| @@ -53,9 +64,20 @@ const getSelectedItems = ( | |||||||
|   selectedItems: LibraryItem["id"][], |   selectedItems: LibraryItem["id"][], | ||||||
| ) => libraryItems.filter((item) => selectedItems.includes(item.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 = ({ | export const LibraryMenu = ({ | ||||||
|   onClose, |   onClose, | ||||||
|   onInsertShape, |   onInsertLibraryItems, | ||||||
|   pendingElements, |   pendingElements, | ||||||
|   onAddToLibrary, |   onAddToLibrary, | ||||||
|   theme, |   theme, | ||||||
| @@ -69,7 +91,7 @@ export const LibraryMenu = ({ | |||||||
| }: { | }: { | ||||||
|   pendingElements: LibraryItem["elements"]; |   pendingElements: LibraryItem["elements"]; | ||||||
|   onClose: () => void; |   onClose: () => void; | ||||||
|   onInsertShape: (elements: LibraryItem["elements"]) => void; |   onInsertLibraryItems: (libraryItems: LibraryItems) => void; | ||||||
|   onAddToLibrary: () => void; |   onAddToLibrary: () => void; | ||||||
|   theme: AppState["theme"]; |   theme: AppState["theme"]; | ||||||
|   files: BinaryFiles; |   files: BinaryFiles; | ||||||
| @@ -82,17 +104,30 @@ export const LibraryMenu = ({ | |||||||
| }) => { | }) => { | ||||||
|   const ref = useRef<HTMLDivElement | null>(null); |   const ref = useRef<HTMLDivElement | null>(null); | ||||||
|  |  | ||||||
|   useOnClickOutside(ref, (event) => { |   const device = useDevice(); | ||||||
|  |  | ||||||
|  |   useOnClickOutside( | ||||||
|  |     ref, | ||||||
|  |     useCallback( | ||||||
|  |       (event) => { | ||||||
|         // If click on the library icon, do nothing. |         // If click on the library icon, do nothing. | ||||||
|         if ((event.target as Element).closest(".ToolIcon__library")) { |         if ((event.target as Element).closest(".ToolIcon__library")) { | ||||||
|           return; |           return; | ||||||
|         } |         } | ||||||
|  |         if (!appState.isLibraryMenuDocked || !device.canDeviceFitSidebar) { | ||||||
|           onClose(); |           onClose(); | ||||||
|   }); |         } | ||||||
|  |       }, | ||||||
|  |       [onClose, appState.isLibraryMenuDocked, device.canDeviceFitSidebar], | ||||||
|  |     ), | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const handleKeyDown = (event: KeyboardEvent) => { |     const handleKeyDown = (event: KeyboardEvent) => { | ||||||
|       if (event.key === KEYS.ESCAPE) { |       if ( | ||||||
|  |         event.key === KEYS.ESCAPE && | ||||||
|  |         (!appState.isLibraryMenuDocked || !device.canDeviceFitSidebar) | ||||||
|  |       ) { | ||||||
|         onClose(); |         onClose(); | ||||||
|       } |       } | ||||||
|     }; |     }; | ||||||
| @@ -100,13 +135,8 @@ export const LibraryMenu = ({ | |||||||
|     return () => { |     return () => { | ||||||
|       document.removeEventListener(EVENT.KEYDOWN, handleKeyDown); |       document.removeEventListener(EVENT.KEYDOWN, handleKeyDown); | ||||||
|     }; |     }; | ||||||
|   }, [onClose]); |   }, [onClose, appState.isLibraryMenuDocked, device.canDeviceFitSidebar]); | ||||||
|  |  | ||||||
|   const [libraryItems, setLibraryItems] = useState<LibraryItems>([]); |  | ||||||
|  |  | ||||||
|   const [loadingState, setIsLoading] = useState< |  | ||||||
|     "preloading" | "loading" | "ready" |  | ||||||
|   >("preloading"); |  | ||||||
|   const [selectedItems, setSelectedItems] = useState<LibraryItem["id"][]>([]); |   const [selectedItems, setSelectedItems] = useState<LibraryItem["id"][]>([]); | ||||||
|   const [showPublishLibraryDialog, setShowPublishLibraryDialog] = |   const [showPublishLibraryDialog, setShowPublishLibraryDialog] = | ||||||
|     useState(false); |     useState(false); | ||||||
| @@ -114,55 +144,35 @@ export const LibraryMenu = ({ | |||||||
|     url: string; |     url: string; | ||||||
|     authorName: string; |     authorName: string; | ||||||
|   }>(null); |   }>(null); | ||||||
|   const loadingTimerRef = useRef<number | null>(null); |  | ||||||
|  |  | ||||||
|   useEffect(() => { |   const [libraryItemsData] = useAtom(libraryItemsAtom, jotaiScope); | ||||||
|     Promise.race([ |  | ||||||
|       new Promise((resolve) => { |  | ||||||
|         loadingTimerRef.current = window.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 () => { |   const removeFromLibrary = useCallback( | ||||||
|     const items = await library.loadLibrary(); |     async (libraryItems: LibraryItems) => { | ||||||
|  |       const nextItems = libraryItems.filter( | ||||||
|     const nextItems = items.filter((item) => !selectedItems.includes(item.id)); |         (item) => !selectedItems.includes(item.id), | ||||||
|     library.saveLibrary(nextItems).catch((error) => { |       ); | ||||||
|       setLibraryItems(items); |       library.setLibrary(nextItems).catch(() => { | ||||||
|         setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") }); |         setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") }); | ||||||
|       }); |       }); | ||||||
|       setSelectedItems([]); |       setSelectedItems([]); | ||||||
|     setLibraryItems(nextItems); |     }, | ||||||
|   }, [library, setAppState, selectedItems, setSelectedItems]); |     [library, setAppState, selectedItems, setSelectedItems], | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   const resetLibrary = useCallback(() => { |   const resetLibrary = useCallback(() => { | ||||||
|     library.resetLibrary(); |     library.resetLibrary(); | ||||||
|     setLibraryItems([]); |  | ||||||
|     focusContainer(); |     focusContainer(); | ||||||
|   }, [library, focusContainer]); |   }, [library, focusContainer]); | ||||||
|  |  | ||||||
|   const addToLibrary = useCallback( |   const addToLibrary = useCallback( | ||||||
|     async (elements: LibraryItem["elements"]) => { |     async (elements: LibraryItem["elements"], libraryItems: LibraryItems) => { | ||||||
|  |       trackEvent("element", "addToLibrary", "ui"); | ||||||
|       if (elements.some((element) => element.type === "image")) { |       if (elements.some((element) => element.type === "image")) { | ||||||
|         return setAppState({ |         return setAppState({ | ||||||
|           errorMessage: "Support for adding images to the library coming soon!", |           errorMessage: "Support for adding images to the library coming soon!", | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|       const items = await library.loadLibrary(); |  | ||||||
|       const nextItems: LibraryItems = [ |       const nextItems: LibraryItems = [ | ||||||
|         { |         { | ||||||
|           status: "unpublished", |           status: "unpublished", | ||||||
| @@ -170,14 +180,12 @@ export const LibraryMenu = ({ | |||||||
|           id: randomId(), |           id: randomId(), | ||||||
|           created: Date.now(), |           created: Date.now(), | ||||||
|         }, |         }, | ||||||
|         ...items, |         ...libraryItems, | ||||||
|       ]; |       ]; | ||||||
|       onAddToLibrary(); |       onAddToLibrary(); | ||||||
|       library.saveLibrary(nextItems).catch((error) => { |       library.setLibrary(nextItems).catch(() => { | ||||||
|         setLibraryItems(items); |  | ||||||
|         setAppState({ errorMessage: t("alerts.errorAddingToLibrary") }); |         setAppState({ errorMessage: t("alerts.errorAddingToLibrary") }); | ||||||
|       }); |       }); | ||||||
|       setLibraryItems(nextItems); |  | ||||||
|     }, |     }, | ||||||
|     [onAddToLibrary, library, setAppState], |     [onAddToLibrary, library, setAppState], | ||||||
|   ); |   ); | ||||||
| @@ -216,7 +224,7 @@ export const LibraryMenu = ({ | |||||||
|   }, [setPublishLibSuccess, publishLibSuccess]); |   }, [setPublishLibSuccess, publishLibSuccess]); | ||||||
|  |  | ||||||
|   const onPublishLibSuccess = useCallback( |   const onPublishLibSuccess = useCallback( | ||||||
|     (data) => { |     (data, libraryItems: LibraryItems) => { | ||||||
|       setShowPublishLibraryDialog(false); |       setShowPublishLibraryDialog(false); | ||||||
|       setPublishLibSuccess({ url: data.url, authorName: data.authorName }); |       setPublishLibSuccess({ url: data.url, authorName: data.authorName }); | ||||||
|       const nextLibItems = libraryItems.slice(); |       const nextLibItems = libraryItems.slice(); | ||||||
| @@ -225,102 +233,71 @@ export const LibraryMenu = ({ | |||||||
|           libItem.status = "published"; |           libItem.status = "published"; | ||||||
|         } |         } | ||||||
|       }); |       }); | ||||||
|       library.saveLibrary(nextLibItems); |       library.setLibrary(nextLibItems); | ||||||
|       setLibraryItems(nextLibItems); |  | ||||||
|     }, |     }, | ||||||
|     [ |     [setShowPublishLibraryDialog, setPublishLibSuccess, selectedItems, library], | ||||||
|       setShowPublishLibraryDialog, |  | ||||||
|       setPublishLibSuccess, |  | ||||||
|       libraryItems, |  | ||||||
|       selectedItems, |  | ||||||
|       library, |  | ||||||
|     ], |  | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   const [lastSelectedItem, setLastSelectedItem] = useState< |   if ( | ||||||
|     LibraryItem["id"] | null |     libraryItemsData.status === "loading" && | ||||||
|   >(null); |     !libraryItemsData.isInitialized | ||||||
|  |   ) { | ||||||
|  |     return ( | ||||||
|  |       <LibraryMenuWrapper ref={ref}> | ||||||
|  |         <div className="layer-ui__library-message"> | ||||||
|  |           <Spinner size="2em" /> | ||||||
|  |           <span>{t("labels.libraryLoadingMessage")}</span> | ||||||
|  |         </div> | ||||||
|  |       </LibraryMenuWrapper> | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   return loadingState === "preloading" ? null : ( |   return ( | ||||||
|     <Island padding={1} ref={ref} className="layer-ui__library"> |     <LibraryMenuWrapper ref={ref}> | ||||||
|       {showPublishLibraryDialog && ( |       {showPublishLibraryDialog && ( | ||||||
|         <PublishLibrary |         <PublishLibrary | ||||||
|           onClose={() => setShowPublishLibraryDialog(false)} |           onClose={() => setShowPublishLibraryDialog(false)} | ||||||
|           libraryItems={getSelectedItems(libraryItems, selectedItems)} |           libraryItems={getSelectedItems( | ||||||
|  |             libraryItemsData.libraryItems, | ||||||
|  |             selectedItems, | ||||||
|  |           )} | ||||||
|           appState={appState} |           appState={appState} | ||||||
|           onSuccess={onPublishLibSuccess} |           onSuccess={(data) => | ||||||
|  |             onPublishLibSuccess(data, libraryItemsData.libraryItems) | ||||||
|  |           } | ||||||
|           onError={(error) => window.alert(error)} |           onError={(error) => window.alert(error)} | ||||||
|           updateItemsInStorage={() => library.saveLibrary(libraryItems)} |           updateItemsInStorage={() => | ||||||
|  |             library.setLibrary(libraryItemsData.libraryItems) | ||||||
|  |           } | ||||||
|           onRemove={(id: string) => |           onRemove={(id: string) => | ||||||
|             setSelectedItems(selectedItems.filter((_id) => _id !== id)) |             setSelectedItems(selectedItems.filter((_id) => _id !== id)) | ||||||
|           } |           } | ||||||
|         /> |         /> | ||||||
|       )} |       )} | ||||||
|       {publishLibSuccess && renderPublishSuccess()} |       {publishLibSuccess && renderPublishSuccess()} | ||||||
|  |  | ||||||
|       {loadingState === "loading" ? ( |  | ||||||
|         <div className="layer-ui__library-message"> |  | ||||||
|           {t("labels.libraryLoadingMessage")} |  | ||||||
|         </div> |  | ||||||
|       ) : ( |  | ||||||
|       <LibraryMenuItems |       <LibraryMenuItems | ||||||
|           libraryItems={libraryItems} |         isLoading={libraryItemsData.status === "loading"} | ||||||
|           onRemoveFromLibrary={removeFromLibrary} |         libraryItems={libraryItemsData.libraryItems} | ||||||
|           onAddToLibrary={addToLibrary} |         onRemoveFromLibrary={() => | ||||||
|           onInsertShape={onInsertShape} |           removeFromLibrary(libraryItemsData.libraryItems) | ||||||
|  |         } | ||||||
|  |         onAddToLibrary={(elements) => | ||||||
|  |           addToLibrary(elements, libraryItemsData.libraryItems) | ||||||
|  |         } | ||||||
|  |         onInsertLibraryItems={onInsertLibraryItems} | ||||||
|         pendingElements={pendingElements} |         pendingElements={pendingElements} | ||||||
|         setAppState={setAppState} |         setAppState={setAppState} | ||||||
|  |         appState={appState} | ||||||
|         libraryReturnUrl={libraryReturnUrl} |         libraryReturnUrl={libraryReturnUrl} | ||||||
|         library={library} |         library={library} | ||||||
|         theme={theme} |         theme={theme} | ||||||
|         files={files} |         files={files} | ||||||
|         id={id} |         id={id} | ||||||
|         selectedItems={selectedItems} |         selectedItems={selectedItems} | ||||||
|           onToggle={(id, event) => { |         onSelectItems={(ids) => setSelectedItems(ids)} | ||||||
|             const shouldSelect = !selectedItems.includes(id); |  | ||||||
|  |  | ||||||
|             if (shouldSelect) { |  | ||||||
|               if (event.shiftKey && lastSelectedItem) { |  | ||||||
|                 const rangeStart = libraryItems.findIndex( |  | ||||||
|                   (item) => item.id === lastSelectedItem, |  | ||||||
|                 ); |  | ||||||
|                 const rangeEnd = libraryItems.findIndex( |  | ||||||
|                   (item) => item.id === id, |  | ||||||
|                 ); |  | ||||||
|  |  | ||||||
|                 if (rangeStart === -1 || rangeEnd === -1) { |  | ||||||
|                   setSelectedItems([...selectedItems, id]); |  | ||||||
|                   return; |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 const selectedItemsMap = arrayToMap(selectedItems); |  | ||||||
|                 const nextSelectedIds = 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)} |         onPublish={() => setShowPublishLibraryDialog(true)} | ||||||
|         resetLibrary={resetLibrary} |         resetLibrary={resetLibrary} | ||||||
|       /> |       /> | ||||||
|       )} |     </LibraryMenuWrapper> | ||||||
|     </Island> |  | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -2,8 +2,17 @@ | |||||||
|  |  | ||||||
| .excalidraw { | .excalidraw { | ||||||
|   .library-menu-items-container { |   .library-menu-items-container { | ||||||
|     .library-actions { |  | ||||||
|     display: flex; |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     height: 100%; | ||||||
|  |     padding: 0.5rem; | ||||||
|  |     box-sizing: border-box; | ||||||
|  |  | ||||||
|  |     .library-actions { | ||||||
|  |       width: 100%; | ||||||
|  |       display: flex; | ||||||
|  |       margin-right: auto; | ||||||
|  |       align-items: center; | ||||||
|  |  | ||||||
|       button .library-actions-counter { |       button .library-actions-counter { | ||||||
|         position: absolute; |         position: absolute; | ||||||
| @@ -87,12 +96,16 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     &__items { |     &__items { | ||||||
|       max-height: 50vh; |       flex: 1; | ||||||
|       overflow: auto; |       overflow-y: auto; | ||||||
|       margin-top: 0.5rem; |       overflow-x: hidden; | ||||||
|  |       margin-bottom: 1rem; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     .separator { |     .separator { | ||||||
|  |       width: 100%; | ||||||
|  |       display: flex; | ||||||
|  |       align-items: center; | ||||||
|       font-weight: 500; |       font-weight: 500; | ||||||
|       font-size: 0.9rem; |       font-size: 0.9rem; | ||||||
|       margin: 0.6em 0.2em; |       margin: 0.6em 0.2em; | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { chunk } from "lodash"; | import { chunk } from "lodash"; | ||||||
| import { useCallback, useState } from "react"; | import React, { useCallback, useState } from "react"; | ||||||
| import { importLibraryFromJSON, saveLibraryAsJSON } from "../data/json"; | import { saveLibraryAsJSON, serializeLibraryAsJSON } from "../data/json"; | ||||||
| import Library from "../data/library"; | import Library from "../data/library"; | ||||||
| import { ExcalidrawElement, NonDeleted } from "../element/types"; | import { ExcalidrawElement, NonDeleted } from "../element/types"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| @@ -11,48 +11,57 @@ import { | |||||||
|   LibraryItem, |   LibraryItem, | ||||||
|   LibraryItems, |   LibraryItems, | ||||||
| } from "../types"; | } from "../types"; | ||||||
| import { muteFSAbortError } from "../utils"; | import { arrayToMap, muteFSAbortError } from "../utils"; | ||||||
| import { useIsMobile } from "./App"; | import { useDevice } from "./App"; | ||||||
| import ConfirmDialog from "./ConfirmDialog"; | import ConfirmDialog from "./ConfirmDialog"; | ||||||
| import { exportToFileIcon, load, publishIcon, trash } from "./icons"; | import { close, exportToFileIcon, load, publishIcon, trash } from "./icons"; | ||||||
| import { LibraryUnit } from "./LibraryUnit"; | import { LibraryUnit } from "./LibraryUnit"; | ||||||
| import Stack from "./Stack"; | import Stack from "./Stack"; | ||||||
| import { ToolButton } from "./ToolButton"; | import { ToolButton } from "./ToolButton"; | ||||||
| import { Tooltip } from "./Tooltip"; | import { Tooltip } from "./Tooltip"; | ||||||
|  |  | ||||||
| import "./LibraryMenuItems.scss"; | import "./LibraryMenuItems.scss"; | ||||||
| import { VERSIONS } from "../constants"; | import { MIME_TYPES, VERSIONS } from "../constants"; | ||||||
|  | import Spinner from "./Spinner"; | ||||||
|  | import { fileOpen } from "../data/filesystem"; | ||||||
|  |  | ||||||
|  | import { SidebarLockButton } from "./SidebarLockButton"; | ||||||
|  | import { trackEvent } from "../analytics"; | ||||||
|  |  | ||||||
| const LibraryMenuItems = ({ | const LibraryMenuItems = ({ | ||||||
|  |   isLoading, | ||||||
|   libraryItems, |   libraryItems, | ||||||
|   onRemoveFromLibrary, |   onRemoveFromLibrary, | ||||||
|   onAddToLibrary, |   onAddToLibrary, | ||||||
|   onInsertShape, |   onInsertLibraryItems, | ||||||
|   pendingElements, |   pendingElements, | ||||||
|   theme, |   theme, | ||||||
|   setAppState, |   setAppState, | ||||||
|  |   appState, | ||||||
|   libraryReturnUrl, |   libraryReturnUrl, | ||||||
|   library, |   library, | ||||||
|   files, |   files, | ||||||
|   id, |   id, | ||||||
|   selectedItems, |   selectedItems, | ||||||
|   onToggle, |   onSelectItems, | ||||||
|   onPublish, |   onPublish, | ||||||
|   resetLibrary, |   resetLibrary, | ||||||
| }: { | }: { | ||||||
|  |   isLoading: boolean; | ||||||
|   libraryItems: LibraryItems; |   libraryItems: LibraryItems; | ||||||
|   pendingElements: LibraryItem["elements"]; |   pendingElements: LibraryItem["elements"]; | ||||||
|   onRemoveFromLibrary: () => void; |   onRemoveFromLibrary: () => void; | ||||||
|   onInsertShape: (elements: LibraryItem["elements"]) => void; |   onInsertLibraryItems: (libraryItems: LibraryItems) => void; | ||||||
|   onAddToLibrary: (elements: LibraryItem["elements"]) => void; |   onAddToLibrary: (elements: LibraryItem["elements"]) => void; | ||||||
|   theme: AppState["theme"]; |   theme: AppState["theme"]; | ||||||
|   files: BinaryFiles; |   files: BinaryFiles; | ||||||
|   setAppState: React.Component<any, AppState>["setState"]; |   setAppState: React.Component<any, AppState>["setState"]; | ||||||
|  |   appState: AppState; | ||||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; |   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||||
|   library: Library; |   library: Library; | ||||||
|   id: string; |   id: string; | ||||||
|   selectedItems: LibraryItem["id"][]; |   selectedItems: LibraryItem["id"][]; | ||||||
|   onToggle: (id: LibraryItem["id"], event: React.MouseEvent) => void; |   onSelectItems: (id: LibraryItem["id"][]) => void; | ||||||
|   onPublish: () => void; |   onPublish: () => void; | ||||||
|   resetLibrary: () => void; |   resetLibrary: () => void; | ||||||
| }) => { | }) => { | ||||||
| @@ -84,9 +93,7 @@ const LibraryMenuItems = ({ | |||||||
|   }, [selectedItems, onRemoveFromLibrary, resetLibrary]); |   }, [selectedItems, onRemoveFromLibrary, resetLibrary]); | ||||||
|  |  | ||||||
|   const [showRemoveLibAlert, setShowRemoveLibAlert] = useState(false); |   const [showRemoveLibAlert, setShowRemoveLibAlert] = useState(false); | ||||||
|  |   const device = useDevice(); | ||||||
|   const isMobile = useIsMobile(); |  | ||||||
|  |  | ||||||
|   const renderLibraryActions = () => { |   const renderLibraryActions = () => { | ||||||
|     const itemsSelected = !!selectedItems.length; |     const itemsSelected = !!selectedItems.length; | ||||||
|     const items = itemsSelected |     const items = itemsSelected | ||||||
| @@ -97,24 +104,34 @@ const LibraryMenuItems = ({ | |||||||
|       : t("buttons.resetLibrary"); |       : t("buttons.resetLibrary"); | ||||||
|     return ( |     return ( | ||||||
|       <div className="library-actions"> |       <div className="library-actions"> | ||||||
|         {(!itemsSelected || !isMobile) && ( |         {!itemsSelected && ( | ||||||
|           <ToolButton |           <ToolButton | ||||||
|             key="import" |             key="import" | ||||||
|             type="button" |             type="button" | ||||||
|             title={t("buttons.load")} |             title={t("buttons.load")} | ||||||
|             aria-label={t("buttons.load")} |             aria-label={t("buttons.load")} | ||||||
|             icon={load} |             icon={load} | ||||||
|             onClick={() => { |             onClick={async () => { | ||||||
|               importLibraryFromJSON(library) |               try { | ||||||
|                 .then(() => { |                 await library.updateLibrary({ | ||||||
|                   // Close and then open to get the libraries updated |                   libraryItems: fileOpen({ | ||||||
|                   setAppState({ isLibraryOpen: false }); |                     description: "Excalidraw library files", | ||||||
|                   setAppState({ isLibraryOpen: true }); |                     // ToDo: Be over-permissive until https://bugs.webkit.org/show_bug.cgi?id=34442 | ||||||
|                 }) |                     // gets resolved. Else, iOS users cannot open `.excalidraw` files. | ||||||
|                 .catch(muteFSAbortError) |                     /* | ||||||
|                 .catch((error) => { |                     extensions: [".json", ".excalidrawlib"], | ||||||
|                   setAppState({ errorMessage: error.message }); |                     */ | ||||||
|  |                   }), | ||||||
|  |                   merge: true, | ||||||
|  |                   openLibraryMenu: true, | ||||||
|                 }); |                 }); | ||||||
|  |               } catch (error: any) { | ||||||
|  |                 if (error?.name === "AbortError") { | ||||||
|  |                   console.warn(error); | ||||||
|  |                   return; | ||||||
|  |                 } | ||||||
|  |                 setAppState({ errorMessage: t("errors.importLibraryError") }); | ||||||
|  |               } | ||||||
|             }} |             }} | ||||||
|             className="library-actions--load" |             className="library-actions--load" | ||||||
|           /> |           /> | ||||||
| @@ -130,7 +147,7 @@ const LibraryMenuItems = ({ | |||||||
|               onClick={async () => { |               onClick={async () => { | ||||||
|                 const libraryItems = itemsSelected |                 const libraryItems = itemsSelected | ||||||
|                   ? items |                   ? items | ||||||
|                   : await library.loadLibrary(); |                   : await library.getLatestLibrary(); | ||||||
|                 saveLibraryAsJSON(libraryItems) |                 saveLibraryAsJSON(libraryItems) | ||||||
|                   .catch(muteFSAbortError) |                   .catch(muteFSAbortError) | ||||||
|                   .catch((error) => { |                   .catch((error) => { | ||||||
| @@ -162,7 +179,7 @@ const LibraryMenuItems = ({ | |||||||
|             </ToolButton> |             </ToolButton> | ||||||
|           </> |           </> | ||||||
|         )} |         )} | ||||||
|         {itemsSelected && !isPublished && ( |         {itemsSelected && ( | ||||||
|           <Tooltip label={t("hints.publishLibrary")}> |           <Tooltip label={t("hints.publishLibrary")}> | ||||||
|             <ToolButton |             <ToolButton | ||||||
|               type="button" |               type="button" | ||||||
| @@ -172,7 +189,7 @@ const LibraryMenuItems = ({ | |||||||
|               className="library-actions--publish" |               className="library-actions--publish" | ||||||
|               onClick={onPublish} |               onClick={onPublish} | ||||||
|             > |             > | ||||||
|               {!isMobile && <label>{t("buttons.publishLibrary")}</label>} |               {!device.isMobile && <label>{t("buttons.publishLibrary")}</label>} | ||||||
|               {selectedItems.length > 0 && ( |               {selectedItems.length > 0 && ( | ||||||
|                 <span className="library-actions-counter"> |                 <span className="library-actions-counter"> | ||||||
|                   {selectedItems.length} |                   {selectedItems.length} | ||||||
| @@ -181,17 +198,89 @@ const LibraryMenuItems = ({ | |||||||
|             </ToolButton> |             </ToolButton> | ||||||
|           </Tooltip> |           </Tooltip> | ||||||
|         )} |         )} | ||||||
|  |         {device.isMobile && ( | ||||||
|  |           <div className="library-menu-browse-button--mobile"> | ||||||
|  |             <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> | ||||||
|  |         )} | ||||||
|       </div> |       </div> | ||||||
|     ); |     ); | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   const CELLS_PER_ROW = isMobile ? 4 : 6; |   const CELLS_PER_ROW = device.isMobile && !device.isSmScreen ? 6 : 4; | ||||||
|  |  | ||||||
|   const referrer = |   const referrer = | ||||||
|     libraryReturnUrl || window.location.origin + window.location.pathname; |     libraryReturnUrl || window.location.origin + window.location.pathname; | ||||||
|   const isPublished = selectedItems.some( |  | ||||||
|     (id) => libraryItems.find((item) => item.id === id)?.status === "published", |   const [lastSelectedItem, setLastSelectedItem] = useState< | ||||||
|  |     LibraryItem["id"] | null | ||||||
|  |   >(null); | ||||||
|  |  | ||||||
|  |   const onItemSelectToggle = ( | ||||||
|  |     id: LibraryItem["id"], | ||||||
|  |     event: React.MouseEvent, | ||||||
|  |   ) => { | ||||||
|  |     const shouldSelect = !selectedItems.includes(id); | ||||||
|  |  | ||||||
|  |     const orderedItems = [...unpublishedItems, ...publishedItems]; | ||||||
|  |  | ||||||
|  |     if (shouldSelect) { | ||||||
|  |       if (event.shiftKey && lastSelectedItem) { | ||||||
|  |         const rangeStart = orderedItems.findIndex( | ||||||
|  |           (item) => item.id === lastSelectedItem, | ||||||
|         ); |         ); | ||||||
|  |         const rangeEnd = orderedItems.findIndex((item) => item.id === id); | ||||||
|  |  | ||||||
|  |         if (rangeStart === -1 || rangeEnd === -1) { | ||||||
|  |           onSelectItems([...selectedItems, id]); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const selectedItemsMap = arrayToMap(selectedItems); | ||||||
|  |         const nextSelectedIds = orderedItems.reduce( | ||||||
|  |           (acc: LibraryItem["id"][], item, idx) => { | ||||||
|  |             if ( | ||||||
|  |               (idx >= rangeStart && idx <= rangeEnd) || | ||||||
|  |               selectedItemsMap.has(item.id) | ||||||
|  |             ) { | ||||||
|  |               acc.push(item.id); | ||||||
|  |             } | ||||||
|  |             return acc; | ||||||
|  |           }, | ||||||
|  |           [], | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         onSelectItems(nextSelectedIds); | ||||||
|  |       } else { | ||||||
|  |         onSelectItems([...selectedItems, id]); | ||||||
|  |       } | ||||||
|  |       setLastSelectedItem(id); | ||||||
|  |     } else { | ||||||
|  |       setLastSelectedItem(null); | ||||||
|  |       onSelectItems(selectedItems.filter((_id) => _id !== id)); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const getInsertedElements = (id: string) => { | ||||||
|  |     let targetElements; | ||||||
|  |     if (selectedItems.includes(id)) { | ||||||
|  |       targetElements = libraryItems.filter((item) => | ||||||
|  |         selectedItems.includes(item.id), | ||||||
|  |       ); | ||||||
|  |     } else { | ||||||
|  |       targetElements = libraryItems.filter((item) => item.id === id); | ||||||
|  |     } | ||||||
|  |     return targetElements; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|   const createLibraryItemCompo = (params: { |   const createLibraryItemCompo = (params: { | ||||||
|     item: |     item: | ||||||
| @@ -213,8 +302,12 @@ const LibraryMenuItems = ({ | |||||||
|           onClick={params.onClick || (() => {})} |           onClick={params.onClick || (() => {})} | ||||||
|           id={params.item?.id || null} |           id={params.item?.id || null} | ||||||
|           selected={!!params.item?.id && selectedItems.includes(params.item.id)} |           selected={!!params.item?.id && selectedItems.includes(params.item.id)} | ||||||
|           onToggle={(id, event) => { |           onToggle={onItemSelectToggle} | ||||||
|             onToggle(id, event); |           onDrag={(id, event) => { | ||||||
|  |             event.dataTransfer.setData( | ||||||
|  |               MIME_TYPES.excalidrawlib, | ||||||
|  |               serializeLibraryAsJSON(getInsertedElements(id)), | ||||||
|  |             ); | ||||||
|           }} |           }} | ||||||
|         /> |         /> | ||||||
|       </Stack.Col> |       </Stack.Col> | ||||||
| @@ -234,7 +327,7 @@ const LibraryMenuItems = ({ | |||||||
|       if (item.id) { |       if (item.id) { | ||||||
|         return createLibraryItemCompo({ |         return createLibraryItemCompo({ | ||||||
|           item, |           item, | ||||||
|           onClick: () => onInsertShape(item.elements), |           onClick: () => onInsertLibraryItems(getInsertedElements(item.id)), | ||||||
|           key: item.id, |           key: item.id, | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
| @@ -273,23 +366,164 @@ const LibraryMenuItems = ({ | |||||||
|     }); |     }); | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|  |   const unpublishedItems = libraryItems.filter( | ||||||
|  |     (item) => item.status !== "published", | ||||||
|  |   ); | ||||||
|   const publishedItems = libraryItems.filter( |   const publishedItems = libraryItems.filter( | ||||||
|     (item) => item.status === "published", |     (item) => item.status === "published", | ||||||
|   ); |   ); | ||||||
|   const unpublishedItems = [ |  | ||||||
|  |   const renderLibraryHeader = () => { | ||||||
|  |     return ( | ||||||
|  |       <> | ||||||
|  |         <div className="layer-ui__library-header" key="library-header"> | ||||||
|  |           {renderLibraryActions()} | ||||||
|  |           {device.canDeviceFitSidebar && ( | ||||||
|  |             <> | ||||||
|  |               <div className="layer-ui__sidebar-lock-button"> | ||||||
|  |                 <SidebarLockButton | ||||||
|  |                   checked={appState.isLibraryMenuDocked} | ||||||
|  |                   onChange={() => { | ||||||
|  |                     document | ||||||
|  |                       .querySelector(".layer-ui__wrapper") | ||||||
|  |                       ?.classList.add("animate"); | ||||||
|  |                     const nextState = !appState.isLibraryMenuDocked; | ||||||
|  |                     setAppState({ | ||||||
|  |                       isLibraryMenuDocked: nextState, | ||||||
|  |                     }); | ||||||
|  |                     trackEvent( | ||||||
|  |                       "library", | ||||||
|  |                       `toggleLibraryDock (${nextState ? "dock" : "undock"})`, | ||||||
|  |                       `sidebar (${device.isMobile ? "mobile" : "desktop"})`, | ||||||
|  |                     ); | ||||||
|  |                   }} | ||||||
|  |                 /> | ||||||
|  |               </div> | ||||||
|  |             </> | ||||||
|  |           )} | ||||||
|  |           {!device.isMobile && ( | ||||||
|  |             <div className="ToolIcon__icon__close"> | ||||||
|  |               <button | ||||||
|  |                 className="Modal__close" | ||||||
|  |                 onClick={() => | ||||||
|  |                   setAppState({ | ||||||
|  |                     isLibraryOpen: false, | ||||||
|  |                   }) | ||||||
|  |                 } | ||||||
|  |                 aria-label={t("buttons.close")} | ||||||
|  |               > | ||||||
|  |                 {close} | ||||||
|  |               </button> | ||||||
|  |             </div> | ||||||
|  |           )} | ||||||
|  |         </div> | ||||||
|  |       </> | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const renderLibraryMenuItems = () => { | ||||||
|  |     return ( | ||||||
|  |       <Stack.Col | ||||||
|  |         className="library-menu-items-container__items" | ||||||
|  |         align="start" | ||||||
|  |         gap={1} | ||||||
|  |         style={{ | ||||||
|  |           flex: publishedItems.length > 0 ? 1 : "0 1 auto", | ||||||
|  |           marginBottom: 0, | ||||||
|  |         }} | ||||||
|  |       > | ||||||
|  |         <> | ||||||
|  |           <div className="separator"> | ||||||
|  |             {(pendingElements.length > 0 || | ||||||
|  |               unpublishedItems.length > 0 || | ||||||
|  |               publishedItems.length > 0) && ( | ||||||
|  |               <div>{t("labels.personalLib")}</div> | ||||||
|  |             )} | ||||||
|  |             {isLoading && ( | ||||||
|  |               <div | ||||||
|  |                 style={{ | ||||||
|  |                   marginLeft: "auto", | ||||||
|  |                   marginRight: "1rem", | ||||||
|  |                   display: "flex", | ||||||
|  |                   alignItems: "center", | ||||||
|  |                   fontWeight: "normal", | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|  |                 <div style={{ transform: "translateY(2px)" }}> | ||||||
|  |                   <Spinner /> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             )} | ||||||
|  |           </div> | ||||||
|  |           {!pendingElements.length && !unpublishedItems.length ? ( | ||||||
|  |             <div | ||||||
|  |               style={{ | ||||||
|  |                 height: 65, | ||||||
|  |                 display: "flex", | ||||||
|  |                 flexDirection: "column", | ||||||
|  |                 alignItems: "center", | ||||||
|  |                 justifyContent: "center", | ||||||
|  |                 width: "100%", | ||||||
|  |                 fontSize: ".9rem", | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               {t("library.noItems")} | ||||||
|  |               <div | ||||||
|  |                 style={{ | ||||||
|  |                   margin: ".6rem 0", | ||||||
|  |                   fontSize: ".8em", | ||||||
|  |                   width: "70%", | ||||||
|  |                   textAlign: "center", | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|  |                 {publishedItems.length > 0 | ||||||
|  |                   ? t("library.hint_emptyPrivateLibrary") | ||||||
|  |                   : t("library.hint_emptyLibrary")} | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           ) : ( | ||||||
|  |             renderLibrarySection([ | ||||||
|               // append pending library item |               // append pending library item | ||||||
|               ...(pendingElements.length |               ...(pendingElements.length | ||||||
|                 ? [{ id: null, elements: pendingElements }] |                 ? [{ id: null, elements: pendingElements }] | ||||||
|                 : []), |                 : []), | ||||||
|     ...libraryItems.filter((item) => item.status !== "published"), |               ...unpublishedItems, | ||||||
|   ]; |             ]) | ||||||
|  |           )} | ||||||
|  |         </> | ||||||
|  |  | ||||||
|  |         <> | ||||||
|  |           {(publishedItems.length > 0 || | ||||||
|  |             (!device.isMobile && | ||||||
|  |               (pendingElements.length > 0 || unpublishedItems.length > 0))) && ( | ||||||
|  |             <div className="separator">{t("labels.excalidrawLib")}</div> | ||||||
|  |           )} | ||||||
|  |           {publishedItems.length > 0 ? ( | ||||||
|  |             renderLibrarySection(publishedItems) | ||||||
|  |           ) : unpublishedItems.length > 0 ? ( | ||||||
|  |             <div | ||||||
|  |               style={{ | ||||||
|  |                 margin: "1rem 0", | ||||||
|  |                 display: "flex", | ||||||
|  |                 flexDirection: "column", | ||||||
|  |                 alignItems: "center", | ||||||
|  |                 justifyContent: "center", | ||||||
|  |                 width: "100%", | ||||||
|  |                 fontSize: ".9rem", | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               {t("library.noItems")} | ||||||
|  |             </div> | ||||||
|  |           ) : null} | ||||||
|  |         </> | ||||||
|  |       </Stack.Col> | ||||||
|  |     ); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const renderLibraryFooter = () => { | ||||||
|     return ( |     return ( | ||||||
|     <div className="library-menu-items-container"> |  | ||||||
|       {showRemoveLibAlert && renderRemoveLibAlert()} |  | ||||||
|       <div className="layer-ui__library-header" key="library-header"> |  | ||||||
|         {renderLibraryActions()} |  | ||||||
|       <a |       <a | ||||||
|  |         className="library-menu-browse-button" | ||||||
|         href={`${process.env.REACT_APP_LIBRARY_URL}?target=${ |         href={`${process.env.REACT_APP_LIBRARY_URL}?target=${ | ||||||
|           window.name || "_blank" |           window.name || "_blank" | ||||||
|         }&referrer=${referrer}&useHash=true&token=${id}&theme=${theme}&version=${ |         }&referrer=${referrer}&useHash=true&token=${id}&theme=${theme}&version=${ | ||||||
| @@ -299,23 +533,25 @@ const LibraryMenuItems = ({ | |||||||
|       > |       > | ||||||
|         {t("labels.libraries")} |         {t("labels.libraries")} | ||||||
|       </a> |       </a> | ||||||
|       </div> |     ); | ||||||
|       <Stack.Col |   }; | ||||||
|         className="library-menu-items-container__items" |  | ||||||
|         align="start" |   return ( | ||||||
|         gap={1} |     <div | ||||||
|  |       className="library-menu-items-container" | ||||||
|  |       style={ | ||||||
|  |         device.isMobile | ||||||
|  |           ? { | ||||||
|  |               minHeight: "200px", | ||||||
|  |               maxHeight: "70vh", | ||||||
|  |             } | ||||||
|  |           : undefined | ||||||
|  |       } | ||||||
|     > |     > | ||||||
|         <> |       {showRemoveLibAlert && renderRemoveLibAlert()} | ||||||
|           <div className="separator">{t("labels.personalLib")}</div> |       {renderLibraryHeader()} | ||||||
|           {renderLibrarySection(unpublishedItems)} |       {renderLibraryMenuItems()} | ||||||
|         </> |       {!device.isMobile && renderLibraryFooter()} | ||||||
|  |  | ||||||
|         <> |  | ||||||
|           <div className="separator">{t("labels.excalidrawLib")} </div> |  | ||||||
|  |  | ||||||
|           {renderLibrarySection(publishedItems)} |  | ||||||
|         </> |  | ||||||
|       </Stack.Col> |  | ||||||
|     </div> |     </div> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ | |||||||
| .excalidraw { | .excalidraw { | ||||||
|   .library-unit { |   .library-unit { | ||||||
|     align-items: center; |     align-items: center; | ||||||
|     border: 1px solid var(--button-gray-2); |     border: 1px solid transparent; | ||||||
|     display: flex; |     display: flex; | ||||||
|     justify-content: center; |     justify-content: center; | ||||||
|     position: relative; |     position: relative; | ||||||
| @@ -21,10 +21,6 @@ | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   &.theme--dark .library-unit { |  | ||||||
|     border-color: rgb(48, 48, 48); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   .library-unit__dragger { |   .library-unit__dragger { | ||||||
|     display: flex; |     display: flex; | ||||||
|     align-items: center; |     align-items: center; | ||||||
|   | |||||||
| @@ -1,8 +1,7 @@ | |||||||
| import clsx from "clsx"; | import clsx from "clsx"; | ||||||
| import oc from "open-color"; | import oc from "open-color"; | ||||||
| import { useEffect, useRef, useState } from "react"; | import { useEffect, useRef, useState } from "react"; | ||||||
| import { MIME_TYPES } from "../constants"; | import { useDevice } from "../components/App"; | ||||||
| import { useIsMobile } from "../components/App"; |  | ||||||
| import { exportToSvg } from "../scene/export"; | import { exportToSvg } from "../scene/export"; | ||||||
| import { BinaryFiles, LibraryItem } from "../types"; | import { BinaryFiles, LibraryItem } from "../types"; | ||||||
| import "./LibraryUnit.scss"; | import "./LibraryUnit.scss"; | ||||||
| @@ -29,6 +28,7 @@ export const LibraryUnit = ({ | |||||||
|   onClick, |   onClick, | ||||||
|   selected, |   selected, | ||||||
|   onToggle, |   onToggle, | ||||||
|  |   onDrag, | ||||||
| }: { | }: { | ||||||
|   id: LibraryItem["id"] | /** for pending item */ null; |   id: LibraryItem["id"] | /** for pending item */ null; | ||||||
|   elements?: LibraryItem["elements"]; |   elements?: LibraryItem["elements"]; | ||||||
| @@ -37,6 +37,7 @@ export const LibraryUnit = ({ | |||||||
|   onClick: () => void; |   onClick: () => void; | ||||||
|   selected: boolean; |   selected: boolean; | ||||||
|   onToggle: (id: string, event: React.MouseEvent) => void; |   onToggle: (id: string, event: React.MouseEvent) => void; | ||||||
|  |   onDrag: (id: string, event: React.DragEvent) => void; | ||||||
| }) => { | }) => { | ||||||
|   const ref = useRef<HTMLDivElement | null>(null); |   const ref = useRef<HTMLDivElement | null>(null); | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
| @@ -66,7 +67,7 @@ export const LibraryUnit = ({ | |||||||
|   }, [elements, files]); |   }, [elements, files]); | ||||||
|  |  | ||||||
|   const [isHovered, setIsHovered] = useState(false); |   const [isHovered, setIsHovered] = useState(false); | ||||||
|   const isMobile = useIsMobile(); |   const isMobile = useDevice().isMobile; | ||||||
|   const adder = isPending && ( |   const adder = isPending && ( | ||||||
|     <div className="library-unit__adder">{PLUS_ICON}</div> |     <div className="library-unit__adder">{PLUS_ICON}</div> | ||||||
|   ); |   ); | ||||||
| @@ -99,11 +100,12 @@ export const LibraryUnit = ({ | |||||||
|             : undefined |             : undefined | ||||||
|         } |         } | ||||||
|         onDragStart={(event) => { |         onDragStart={(event) => { | ||||||
|  |           if (!id) { | ||||||
|  |             event.preventDefault(); | ||||||
|  |             return; | ||||||
|  |           } | ||||||
|           setIsHovered(false); |           setIsHovered(false); | ||||||
|           event.dataTransfer.setData( |           onDrag(id, event); | ||||||
|             MIME_TYPES.excalidrawlib, |  | ||||||
|             JSON.stringify(elements), |  | ||||||
|           ); |  | ||||||
|         }} |         }} | ||||||
|       /> |       /> | ||||||
|       {adder} |       {adder} | ||||||
|   | |||||||
| @@ -1,10 +1,30 @@ | |||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
|  | import { useState, useEffect } from "react"; | ||||||
|  | import Spinner from "./Spinner"; | ||||||
|  |  | ||||||
|  | export const LoadingMessage: React.FC<{ delay?: number }> = ({ delay }) => { | ||||||
|  |   const [isWaiting, setIsWaiting] = useState(!!delay); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (!delay) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     const timer = setTimeout(() => { | ||||||
|  |       setIsWaiting(false); | ||||||
|  |     }, delay); | ||||||
|  |     return () => clearTimeout(timer); | ||||||
|  |   }, [delay]); | ||||||
|  |  | ||||||
|  |   if (isWaiting) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
| export const LoadingMessage = () => { |  | ||||||
|   // !! KEEP THIS IN SYNC WITH index.html !! |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="LoadingMessage"> |     <div className="LoadingMessage"> | ||||||
|       <span>{t("labels.loadingScene")}</span> |       <div> | ||||||
|  |         <Spinner /> | ||||||
|  |       </div> | ||||||
|  |       <div className="LoadingMessage-text">{t("labels.loadingScene")}</div> | ||||||
|     </div> |     </div> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import { NonDeletedExcalidrawElement } from "../element/types"; | |||||||
| import { FixedSideContainer } from "./FixedSideContainer"; | import { FixedSideContainer } from "./FixedSideContainer"; | ||||||
| import { Island } from "./Island"; | import { Island } from "./Island"; | ||||||
| import { HintViewer } from "./HintViewer"; | import { HintViewer } from "./HintViewer"; | ||||||
| import { calculateScrollCenter } from "../scene"; | import { calculateScrollCenter, getSelectedElements } from "../scene"; | ||||||
| import { SelectedShapeActions, ShapesSwitcher } from "./Actions"; | import { SelectedShapeActions, ShapesSwitcher } from "./Actions"; | ||||||
| import { Section } from "./Section"; | import { Section } from "./Section"; | ||||||
| import CollabButton from "./CollabButton"; | import CollabButton from "./CollabButton"; | ||||||
| @@ -32,14 +32,17 @@ type MobileMenuProps = { | |||||||
|   onPenModeToggle: () => void; |   onPenModeToggle: () => void; | ||||||
|   canvas: HTMLCanvasElement | null; |   canvas: HTMLCanvasElement | null; | ||||||
|   isCollaborating: boolean; |   isCollaborating: boolean; | ||||||
|   renderCustomFooter?: (isMobile: boolean, appState: AppState) => JSX.Element; |   renderCustomFooter?: ( | ||||||
|   viewModeEnabled: boolean; |     isMobile: boolean, | ||||||
|  |     appState: AppState, | ||||||
|  |   ) => JSX.Element | null; | ||||||
|   showThemeBtn: boolean; |   showThemeBtn: boolean; | ||||||
|   onImageAction: (data: { insertOnCanvasDirectly: boolean }) => void; |   onImageAction: (data: { insertOnCanvasDirectly: boolean }) => void; | ||||||
|   renderTopRightUI?: ( |   renderTopRightUI?: ( | ||||||
|     isMobile: boolean, |     isMobile: boolean, | ||||||
|     appState: AppState, |     appState: AppState, | ||||||
|   ) => JSX.Element | null; |   ) => JSX.Element | null; | ||||||
|  |   renderStats: () => JSX.Element | null; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const MobileMenu = ({ | export const MobileMenu = ({ | ||||||
| @@ -56,10 +59,10 @@ export const MobileMenu = ({ | |||||||
|   canvas, |   canvas, | ||||||
|   isCollaborating, |   isCollaborating, | ||||||
|   renderCustomFooter, |   renderCustomFooter, | ||||||
|   viewModeEnabled, |  | ||||||
|   showThemeBtn, |   showThemeBtn, | ||||||
|   onImageAction, |   onImageAction, | ||||||
|   renderTopRightUI, |   renderTopRightUI, | ||||||
|  |   renderStats, | ||||||
| }: MobileMenuProps) => { | }: MobileMenuProps) => { | ||||||
|   const renderToolbar = () => { |   const renderToolbar = () => { | ||||||
|     return ( |     return ( | ||||||
| @@ -72,8 +75,9 @@ export const MobileMenu = ({ | |||||||
|                   {heading} |                   {heading} | ||||||
|                   <Stack.Row gap={1}> |                   <Stack.Row gap={1}> | ||||||
|                     <ShapesSwitcher |                     <ShapesSwitcher | ||||||
|  |                       appState={appState} | ||||||
|                       canvas={canvas} |                       canvas={canvas} | ||||||
|                       elementType={appState.elementType} |                       activeTool={appState.activeTool} | ||||||
|                       setAppState={setAppState} |                       setAppState={setAppState} | ||||||
|                       onImageAction={({ pointerType }) => { |                       onImageAction={({ pointerType }) => { | ||||||
|                         onImageAction({ |                         onImageAction({ | ||||||
| @@ -85,7 +89,7 @@ export const MobileMenu = ({ | |||||||
|                 </Island> |                 </Island> | ||||||
|                 {renderTopRightUI && renderTopRightUI(true, appState)} |                 {renderTopRightUI && renderTopRightUI(true, appState)} | ||||||
|                 <LockButton |                 <LockButton | ||||||
|                   checked={appState.elementLocked} |                   checked={appState.activeTool.locked} | ||||||
|                   onChange={onLockToggle} |                   onChange={onLockToggle} | ||||||
|                   title={t("toolBar.lock")} |                   title={t("toolBar.lock")} | ||||||
|                   isMobile |                   isMobile | ||||||
| @@ -113,19 +117,29 @@ export const MobileMenu = ({ | |||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   const renderAppToolbar = () => { |   const renderAppToolbar = () => { | ||||||
|     if (viewModeEnabled) { |     // Render eraser conditionally in mobile | ||||||
|  |     const showEraser = | ||||||
|  |       !appState.viewModeEnabled && | ||||||
|  |       !appState.editingElement && | ||||||
|  |       getSelectedElements(elements, appState).length === 0; | ||||||
|  |  | ||||||
|  |     if (appState.viewModeEnabled) { | ||||||
|       return ( |       return ( | ||||||
|         <div className="App-toolbar-content"> |         <div className="App-toolbar-content"> | ||||||
|           {actionManager.renderAction("toggleCanvasMenu")} |           {actionManager.renderAction("toggleCanvasMenu")} | ||||||
|         </div> |         </div> | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|       <div className="App-toolbar-content"> |       <div className="App-toolbar-content"> | ||||||
|         {actionManager.renderAction("toggleCanvasMenu")} |         {actionManager.renderAction("toggleCanvasMenu")} | ||||||
|         {actionManager.renderAction("toggleEditMenu")} |         {actionManager.renderAction("toggleEditMenu")} | ||||||
|  |  | ||||||
|         {actionManager.renderAction("undo")} |         {actionManager.renderAction("undo")} | ||||||
|         {actionManager.renderAction("redo")} |         {actionManager.renderAction("redo")} | ||||||
|  |         {showEraser && actionManager.renderAction("eraser")} | ||||||
|  |  | ||||||
|         {actionManager.renderAction( |         {actionManager.renderAction( | ||||||
|           appState.multiElement ? "finalize" : "duplicateSelection", |           appState.multiElement ? "finalize" : "duplicateSelection", | ||||||
|         )} |         )} | ||||||
| @@ -135,7 +149,7 @@ export const MobileMenu = ({ | |||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   const renderCanvasActions = () => { |   const renderCanvasActions = () => { | ||||||
|     if (viewModeEnabled) { |     if (appState.viewModeEnabled) { | ||||||
|       return ( |       return ( | ||||||
|         <> |         <> | ||||||
|           {renderJSONExportDialog()} |           {renderJSONExportDialog()} | ||||||
| @@ -169,7 +183,8 @@ export const MobileMenu = ({ | |||||||
|   }; |   }; | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       {!viewModeEnabled && renderToolbar()} |       {!appState.viewModeEnabled && renderToolbar()} | ||||||
|  |       {renderStats()} | ||||||
|       <div |       <div | ||||||
|         className="App-bottom-bar" |         className="App-bottom-bar" | ||||||
|         style={{ |         style={{ | ||||||
| @@ -188,40 +203,33 @@ export const MobileMenu = ({ | |||||||
|                   {appState.collaborators.size > 0 && ( |                   {appState.collaborators.size > 0 && ( | ||||||
|                     <fieldset> |                     <fieldset> | ||||||
|                       <legend>{t("labels.collaborators")}</legend> |                       <legend>{t("labels.collaborators")}</legend> | ||||||
|                       <UserList mobile> |                       <UserList | ||||||
|                         {Array.from(appState.collaborators) |                         mobile | ||||||
|                           // Collaborator is either not initialized or is actually the current user. |                         collaborators={appState.collaborators} | ||||||
|                           .filter( |                         actionManager={actionManager} | ||||||
|                             ([_, client]) => Object.keys(client).length !== 0, |                       /> | ||||||
|                           ) |  | ||||||
|                           .map(([clientId, client]) => ( |  | ||||||
|                             <React.Fragment key={clientId}> |  | ||||||
|                               {actionManager.renderAction("goToCollaborator", { |  | ||||||
|                                 id: clientId, |  | ||||||
|                               })} |  | ||||||
|                             </React.Fragment> |  | ||||||
|                           ))} |  | ||||||
|                       </UserList> |  | ||||||
|                     </fieldset> |                     </fieldset> | ||||||
|                   )} |                   )} | ||||||
|                 </Stack.Col> |                 </Stack.Col> | ||||||
|               </div> |               </div> | ||||||
|             </Section> |             </Section> | ||||||
|           ) : appState.openMenu === "shape" && |           ) : appState.openMenu === "shape" && | ||||||
|             !viewModeEnabled && |             !appState.viewModeEnabled && | ||||||
|             showSelectedShapeActions(appState, elements) ? ( |             showSelectedShapeActions(appState, elements) ? ( | ||||||
|             <Section className="App-mobile-menu" heading="selectedShapeActions"> |             <Section className="App-mobile-menu" heading="selectedShapeActions"> | ||||||
|               <SelectedShapeActions |               <SelectedShapeActions | ||||||
|                 appState={appState} |                 appState={appState} | ||||||
|                 elements={elements} |                 elements={elements} | ||||||
|                 renderAction={actionManager.renderAction} |                 renderAction={actionManager.renderAction} | ||||||
|                 elementType={appState.elementType} |                 activeTool={appState.activeTool.type} | ||||||
|               /> |               /> | ||||||
|             </Section> |             </Section> | ||||||
|           ) : null} |           ) : null} | ||||||
|           <footer className="App-toolbar"> |           <footer className="App-toolbar"> | ||||||
|             {renderAppToolbar()} |             {renderAppToolbar()} | ||||||
|             {appState.scrolledOutside && !appState.openMenu && ( |             {appState.scrolledOutside && | ||||||
|  |               !appState.openMenu && | ||||||
|  |               !appState.isLibraryOpen && ( | ||||||
|                 <button |                 <button | ||||||
|                   className="scroll-back-to-content" |                   className="scroll-back-to-content" | ||||||
|                   onClick={() => { |                   onClick={() => { | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import React, { useState, useLayoutEffect, useRef } from "react"; | |||||||
| import { createPortal } from "react-dom"; | import { createPortal } from "react-dom"; | ||||||
| import clsx from "clsx"; | import clsx from "clsx"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { useExcalidrawContainer, useIsMobile } from "./App"; | import { useExcalidrawContainer, useDevice } from "./App"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
| import { THEME } from "../constants"; | import { THEME } from "../constants"; | ||||||
|  |  | ||||||
| @@ -59,17 +59,17 @@ export const Modal = (props: { | |||||||
| const useBodyRoot = (theme: AppState["theme"]) => { | const useBodyRoot = (theme: AppState["theme"]) => { | ||||||
|   const [div, setDiv] = useState<HTMLDivElement | null>(null); |   const [div, setDiv] = useState<HTMLDivElement | null>(null); | ||||||
|  |  | ||||||
|   const isMobile = useIsMobile(); |   const device = useDevice(); | ||||||
|   const isMobileRef = useRef(isMobile); |   const isMobileRef = useRef(device.isMobile); | ||||||
|   isMobileRef.current = isMobile; |   isMobileRef.current = device.isMobile; | ||||||
|  |  | ||||||
|   const { container: excalidrawContainer } = useExcalidrawContainer(); |   const { container: excalidrawContainer } = useExcalidrawContainer(); | ||||||
|  |  | ||||||
|   useLayoutEffect(() => { |   useLayoutEffect(() => { | ||||||
|     if (div) { |     if (div) { | ||||||
|       div.classList.toggle("excalidraw--mobile", isMobile); |       div.classList.toggle("excalidraw--mobile", device.isMobile); | ||||||
|     } |     } | ||||||
|   }, [div, isMobile]); |   }, [div, device.isMobile]); | ||||||
|  |  | ||||||
|   useLayoutEffect(() => { |   useLayoutEffect(() => { | ||||||
|     const isDarkTheme = |     const isDarkTheme = | ||||||
|   | |||||||
| @@ -2,5 +2,6 @@ | |||||||
|   .popover { |   .popover { | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     z-index: 10; |     z-index: 10; | ||||||
|  |     padding: 5px 0 5px; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
| import React, { useLayoutEffect, useRef, useEffect } from "react"; | import React, { useLayoutEffect, useRef, useEffect } from "react"; | ||||||
| import "./Popover.scss"; | import "./Popover.scss"; | ||||||
| import { unstable_batchedUpdates } from "react-dom"; | import { unstable_batchedUpdates } from "react-dom"; | ||||||
|  | import { queryFocusableElements } from "../utils"; | ||||||
|  | import { KEYS } from "../keys"; | ||||||
|  |  | ||||||
| type Props = { | type Props = { | ||||||
|   top?: number; |   top?: number; | ||||||
| @@ -27,17 +29,67 @@ export const Popover = ({ | |||||||
| }: Props) => { | }: Props) => { | ||||||
|   const popoverRef = useRef<HTMLDivElement>(null); |   const popoverRef = useRef<HTMLDivElement>(null); | ||||||
|  |  | ||||||
|  |   const container = popoverRef.current; | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (!container) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const handleKeyDown = (event: KeyboardEvent) => { | ||||||
|  |       if (event.key === KEYS.TAB) { | ||||||
|  |         const focusableElements = queryFocusableElements(container); | ||||||
|  |         const { activeElement } = document; | ||||||
|  |         const currentIndex = focusableElements.findIndex( | ||||||
|  |           (element) => element === activeElement, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         if (currentIndex === 0 && event.shiftKey) { | ||||||
|  |           focusableElements[focusableElements.length - 1].focus(); | ||||||
|  |           event.preventDefault(); | ||||||
|  |           event.stopImmediatePropagation(); | ||||||
|  |         } else if ( | ||||||
|  |           currentIndex === focusableElements.length - 1 && | ||||||
|  |           !event.shiftKey | ||||||
|  |         ) { | ||||||
|  |           focusableElements[0].focus(); | ||||||
|  |           event.preventDefault(); | ||||||
|  |           event.stopImmediatePropagation(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     container.addEventListener("keydown", handleKeyDown); | ||||||
|  |  | ||||||
|  |     return () => container.removeEventListener("keydown", handleKeyDown); | ||||||
|  |   }, [container]); | ||||||
|  |  | ||||||
|   // ensure the popover doesn't overflow the viewport |   // ensure the popover doesn't overflow the viewport | ||||||
|   useLayoutEffect(() => { |   useLayoutEffect(() => { | ||||||
|     if (fitInViewport && popoverRef.current) { |     if (fitInViewport && popoverRef.current) { | ||||||
|       const element = popoverRef.current; |       const element = popoverRef.current; | ||||||
|       const { x, y, width, height } = element.getBoundingClientRect(); |       const { x, y, width, height } = element.getBoundingClientRect(); | ||||||
|  |       const { innerWidth: viewportWidth, innerHeight: viewportHeight } = window; | ||||||
|  |  | ||||||
|  |       //Position correctly when clicked on rightmost part or the bottom part of viewport | ||||||
|       if (x + width - offsetLeft > viewportWidth) { |       if (x + width - offsetLeft > viewportWidth) { | ||||||
|         element.style.left = `${viewportWidth - width}px`; |         element.style.left = `${viewportWidth - width - 10}px`; | ||||||
|       } |       } | ||||||
|       if (y + height - offsetTop > viewportHeight) { |       if (y + height - offsetTop > viewportHeight) { | ||||||
|         element.style.top = `${viewportHeight - height}px`; |         element.style.top = `${viewportHeight - height}px`; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       //Resize to fit viewport on smaller screens | ||||||
|  |       if (height >= viewportHeight) { | ||||||
|  |         element.style.height = `${viewportHeight - 20}px`; | ||||||
|  |         element.style.top = "10px"; | ||||||
|  |         element.style.overflowY = "scroll"; | ||||||
|  |       } | ||||||
|  |       if (width >= viewportWidth) { | ||||||
|  |         element.style.width = `${viewportWidth}px`; | ||||||
|  |         element.style.left = "0px"; | ||||||
|  |         element.style.overflowX = "scroll"; | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   }, [fitInViewport, viewportWidth, viewportHeight, offsetLeft, offsetTop]); |   }, [fitInViewport, viewportWidth, viewportHeight, offsetLeft, offsetTop]); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -82,6 +82,10 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     &-warning { | ||||||
|  |       color: $oc-red-6; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     &-note { |     &-note { | ||||||
|       padding: 1em; |       padding: 1em; | ||||||
|       font-style: italic; |       font-style: italic; | ||||||
|   | |||||||
| @@ -295,6 +295,11 @@ const PublishLibrary = ({ | |||||||
|   }, [clonedLibItems, onClose, updateItemsInStorage, libraryData]); |   }, [clonedLibItems, onClose, updateItemsInStorage, libraryData]); | ||||||
|  |  | ||||||
|   const shouldRenderForm = !!libraryItems.length; |   const shouldRenderForm = !!libraryItems.length; | ||||||
|  |  | ||||||
|  |   const containsPublishedItems = libraryItems.some( | ||||||
|  |     (item) => item.status === "published", | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Dialog |     <Dialog | ||||||
|       onCloseRequest={onDialogClose} |       onCloseRequest={onDialogClose} | ||||||
| @@ -329,6 +334,11 @@ const PublishLibrary = ({ | |||||||
|           <div className="publish-library-note"> |           <div className="publish-library-note"> | ||||||
|             {t("publishDialog.noteItems")} |             {t("publishDialog.noteItems")} | ||||||
|           </div> |           </div> | ||||||
|  |           {containsPublishedItems && ( | ||||||
|  |             <span className="publish-library-note publish-library-warning"> | ||||||
|  |               {t("publishDialog.republishWarning")} | ||||||
|  |             </span> | ||||||
|  |           )} | ||||||
|           {renderLibraryItems()} |           {renderLibraryItems()} | ||||||
|           <div className="publish-library__fields"> |           <div className="publish-library__fields"> | ||||||
|             <label> |             <label> | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								src/components/SidebarLockButton.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/components/SidebarLockButton.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | @import "../css/variables.module"; | ||||||
|  |  | ||||||
|  | .excalidraw { | ||||||
|  |   .layer-ui__sidebar-lock-button { | ||||||
|  |     @include toolbarButtonColorStates; | ||||||
|  |     margin-right: 0.2rem; | ||||||
|  |   } | ||||||
|  |   .ToolIcon_type_floating .side_lock_icon { | ||||||
|  |     width: calc(var(--space-factor) * 7); | ||||||
|  |     height: calc(var(--space-factor) * 7); | ||||||
|  |     svg { | ||||||
|  |       // mirror | ||||||
|  |       transform: scale(-1, 1); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .ToolIcon_type_checkbox { | ||||||
|  |     &:not(.ToolIcon_toggle_opaque):checked + .side_lock_icon { | ||||||
|  |       background-color: var(--color-primary); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										46
									
								
								src/components/SidebarLockButton.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								src/components/SidebarLockButton.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | import "./ToolIcon.scss"; | ||||||
|  |  | ||||||
|  | import React from "react"; | ||||||
|  | import clsx from "clsx"; | ||||||
|  | import { ToolButtonSize } from "./ToolButton"; | ||||||
|  | import { t } from "../i18n"; | ||||||
|  | import { Tooltip } from "./Tooltip"; | ||||||
|  |  | ||||||
|  | import "./SidebarLockButton.scss"; | ||||||
|  |  | ||||||
|  | type SidebarLockIconProps = { | ||||||
|  |   checked: boolean; | ||||||
|  |   onChange?(): void; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const DEFAULT_SIZE: ToolButtonSize = "medium"; | ||||||
|  |  | ||||||
|  | const SIDE_LIBRARY_TOGGLE_ICON = ( | ||||||
|  |   <svg viewBox="0 0 24 24" fill="#ffffff"> | ||||||
|  |     <path d="M19 22H5a3 3 0 01-3-3V5a3 3 0 013-3h14a3 3 0 013 3v14a3 3 0 01-3 3zm0-18h-9v16h9a1.01 1.01 0 001-1V5a1.01 1.01 0 00-1-1z"></path> | ||||||
|  |   </svg> | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | export const SidebarLockButton = (props: SidebarLockIconProps) => { | ||||||
|  |   return ( | ||||||
|  |     <Tooltip label={t("labels.sidebarLock")}> | ||||||
|  |       <label | ||||||
|  |         className={clsx( | ||||||
|  |           "ToolIcon ToolIcon__lock ToolIcon_type_floating", | ||||||
|  |           `ToolIcon_size_${DEFAULT_SIZE}`, | ||||||
|  |         )} | ||||||
|  |       > | ||||||
|  |         <input | ||||||
|  |           className="ToolIcon_type_checkbox" | ||||||
|  |           type="checkbox" | ||||||
|  |           onChange={props.onChange} | ||||||
|  |           checked={props.checked} | ||||||
|  |           aria-label={t("labels.sidebarLock")} | ||||||
|  |         />{" "} | ||||||
|  |         <div className="ToolIcon__icon side_lock_icon" tabIndex={0}> | ||||||
|  |           {SIDE_LIBRARY_TOGGLE_ICON} | ||||||
|  |         </div>{" "} | ||||||
|  |       </label>{" "} | ||||||
|  |     </Tooltip> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
| @@ -3,11 +3,24 @@ | |||||||
| .excalidraw { | .excalidraw { | ||||||
|   .single-library-item { |   .single-library-item { | ||||||
|     position: relative; |     position: relative; | ||||||
|  |  | ||||||
|  |     &-status { | ||||||
|  |       position: absolute; | ||||||
|  |       top: 0.3rem; | ||||||
|  |       left: 0.3rem; | ||||||
|  |       font-size: 0.7rem; | ||||||
|  |       color: $oc-red-7; | ||||||
|  |       background: rgba(255, 255, 255, 0.9); | ||||||
|  |       padding: 0.1rem 0.2rem; | ||||||
|  |       border-radius: 0.2rem; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     &__svg { |     &__svg { | ||||||
|  |       background-color: $oc-white; | ||||||
|  |       padding: 0.3rem; | ||||||
|       width: 7.5rem; |       width: 7.5rem; | ||||||
|       height: 7.5rem; |       height: 7.5rem; | ||||||
|       border: 1px solid var(--button-gray-2); |       border: 1px solid var(--button-gray-2); | ||||||
|       margin: 0.3rem; |  | ||||||
|       svg { |       svg { | ||||||
|         width: 100%; |         width: 100%; | ||||||
|         height: 100%; |         height: 100%; | ||||||
| @@ -40,7 +53,7 @@ | |||||||
|     &--remove { |     &--remove { | ||||||
|       position: absolute; |       position: absolute; | ||||||
|       top: 0.2rem; |       top: 0.2rem; | ||||||
|       right: 1.3rem; |       right: 1rem; | ||||||
|  |  | ||||||
|       .ToolIcon__icon { |       .ToolIcon__icon { | ||||||
|         margin: 0; |         margin: 0; | ||||||
|   | |||||||
| @@ -45,6 +45,11 @@ const SingleLibraryItem = ({ | |||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="single-library-item"> |     <div className="single-library-item"> | ||||||
|  |       {libItem.status === "published" && ( | ||||||
|  |         <span className="single-library-item-status"> | ||||||
|  |           {t("labels.statusPublished")} | ||||||
|  |         </span> | ||||||
|  |       )} | ||||||
|       <div ref={svgRef} className="single-library-item__svg" /> |       <div ref={svgRef} className="single-library-item__svg" /> | ||||||
|       <ToolButton |       <ToolButton | ||||||
|         aria-label={t("buttons.remove")} |         aria-label={t("buttons.remove")} | ||||||
|   | |||||||
| @@ -41,6 +41,7 @@ const ColStack = ({ | |||||||
|   align, |   align, | ||||||
|   justifyContent, |   justifyContent, | ||||||
|   className, |   className, | ||||||
|  |   style, | ||||||
| }: StackProps) => { | }: StackProps) => { | ||||||
|   return ( |   return ( | ||||||
|     <div |     <div | ||||||
| @@ -49,6 +50,7 @@ const ColStack = ({ | |||||||
|         "--gap": gap, |         "--gap": gap, | ||||||
|         justifyItems: align, |         justifyItems: align, | ||||||
|         justifyContent, |         justifyContent, | ||||||
|  |         ...style, | ||||||
|       }} |       }} | ||||||
|     > |     > | ||||||
|       {children} |       {children} | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ | |||||||
|     right: 12px; |     right: 12px; | ||||||
|     font-size: 12px; |     font-size: 12px; | ||||||
|     z-index: 10; |     z-index: 10; | ||||||
|  |     pointer-events: all; | ||||||
|  |  | ||||||
|     h3 { |     h3 { | ||||||
|       margin: 0 24px 8px 0; |       margin: 0 24px 8px 0; | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import React from "react"; | |||||||
| import { getCommonBounds } from "../element/bounds"; | import { getCommonBounds } from "../element/bounds"; | ||||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | import { NonDeletedExcalidrawElement } from "../element/types"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "../components/App"; | import { useDevice } from "../components/App"; | ||||||
| import { getTargetElements } from "../scene"; | import { getTargetElements } from "../scene"; | ||||||
| import { AppState, ExcalidrawProps } from "../types"; | import { AppState, ExcalidrawProps } from "../types"; | ||||||
| import { close } from "./icons"; | import { close } from "./icons"; | ||||||
| @@ -16,16 +16,13 @@ export const Stats = (props: { | |||||||
|   onClose: () => void; |   onClose: () => void; | ||||||
|   renderCustomStats: ExcalidrawProps["renderCustomStats"]; |   renderCustomStats: ExcalidrawProps["renderCustomStats"]; | ||||||
| }) => { | }) => { | ||||||
|   const isMobile = useIsMobile(); |   const device = useDevice(); | ||||||
|  |  | ||||||
|   const boundingBox = getCommonBounds(props.elements); |   const boundingBox = getCommonBounds(props.elements); | ||||||
|   const selectedElements = getTargetElements(props.elements, props.appState); |   const selectedElements = getTargetElements(props.elements, props.appState); | ||||||
|   const selectedBoundingBox = getCommonBounds(selectedElements); |   const selectedBoundingBox = getCommonBounds(selectedElements); | ||||||
|  |   if (device.isMobile && props.appState.openMenu) { | ||||||
|   if (isMobile && props.appState.openMenu) { |  | ||||||
|     return null; |     return null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="Stats"> |     <div className="Stats"> | ||||||
|       <Island padding={2}> |       <Island padding={2}> | ||||||
|   | |||||||
| @@ -2,6 +2,9 @@ | |||||||
|  |  | ||||||
| .excalidraw { | .excalidraw { | ||||||
|   .Toast { |   .Toast { | ||||||
|  |     $closeButtonSize: 1.2rem; | ||||||
|  |     $closeButtonPadding: 0.4rem; | ||||||
|  |  | ||||||
|     animation: fade-in 0.5s; |     animation: fade-in 0.5s; | ||||||
|     background-color: var(--button-gray-1); |     background-color: var(--button-gray-1); | ||||||
|     border-radius: 4px; |     border-radius: 4px; | ||||||
| @@ -15,13 +18,26 @@ | |||||||
|     text-align: center; |     text-align: center; | ||||||
|     width: 300px; |     width: 300px; | ||||||
|     z-index: 999999; |     z-index: 999999; | ||||||
|   } |  | ||||||
|  |  | ||||||
|     .Toast__message { |     .Toast__message { | ||||||
|  |       padding: 0 $closeButtonSize + ($closeButtonPadding); | ||||||
|       color: var(--popup-text-color); |       color: var(--popup-text-color); | ||||||
|       white-space: pre-wrap; |       white-space: pre-wrap; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     .close { | ||||||
|  |       position: absolute; | ||||||
|  |       top: 0; | ||||||
|  |       right: 0; | ||||||
|  |       padding: $closeButtonPadding; | ||||||
|  |  | ||||||
|  |       .ToolIcon__icon { | ||||||
|  |         width: $closeButtonSize; | ||||||
|  |         height: $closeButtonSize; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   @keyframes fade-in { |   @keyframes fade-in { | ||||||
|     from { |     from { | ||||||
|       opacity: 0; |       opacity: 0; | ||||||
|   | |||||||
| @@ -1,34 +1,59 @@ | |||||||
| import { useCallback, useEffect, useRef } from "react"; | import { useCallback, useEffect, useRef } from "react"; | ||||||
| import { TOAST_TIMEOUT } from "../constants"; | import { close } from "./icons"; | ||||||
| import "./Toast.scss"; | import "./Toast.scss"; | ||||||
|  | import { ToolButton } from "./ToolButton"; | ||||||
|  |  | ||||||
|  | const DEFAULT_TOAST_TIMEOUT = 5000; | ||||||
|  |  | ||||||
| export const Toast = ({ | export const Toast = ({ | ||||||
|   message, |   message, | ||||||
|   clearToast, |   onClose, | ||||||
|  |   closable = false, | ||||||
|  |   // To prevent autoclose, pass duration as Infinity | ||||||
|  |   duration = DEFAULT_TOAST_TIMEOUT, | ||||||
| }: { | }: { | ||||||
|   message: string; |   message: string; | ||||||
|   clearToast: () => void; |   onClose: () => void; | ||||||
|  |   closable?: boolean; | ||||||
|  |   duration?: number; | ||||||
| }) => { | }) => { | ||||||
|   const timerRef = useRef<number>(0); |   const timerRef = useRef<number>(0); | ||||||
|  |   const shouldAutoClose = duration !== Infinity; | ||||||
|   const scheduleTimeout = useCallback( |   const scheduleTimeout = useCallback(() => { | ||||||
|     () => |     if (!shouldAutoClose) { | ||||||
|       (timerRef.current = window.setTimeout(() => clearToast(), TOAST_TIMEOUT)), |       return; | ||||||
|     [clearToast], |     } | ||||||
|   ); |     timerRef.current = window.setTimeout(() => onClose(), duration); | ||||||
|  |   }, [onClose, duration, shouldAutoClose]); | ||||||
|  |  | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|  |     if (!shouldAutoClose) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|     scheduleTimeout(); |     scheduleTimeout(); | ||||||
|     return () => clearTimeout(timerRef.current); |     return () => clearTimeout(timerRef.current); | ||||||
|   }, [scheduleTimeout, message]); |   }, [scheduleTimeout, message, duration, shouldAutoClose]); | ||||||
|  |  | ||||||
|  |   const onMouseEnter = shouldAutoClose | ||||||
|  |     ? () => clearTimeout(timerRef?.current) | ||||||
|  |     : undefined; | ||||||
|  |   const onMouseLeave = shouldAutoClose ? scheduleTimeout : undefined; | ||||||
|   return ( |   return ( | ||||||
|     <div |     <div | ||||||
|       className="Toast" |       className="Toast" | ||||||
|       onMouseEnter={() => clearTimeout(timerRef?.current)} |       onMouseEnter={onMouseEnter} | ||||||
|       onMouseLeave={scheduleTimeout} |       onMouseLeave={onMouseLeave} | ||||||
|     > |     > | ||||||
|       <p className="Toast__message">{message}</p> |       <p className="Toast__message">{message}</p> | ||||||
|  |       {closable && ( | ||||||
|  |         <ToolButton | ||||||
|  |           icon={close} | ||||||
|  |           aria-label="close" | ||||||
|  |           type="icon" | ||||||
|  |           onClick={onClose} | ||||||
|  |           className="close" | ||||||
|  |         /> | ||||||
|  |       )} | ||||||
|     </div> |     </div> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -48,6 +48,7 @@ type ToolButtonProps = | |||||||
|       type: "radio"; |       type: "radio"; | ||||||
|       checked: boolean; |       checked: boolean; | ||||||
|       onChange?(data: { pointerType: PointerType | null }): void; |       onChange?(data: { pointerType: PointerType | null }): void; | ||||||
|  |       onPointerDown?(data: { pointerType: PointerType }): void; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
| export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => { | export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => { | ||||||
| @@ -149,6 +150,7 @@ export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => { | |||||||
|       title={props.title} |       title={props.title} | ||||||
|       onPointerDown={(event) => { |       onPointerDown={(event) => { | ||||||
|         lastPointerTypeRef.current = event.pointerType || null; |         lastPointerTypeRef.current = event.pointerType || null; | ||||||
|  |         props.onPointerDown?.({ pointerType: event.pointerType || null }); | ||||||
|       }} |       }} | ||||||
|       onPointerUp={() => { |       onPointerUp={() => { | ||||||
|         requestAnimationFrame(() => { |         requestAnimationFrame(() => { | ||||||
|   | |||||||
| @@ -155,7 +155,7 @@ | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       width: 2rem; |       width: 2rem; | ||||||
|       height: 2em; |       height: 2rem; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -212,16 +212,14 @@ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     .ToolIcon.ToolIcon__library { |     .ToolIcon.ToolIcon__library { | ||||||
|       top: 100px; |       top: calc(var(--sat) + 100px); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     .ToolIcon.ToolIcon__lock { |     .ToolIcon.ToolIcon__lock { | ||||||
|       margin-inline-end: 0; |       top: calc(var(--sat) + 60px); | ||||||
|       top: 60px; |  | ||||||
|     } |     } | ||||||
|     .ToolIcon.ToolIcon__penMode { |     .ToolIcon.ToolIcon__penMode { | ||||||
|       margin-inline-end: 0; |       top: calc(var(--sat) + 140px); | ||||||
|       top: 140px; |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,26 +1,5 @@ | |||||||
| @import "open-color/open-color.scss"; | @import "open-color/open-color.scss"; | ||||||
|  | @import "../css/variables.module"; | ||||||
| @mixin toolbarButtonColorStates { |  | ||||||
|   .ToolIcon_type_radio, |  | ||||||
|   .ToolIcon_type_checkbox { |  | ||||||
|     & + .ToolIcon__icon:active { |  | ||||||
|       background: var(--color-primary-light); |  | ||||||
|     } |  | ||||||
|     &:checked + .ToolIcon__icon { |  | ||||||
|       background: var(--color-primary); |  | ||||||
|       --icon-fill-color: #{$oc-white}; |  | ||||||
|       --keybinding-color: #{$oc-white}; |  | ||||||
|     } |  | ||||||
|     &:checked + .ToolIcon__icon:active { |  | ||||||
|       background: var(--color-primary-darker); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   .ToolIcon__keybinding { |  | ||||||
|     bottom: 4px; |  | ||||||
|     right: 4px; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .excalidraw { | .excalidraw { | ||||||
|   .App-toolbar-container { |   .App-toolbar-container { | ||||||
| @@ -53,7 +32,6 @@ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     .ToolIcon.ToolIcon__lock { |     .ToolIcon.ToolIcon__lock { | ||||||
|       margin-inline-end: var(--space-factor); |  | ||||||
|       &.ToolIcon_type_floating { |       &.ToolIcon_type_floating { | ||||||
|         margin-left: 0.1rem; |         margin-left: 0.1rem; | ||||||
|       } |       } | ||||||
| @@ -87,8 +65,14 @@ | |||||||
|  |  | ||||||
|     .ToolIcon { |     .ToolIcon { | ||||||
|       &:hover { |       &:hover { | ||||||
|         --icon-fill-color: var(--color-primary-chubb); |         --icon-fill-color: var( | ||||||
|         --keybinding-color: var(--color-primary-chubb); |           --color-primary-contrast-offset, | ||||||
|  |           var(--color-primary) | ||||||
|  |         ); | ||||||
|  |         --keybinding-color: var( | ||||||
|  |           --color-primary-contrast-offset, | ||||||
|  |           var(--color-primary) | ||||||
|  |         ); | ||||||
|       } |       } | ||||||
|       &:active { |       &:active { | ||||||
|         --icon-fill-color: #{$oc-gray-9}; |         --icon-fill-color: #{$oc-gray-9}; | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| // container in body where the actual tooltip is appended to | // container in body where the actual tooltip is appended to | ||||||
| .excalidraw-tooltip { | .excalidraw-tooltip { | ||||||
|   position: absolute; |   position: fixed; | ||||||
|   z-index: 1000; |   z-index: 1000; | ||||||
|  |  | ||||||
|   padding: 8px; |   padding: 8px; | ||||||
|   | |||||||
| @@ -2,17 +2,51 @@ import "./UserList.scss"; | |||||||
|  |  | ||||||
| import React from "react"; | import React from "react"; | ||||||
| import clsx from "clsx"; | import clsx from "clsx"; | ||||||
|  | import { AppState, Collaborator } from "../types"; | ||||||
|  | import { Tooltip } from "./Tooltip"; | ||||||
|  | import { ActionManager } from "../actions/manager"; | ||||||
|  |  | ||||||
| type UserListProps = { | export const UserList: React.FC<{ | ||||||
|   children: React.ReactNode; |  | ||||||
|   className?: string; |   className?: string; | ||||||
|   mobile?: boolean; |   mobile?: boolean; | ||||||
| }; |   collaborators: AppState["collaborators"]; | ||||||
|  |   actionManager: ActionManager; | ||||||
|  | }> = ({ className, mobile, collaborators, actionManager }) => { | ||||||
|  |   const uniqueCollaborators = new Map<string, Collaborator>(); | ||||||
|  |  | ||||||
|  |   collaborators.forEach((collaborator, socketId) => { | ||||||
|  |     uniqueCollaborators.set( | ||||||
|  |       // filter on user id, else fall back on unique socketId | ||||||
|  |       collaborator.id || socketId, | ||||||
|  |       collaborator, | ||||||
|  |     ); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const avatars = | ||||||
|  |     uniqueCollaborators.size > 0 && | ||||||
|  |     Array.from(uniqueCollaborators) | ||||||
|  |       .filter(([_, client]) => Object.keys(client).length !== 0) | ||||||
|  |       .map(([clientId, collaborator]) => { | ||||||
|  |         const avatarJSX = actionManager.renderAction("goToCollaborator", [ | ||||||
|  |           clientId, | ||||||
|  |           collaborator, | ||||||
|  |         ]); | ||||||
|  |  | ||||||
|  |         return mobile ? ( | ||||||
|  |           <Tooltip | ||||||
|  |             label={collaborator.username || "Unknown user"} | ||||||
|  |             key={clientId} | ||||||
|  |           > | ||||||
|  |             {avatarJSX} | ||||||
|  |           </Tooltip> | ||||||
|  |         ) : ( | ||||||
|  |           <React.Fragment key={clientId}>{avatarJSX}</React.Fragment> | ||||||
|  |         ); | ||||||
|  |       }); | ||||||
|  |  | ||||||
| export const UserList = ({ children, className, mobile }: UserListProps) => { |  | ||||||
|   return ( |   return ( | ||||||
|     <div className={clsx("UserList", className, { UserList_mobile: mobile })}> |     <div className={clsx("UserList", className, { UserList_mobile: mobile })}> | ||||||
|       {children} |       {avatars} | ||||||
|     </div> |     </div> | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -885,6 +885,40 @@ export const TextAlignRightIcon = React.memo(({ theme }: { theme: Theme }) => | |||||||
|   ), |   ), | ||||||
| ); | ); | ||||||
|  |  | ||||||
|  | export const TextAlignTopIcon = React.memo(({ theme }: { theme: Theme }) => | ||||||
|  |   createIcon( | ||||||
|  |     <path | ||||||
|  |       d="m16,132l416,0c8.837,0 16,-7.163 16,-16l0,-40c0,-8.837 -7.163,-16 -16,-16l-416,0c-8.837,0 -16,7.163 -16,16l0,40c0,8.837 7.163,16 16,16zm0,160l416,0c8.837,0 16,-7.163 16,-16l0,-40c0,-8.837 -7.163,-16 -16,-16l-416,0c-8.837,0 -16,7.163 -16,16l0,40c0,8.837 7.163,16 16,16z" | ||||||
|  |       fill={iconFillColor(theme)} | ||||||
|  |       strokeLinecap="round" | ||||||
|  |     />, | ||||||
|  |     { width: 448, height: 512 }, | ||||||
|  |   ), | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | export const TextAlignBottomIcon = React.memo(({ theme }: { theme: Theme }) => | ||||||
|  |   createIcon( | ||||||
|  |     <path | ||||||
|  |       d="M16,292L432,292C440.837,292 448,284.837 448,276L448,236C448,227.163 440.837,220 432,220L16,220C7.163,220 0,227.163 0,236L0,276C0,284.837 7.163,292 16,292ZM16,452L432,452C440.837,452 448,444.837 448,436L448,396C448,387.163 440.837,380 432,380L16,380C7.163,380 0,387.163 0,396L0,436C0,444.837 7.163,452 16,452Z" | ||||||
|  |       fill={iconFillColor(theme)} | ||||||
|  |       strokeLinecap="round" | ||||||
|  |     />, | ||||||
|  |     { width: 448, height: 512 }, | ||||||
|  |   ), | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | export const TextAlignMiddleIcon = React.memo(({ theme }: { theme: Theme }) => | ||||||
|  |   createIcon( | ||||||
|  |     <path | ||||||
|  |       transform="matrix(1,0,0,1,0,80)" | ||||||
|  |       d="M16,132L432,132C440.837,132 448,124.837 448,116L448,76C448,67.163 440.837,60 432,60L16,60C7.163,60 0,67.163 0,76L0,116C0,124.837 7.163,132 16,132ZM16,292L432,292C440.837,292 448,284.837 448,276L448,236C448,227.163 440.837,220 432,220L16,220C7.163,220 0,227.163 0,236L0,276C0,284.837 7.163,292 16,292Z" | ||||||
|  |       fill={iconFillColor(theme)} | ||||||
|  |       strokeLinecap="round" | ||||||
|  |     />, | ||||||
|  |     { width: 448, height: 512 }, | ||||||
|  |   ), | ||||||
|  | ); | ||||||
|  |  | ||||||
| export const publishIcon = createIcon( | export const publishIcon = createIcon( | ||||||
|   <path |   <path | ||||||
|     d="M537.6 226.6c4.1-10.7 6.4-22.4 6.4-34.6 0-53-43-96-96-96-19.7 0-38.1 6-53.3 16.2C367 64.2 315.3 32 256 32c-88.4 0-160 71.6-160 160 0 2.7.1 5.4.2 8.1C40.2 219.8 0 273.2 0 336c0 79.5 64.5 144 144 144h368c70.7 0 128-57.3 128-128 0-61.9-44-113.6-102.4-125.4zM393.4 288H328v112c0 8.8-7.2 16-16 16h-48c-8.8 0-16-7.2-16-16V288h-65.4c-14.3 0-21.4-17.2-11.3-27.3l105.4-105.4c6.2-6.2 16.4-6.2 22.6 0l105.4 105.4c10.1 10.1 2.9 27.3-11.3 27.3z" |     d="M537.6 226.6c4.1-10.7 6.4-22.4 6.4-34.6 0-53-43-96-96-96-19.7 0-38.1 6-53.3 16.2C367 64.2 315.3 32 256 32c-88.4 0-160 71.6-160 160 0 2.7.1 5.4.2 8.1C40.2 219.8 0 273.2 0 336c0 79.5 64.5 144 144 144h368c70.7 0 128-57.3 128-128 0-61.9-44-113.6-102.4-125.4zM393.4 288H328v112c0 8.8-7.2 16-16 16h-48c-8.8 0-16-7.2-16-16V288h-65.4c-14.3 0-21.4-17.2-11.3-27.3l105.4-105.4c6.2-6.2 16.4-6.2 22.6 0l105.4 105.4c10.1 10.1 2.9 27.3-11.3 27.3z" | ||||||
| @@ -900,3 +934,7 @@ export const editIcon = createIcon( | |||||||
|   ></path>, |   ></path>, | ||||||
|   { width: 640, height: 512 }, |   { width: 640, height: 512 }, | ||||||
| ); | ); | ||||||
|  |  | ||||||
|  | export const eraser = createIcon( | ||||||
|  |   <path d="M480 416C497.7 416 512 430.3 512 448C512 465.7 497.7 480 480 480H150.6C133.7 480 117.4 473.3 105.4 461.3L25.37 381.3C.3786 356.3 .3786 315.7 25.37 290.7L258.7 57.37C283.7 32.38 324.3 32.38 349.3 57.37L486.6 194.7C511.6 219.7 511.6 260.3 486.6 285.3L355.9 416H480zM265.4 416L332.7 348.7L195.3 211.3L70.63 336L150.6 416L265.4 416z" />, | ||||||
|  | ); | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user