Compare commits
	
		
			1 Commits
		
	
	
		
			expose_app
			...
			aakansha-b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 68636236a1 | 
| @@ -1,10 +1,10 @@ | |||||||
| * | * | ||||||
| !.env |  | ||||||
| !.eslintrc.json |  | ||||||
| !.npmrc |  | ||||||
| !.prettierrc |  | ||||||
| !package.json |  | ||||||
| !public/ | !public/ | ||||||
| !src/ | !src/ | ||||||
|  | !.npmrc | ||||||
|  | !.eslintrc.json | ||||||
|  | !.prettierrc | ||||||
|  | !package-lock.json | ||||||
|  | !package.json | ||||||
| !tsconfig.json | !tsconfig.json | ||||||
| !yarn.lock | !.env | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.env
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,5 @@ | |||||||
| REACT_APP_BACKEND_V1_GET_URL=https://json.excalidraw.com/api/v1/ | REACT_APP_BACKEND_V1_GET_URL=https://json.excalidraw.com/api/v1/ | ||||||
| REACT_APP_BACKEND_V2_GET_URL=https://json.excalidraw.com/api/v2/ | REACT_APP_BACKEND_V2_GET_URL=https://json.excalidraw.com/api/v2/ | ||||||
| REACT_APP_BACKEND_V2_POST_URL=https://json.excalidraw.com/api/v2/post/ | REACT_APP_BACKEND_V2_POST_URL=https://json.excalidraw.com/api/v2/post/ | ||||||
| REACT_APP_SOCKET_SERVER_URL=https://portal.excalidraw.com | REACT_APP_SOCKET_SERVER_URL=https://excalidraw-socket.herokuapp.com | ||||||
| REACT_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyAd15pYlMci_xIp9ko6wkEsDzAAA0Dn0RU","authDomain":"excalidraw-room-persistence.firebaseapp.com","databaseURL":"https://excalidraw-room-persistence.firebaseio.com","projectId":"excalidraw-room-persistence","storageBucket":"excalidraw-room-persistence.appspot.com","messagingSenderId":"654800341332","appId":"1:654800341332:web:4a692de832b55bd57ce0c1"}' | REACT_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyAd15pYlMci_xIp9ko6wkEsDzAAA0Dn0RU","authDomain":"excalidraw-room-persistence.firebaseapp.com","databaseURL":"https://excalidraw-room-persistence.firebaseio.com","projectId":"excalidraw-room-persistence","storageBucket":"excalidraw-room-persistence.appspot.com","messagingSenderId":"654800341332","appId":"1:654800341332:web:4a692de832b55bd57ce0c1"}' | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| REACT_APP_GOOGLE_ANALYTICS_ID=UA-387204-13 | REACT_APP_INCLUDE_GTAG=true | ||||||
|   | |||||||
| @@ -4,5 +4,3 @@ package-lock.json | |||||||
| .vscode/ | .vscode/ | ||||||
| firebase/ | firebase/ | ||||||
| dist/ | dist/ | ||||||
| public/workbox |  | ||||||
| src/packages/excalidraw/types |  | ||||||
|   | |||||||
| @@ -1,6 +1,40 @@ | |||||||
| { | { | ||||||
|   "extends": ["@excalidraw/eslint-config", "react-app"], |   "extends": ["prettier", "react-app"], | ||||||
|  |   "plugins": ["prettier"], | ||||||
|   "rules": { |   "rules": { | ||||||
|     "import/no-anonymous-default-export": "off" |     "curly": "warn", | ||||||
|  |     "dot-notation": "warn", | ||||||
|  |     "import/no-anonymous-default-export": "off", | ||||||
|  |     "no-console": [ | ||||||
|  |       "warn", | ||||||
|  |       { | ||||||
|  |         "allow": ["warn", "error", "info"] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "no-else-return": "warn", | ||||||
|  |     "no-lonely-if": "warn", | ||||||
|  |     "no-restricted-syntax": [ | ||||||
|  |       "warn", | ||||||
|  |       { | ||||||
|  |         "message": "Use 't(...)' instead of literal text in JSX", | ||||||
|  |         "selector": "JSXText[value=/\\w/]" | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "no-unneeded-ternary": "warn", | ||||||
|  |     "no-unused-expressions": "warn", | ||||||
|  |     "no-unused-vars": "warn", | ||||||
|  |     "no-useless-return": "warn", | ||||||
|  |     "no-var": "warn", | ||||||
|  |     "object-shorthand": "warn", | ||||||
|  |     "one-var": ["warn", "never"], | ||||||
|  |     "prefer-arrow-callback": "warn", | ||||||
|  |     "prefer-const": [ | ||||||
|  |       "warn", | ||||||
|  |       { | ||||||
|  |         "destructuring": "all" | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "prefer-template": "warn", | ||||||
|  |     "prettier/prettier": "warn" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/assets/crowdin.svg
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,6 +0,0 @@ | |||||||
| <svg height="50" viewBox="0 0 257 50" xmlns="http://www.w3.org/2000/svg" fill-rule="evenodd" clip-rule="evenodd" stroke-linejoin="round" stroke-miterlimit="2"> |  | ||||||
| 	<path fill="#fff" d="M-7.977-9.253h288.95v78.13H-7.977z" /> |  | ||||||
| 	<path d="M67.626 32.315c-1.34 0-2.207 0-2.207-1.025 0-.236.079-.551.236-.946l4.02-8.907h12.929c1.34 0 2.128-.08 2.128.946 0 .315-.078.63-.236.946l-.788 1.734h5.439l1.104-2.444c.157-.394.157-.71.157-1.025 0-2.207-2.365-3.31-4.257-3.31H65.655l-5.754 12.691c-.158.394-.158.71-.158 1.025 0 2.365 1.97 3.547 4.73 3.547h20.26l1.26-3.232H67.627zm42.727-14.11H95.059l-6.937 17.342h5.518l5.519-14.032h8.435c1.34 0 2.05-.157 2.05.868 0 .315-.08.63-.237.946l-.789 1.734h5.518l1.104-2.444c.158-.394.158-.71.158-1.025 0-1.025-.552-1.892-1.734-2.522-.946-.473-2.208-.868-3.311-.868zm30.35 0h-21.285l-5.754 12.691c-.158.316-.158.63-.158 1.025 0 1.97 1.419 3.547 3.232 3.547h21.52l5.834-13.007c.158-.394.158-.71.158-1.024 0-2.05-1.734-3.233-3.547-3.233zm-6.701 14.19h-12.85c-1.34 0-1.97-.159-1.97-1.183 0-.316.079-.631.236-.946l4.178-8.908h12.929c1.26 0 1.891-.08 1.891.946 0 .315-.078.63-.236 1.025l-4.178 9.065zm13.953 3.152h28.695l7.41-17.264h-5.676l-6.149 14.032h-9.223l6.149-14.11h-5.676l-6.386 14.031h-6.306c-1.34 0-2.05-.157-2.05-1.182 0-.315.08-.63.237-.946l5.282-11.982h-5.519l-5.518 12.455c-1.103 3.39 2.207 4.966 4.73 4.966zm67.874-23.649l-5.913 1.577-1.97 4.73h-14.584c-3.548 0-6.7 1.576-8.278 4.73l-3.941 9.46c-.788 1.576.63 3.152 3.31 3.152h21.128l10.248-23.649zm-27.591 20.496c-1.183 0-1.735-.788-1.577-1.577l3.469-7.567c.788-1.813 2.68-1.892 4.414-1.892h11.825l-4.73 11.036h-13.401zm26.802 3.153l7.49-17.737-6.307 1.183-7.095 16.554h5.912zm8.435-19.944l1.656-3.705-6.228 1.261-1.577 3.705 6.15-1.26zm22.23 2.601h-20.417l-7.094 17.343h5.518l5.518-14.19h13.48c1.34 0 2.05-.078 2.05 1.026 0 .315-.08.63-.237.946l-5.518 12.297h5.518l5.834-13.007c.157-.315.157-.63.157-1.025 0-1.025-.552-1.892-1.734-2.522-.867-.473-1.892-.868-3.074-.868zm-192.82.868c-8.672-1.025-16.476.71-17.58 6.148 0 .237-.157 1.262-.157 1.42l1.419.157v2.207l-1.34-.157c.551 5.597 3.626 7.252 6.858 7.331h.236c1.42.079 2.917-.237 4.178-.788.08 0 .08-.08.08-.08v-.157c0-.079-.08-.079-.08-.157-.078 0-.078-.08-.157-.08-2.996.395-5.755-2.049-5.755-7.015 0-6.228 4.888-8.514 12.298-8.514.236.158.315-.237 0-.315zM36.803 30.344c.788 0 1.498.158 2.207.237.237 1.655 1.025 3.232 2.208 4.336-1.183-.158-2.208-.71-3.075-1.498a6.051 6.051 0 01-1.34-3.075zm2.68-5.439c0 .237-.157.552-.236.946h-1.025c-.552 0-1.025-.079-1.576-.158v-.157c.63-3.39 4.02-4.73 7.252-5.36a7.997 7.997 0 00-2.76 1.812c-.787.868-1.34 1.813-1.655 2.917z" fill="#2e3340" fill-rule="nonzero" /> |  | ||||||
| 	<path d="M56.274 14.105c-6.543-1.813-34.055-4.02-34.055 11.273.946.158 1.577.315 2.05.394-.079 1.183 0 2.444 0 3.626l-2.444-.394c0 8.83 6.464 11.667 11.588 11.667.868 0 1.656-.078 2.523-.157 2.128-.237 4.178-.867 5.991-1.892.079 0 .079-.08.079-.08v-.157c0-.079-.079-.079-.079-.157-.079 0-.079-.08-.157-.08-4.336.868-10.17-.315-10.17-10.563 0-13.637 19.156-12.77 24.753-13.007.08 0 .08-.079.08-.079v-.157c0-.08 0-.08-.08-.158 0-.079 0-.079-.079-.079zM33.414 39.41a9.362 9.362 0 01-6.78-2.286c-1.892-1.656-3.074-3.942-3.31-6.385 1.655.236 3.704.394 5.438.473a9.43 9.43 0 001.577 4.808c.946 1.42 2.207 2.602 3.705 3.39h-.63zM28.92 24.984l-2.601-.237-2.602-.315c0-7.962 12.77-11.036 18.683-10.484-5.912 1.34-13.086 4.099-13.48 11.036z" fill="#2e3340" fill-rule="nonzero" /> |  | ||||||
| 	<path d="M59.664 9.533c-7.962-2.68-17.027-4.02-25.462-3.941-12.22 0-27.67 3.626-28.064 16.081l3.31.788c-.393 1.577-.393 4.81-.393 4.81s-1.892-.553-2.917-.79c0 14.821 8.671 18.526 17.027 18.526 3.39 0 6.701-.552 9.854-1.734.08 0 .08-.08.08-.08v-.157c0-.079-.08-.079-.08-.157h-.157c-2.602 0-4.651.867-8.75-2.05-7.963-5.597-7.017-20.102 2.128-26.408 9.46-6.701 29.798-4.573 33.267-4.415h.157s.079 0 .079-.079v-.236l-.079-.158zm-36.42 34.292c-9.932 0-14.978-5.36-15.45-15.609 2.68.71 5.202 1.34 7.961 1.734-.157 4.02 1.262 7.962 4.02 11.037a12.488 12.488 0 005.046 2.916l-1.577-.078zM45.632 7.956c-12.06 0-26.014 1.42-28.773 14.584 0 0-7.41-1.182-9.066-1.576C9.843 4.409 38.38 5.67 49.89 7.956h-4.257z" fill="#2e3340" fill-rule="nonzero" /> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 4.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								.github/assets/logo.png
									
									
									
									
										vendored
									
									
								
							
							
						
						| Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 39 KiB | 
							
								
								
									
										9
									
								
								.github/assets/sentry.svg
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,9 +0,0 @@ | |||||||
| <svg class="__sntry__ css-15xgryy e10nushx5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 222 66" height="50" style="background-color: rgb(255, 255, 255);"> |  | ||||||
| 	<defs> |  | ||||||
| 		<style type="text/css"> |  | ||||||
| 			@media (prefers-color-scheme: dark) {svg.__sntry__ { background-color: #584674 !important; }path.__sntry__ { fill: #ffffff !important; }} |  | ||||||
| 		</style> |  | ||||||
| 	</defs> |  | ||||||
| 	<path d="M29,2.26a4.67,4.67,0,0,0-8,0L14.42,13.53A32.21,32.21,0,0,1,32.17,40.19H27.55A27.68,27.68,0,0,0,12.09,17.47L6,28a15.92,15.92,0,0,1,9.23,12.17H4.62A.76.76,0,0,1,4,39.06l2.94-5a10.74,10.74,0,0,0-3.36-1.9l-2.91,5a4.54,4.54,0,0,0,1.69,6.24A4.66,4.66,0,0,0,4.62,44H19.15a19.4,19.4,0,0,0-8-17.31l2.31-4A23.87,23.87,0,0,1,23.76,44H36.07a35.88,35.88,0,0,0-16.41-31.8l4.67-8a.77.77,0,0,1,1.05-.27c.53.29,20.29,34.77,20.66,35.17a.76.76,0,0,1-.68,1.13H40.6q.09,1.91,0,3.81h4.78A4.59,4.59,0,0,0,50,39.43a4.49,4.49,0,0,0-.62-2.28Z M124.32,28.28,109.56,9.22h-3.68V34.77h3.73V15.19l15.18,19.58h3.26V9.22h-3.73ZM87.15,23.54h13.23V20.22H87.14V12.53h14.93V9.21H83.34V34.77h18.92V31.45H87.14ZM71.59,20.3h0C66.44,19.06,65,18.08,65,15.7c0-2.14,1.89-3.59,4.71-3.59a12.06,12.06,0,0,1,7.07,2.55l2-2.83a14.1,14.1,0,0,0-9-3c-5.06,0-8.59,3-8.59,7.27,0,4.6,3,6.19,8.46,7.52C74.51,24.74,76,25.78,76,28.11s-2,3.77-5.09,3.77a12.34,12.34,0,0,1-8.3-3.26l-2.25,2.69a15.94,15.94,0,0,0,10.42,3.85c5.48,0,9-2.95,9-7.51C79.75,23.79,77.47,21.72,71.59,20.3ZM195.7,9.22l-7.69,12-7.64-12h-4.46L186,24.67V34.78h3.84V24.55L200,9.22Zm-64.63,3.46h8.37v22.1h3.84V12.68h8.37V9.22H131.08ZM169.41,24.8c3.86-1.07,6-3.77,6-7.63,0-4.91-3.59-8-9.38-8H154.67V34.76h3.8V25.58h6.45l6.48,9.2h4.44l-7-9.82Zm-10.95-2.5V12.6h7.17c3.74,0,5.88,1.77,5.88,4.84s-2.29,4.86-5.84,4.86Z" transform="translate(11, 11)" fill="#362d59" class="__sntry__"> |  | ||||||
| 	</path> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										3
									
								
								.github/assets/vercel.svg
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +0,0 @@ | |||||||
| <svg height="50" viewBox="0 0 164 50" xmlns="http://www.w3.org/2000/svg" style="background:#fff" fill-rule="evenodd" clip-rule="evenodd" stroke-linejoin="round" stroke-miterlimit="2"> |  | ||||||
| 	<path d="M78.21 15.587c-5.672 0-9.762 3.864-9.762 9.661s4.604 9.66 10.276 9.66c3.427 0 6.448-1.416 8.319-3.805l-3.931-2.372c-1.038 1.186-2.615 1.879-4.388 1.879-2.461 0-4.552-1.342-5.328-3.489h14.397c.113-.601.18-1.223.18-1.879 0-5.79-4.09-9.655-9.763-9.655zm-4.86 7.783c.642-2.142 2.399-3.489 4.855-3.489 2.461 0 4.219 1.347 4.855 3.489h-9.71zm60.187-7.783c-5.673 0-9.763 3.864-9.763 9.661s4.604 9.66 10.276 9.66c3.427 0 6.449-1.416 8.319-3.805l-3.931-2.372c-1.038 1.186-2.615 1.879-4.388 1.879-2.461 0-4.552-1.342-5.328-3.489h14.397c.113-.601.18-1.223.18-1.879 0-5.79-4.09-9.655-9.762-9.655zm-4.856 7.783c.642-2.142 2.4-3.489 4.856-3.489 2.46 0 4.218 1.347 4.855 3.489h-9.711zm-20.054 1.878c0 3.22 2.015 5.367 5.139 5.367 2.116 0 3.704-1.003 4.52-2.64l3.947 2.378c-1.634 2.843-4.696 4.556-8.467 4.556-5.678 0-9.763-3.864-9.763-9.661s4.09-9.66 9.763-9.66c3.77 0 6.828 1.712 8.467 4.556l-3.946 2.377c-.817-1.637-2.405-2.64-4.521-2.64-3.12 0-5.139 2.147-5.139 5.367zm42.378-15.565v24.69h-4.624V9.682h4.624zM24.73 7l18.985 34.35H5.744L24.73 7zm47.465 2.683L57.956 35.446 43.72 9.683h5.338l8.9 16.102 8.898-16.102h5.339zm30.268 6.44v5.202a5.634 5.634 0 00-1.644-.263c-2.985 0-5.138 2.147-5.138 5.367v7.943h-4.624V16.124h4.624v4.938c0-2.727 3.036-4.938 6.782-4.938z" fill-rule="nonzero" /> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										34
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,34 +0,0 @@ | |||||||
| version: 2 |  | ||||||
| updates: |  | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: / |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       day: sunday |  | ||||||
|       time: "01:00" |  | ||||||
|     reviewers: |  | ||||||
|       - lipis |  | ||||||
|     assignees: |  | ||||||
|       - lipis |  | ||||||
|  |  | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: /src/packages/excalidraw/ |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       day: sunday |  | ||||||
|       time: "01:00" |  | ||||||
|     reviewers: |  | ||||||
|       - ad1992 |  | ||||||
|     assignees: |  | ||||||
|       - ad1992 |  | ||||||
|  |  | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: /src/packages/utils/ |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       day: sunday |  | ||||||
|       time: "01:00" |  | ||||||
|     reviewers: |  | ||||||
|       - ad1992 |  | ||||||
|     assignees: |  | ||||||
|       - ad1992 |  | ||||||
							
								
								
									
										26
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,26 +0,0 @@ | |||||||
| name: Auto release @excalidraw/excalidraw-next |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - master |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   Auto-release-excalidraw-next: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|  |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v2 |  | ||||||
|         with: |  | ||||||
|           fetch-depth: 2 |  | ||||||
|       - name: Setup Node.js 14.x |  | ||||||
|         uses: actions/setup-node@v2 |  | ||||||
|         with: |  | ||||||
|           node-version: 14.x |  | ||||||
|       - name: Set up publish access |  | ||||||
|         run: | |  | ||||||
|           npm config set //registry.npmjs.org/:_authToken ${NPM_TOKEN} |  | ||||||
|         env: |  | ||||||
|           NPM_TOKEN: ${{ secrets.NPM_TOKEN }} |  | ||||||
|       - name: Auto release |  | ||||||
|         run: | |  | ||||||
|           yarn autorelease |  | ||||||
							
								
								
									
										4
									
								
								.github/workflows/build-docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -4,10 +4,12 @@ on: | |||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - master |       - master | ||||||
|  |   pull_request: | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build-docker: |   build-docker: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v1 | ||||||
|       - run: docker build -t excalidraw . |       - run: docker build -t excalidraw . | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -7,23 +7,27 @@ on: | |||||||
|   pull_request: |   pull_request: | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   packages: |   build: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v1 | ||||||
|       - name: Setup Node.js 14.x |  | ||||||
|         uses: actions/setup-node@v2 |       - name: Setup Node.js 12.x | ||||||
|  |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version: 14.x |           node-version: 12.x | ||||||
|  |  | ||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|         run: | |         run: | | ||||||
|           yarn --frozen-lockfile |           npm ci | ||||||
|           yarn --cwd src/packages/excalidraw |           npm ci --prefix src/packages/excalidraw | ||||||
|           yarn --cwd src/packages/utils |           npm ci --prefix src/packages/utils | ||||||
|  |  | ||||||
|       - name: Build @excalidraw/excalidraw |       - name: Build @excalidraw/excalidraw | ||||||
|         run: | |         run: | | ||||||
|           yarn --cwd src/packages/excalidraw run pack |           npm run pack --prefix src/packages/excalidraw | ||||||
|  |  | ||||||
|       - name: Build @excalidraw/utils |       - name: Build @excalidraw/utils | ||||||
|         run: | |         run: | | ||||||
|           yarn --cwd src/packages/utils run pack |           npm run pack --prefix src/packages/utils | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								.github/workflows/cancel.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,17 +0,0 @@ | |||||||
| name: Cancel previous runs |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - master |  | ||||||
|   pull_request: |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   cancel: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     timeout-minutes: 3 |  | ||||||
|     steps: |  | ||||||
|       - uses: styfle/cancel-workflow-action@0.6.0 |  | ||||||
|         with: |  | ||||||
|           workflow_id: 400555, 400556, 905313, 1451724, 1710116, 3185001, 3438604 |  | ||||||
|           access_token: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
							
								
								
									
										26
									
								
								.github/workflows/changelog-check.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | |||||||
|  | name: Changelog in sync for packages | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |   pull_request: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   test: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v1 | ||||||
|  |  | ||||||
|  |       - name: Setup Node.js 12.x | ||||||
|  |         uses: actions/setup-node@v1 | ||||||
|  |         with: | ||||||
|  |           node-version: 12.x | ||||||
|  |  | ||||||
|  |       - name: Install and run changelog check | ||||||
|  |         run: | | ||||||
|  |           npm ci | ||||||
|  |           npm run changelog:check | ||||||
|  |         env: | ||||||
|  |           CI: true | ||||||
							
								
								
									
										33
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,33 @@ | |||||||
|  | name: "CodeQL" | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [master] | ||||||
|  |   pull_request: | ||||||
|  |     branches: [master] | ||||||
|  |   schedule: | ||||||
|  |     - cron: "18 7 * * 0" | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   analyze: | ||||||
|  |     name: Analyze | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|  |       matrix: | ||||||
|  |         language: ["typescript"] | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout repository | ||||||
|  |         uses: actions/checkout@v2 | ||||||
|  |  | ||||||
|  |       - name: Initialize CodeQL | ||||||
|  |         uses: github/codeql-action/init@v1 | ||||||
|  |         with: | ||||||
|  |           languages: ${{ matrix.language }} | ||||||
|  |       - name: Autobuild | ||||||
|  |         uses: github/codeql-action/autobuild@v1 | ||||||
|  |  | ||||||
|  |       - name: Perform CodeQL Analysis | ||||||
|  |         uses: github/codeql-action/analyze@v1 | ||||||
							
								
								
									
										24
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,22 +1,28 @@ | |||||||
| name: Lint | name: Lint | ||||||
|  |  | ||||||
| on: pull_request | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |   pull_request: | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   lint: |   lint: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v1 | ||||||
|  |  | ||||||
|       - name: Setup Node.js 14.x |       - name: Setup Node.js 12.x | ||||||
|         uses: actions/setup-node@v2 |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version: 14.x |           node-version: 12.x | ||||||
|  |  | ||||||
|       - name: Install and lint |       - name: Install and lint | ||||||
|         run: | |         run: | | ||||||
|           yarn --frozen-lockfile |           npm ci | ||||||
|           yarn test:other |           npm run test:other | ||||||
|           yarn test:code |           npm run test:code | ||||||
|           yarn test:typecheck |           npm run test:typecheck | ||||||
|  |         env: | ||||||
|  |           CI: true | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								.github/workflows/locales-coverage.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -3,7 +3,7 @@ name: Build locales coverage | |||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - l10n_master |       - "l10n_master" | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   locales: |   locales: | ||||||
| @@ -14,34 +14,19 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} |           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} | ||||||
|  |  | ||||||
|       - name: Setup Node.js 14.x |       - name: Setup Node.js 12.x | ||||||
|         uses: actions/setup-node@v2 |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version: 14.x |           node-version: 12.x | ||||||
|  |  | ||||||
|       - name: Create report file |       - name: Create report file | ||||||
|         run: | |         run: | | ||||||
|           yarn locales-coverage |           npm run locales-coverage | ||||||
|           FILE_CHANGED=$(git diff src/locales/percentages.json) |           FILE_CHANGED=$(git diff src/locales/percentages.json) | ||||||
|           if [ ! -z "${FILE_CHANGED}" ]; then |           if [ ! -z "${FILE_CHANGED}" ]; then | ||||||
|             git config --global user.name 'Excalidraw Bot' |             git config --global user.name 'Kostas Bariotis' | ||||||
|             git config --global user.email 'bot@excalidraw.com' |             git config --global user.email 'konmpar@gmail.com' | ||||||
|             git add src/locales/percentages.json |             git add src/locales/percentages.json | ||||||
|             git commit -am "Auto commit: Calculate translation coverage" |             git commit -am "Auto commit: Calculate translation coverage" | ||||||
|             git push |             git push | ||||||
|           fi |           fi | ||||||
|       - name: Construct comment body |  | ||||||
|         id: getCommentBody |  | ||||||
|         run: | |  | ||||||
|           body=$(npm run locales-coverage:description | grep '^[^>]') |  | ||||||
|           body="${body//'%'/'%25'}" |  | ||||||
|           body="${body//$'\n'/'%0A'}" |  | ||||||
|           body="${body//$'\r'/'%0D'}" |  | ||||||
|           echo ::set-output name=body::$body |  | ||||||
|  |  | ||||||
|       - name: Update description with coverage |  | ||||||
|         uses: kt3k/update-pr-description@v1.0.1 |  | ||||||
|         with: |  | ||||||
|           pr_body: ${{ steps.getCommentBody.outputs.body }} |  | ||||||
|           pr_title: "chore: Update translations from Crowdin" |  | ||||||
|           github_token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} |  | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								.github/workflows/semantic-pr-title.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,16 +0,0 @@ | |||||||
| name: Semantic PR title |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   pull_request_target: |  | ||||||
|     types: |  | ||||||
|       - opened |  | ||||||
|       - edited |  | ||||||
|       - synchronize |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   semantic: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - uses: amannn/action-semantic-pull-request@v3.0.0 |  | ||||||
|         env: |  | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
							
								
								
									
										20
									
								
								.github/workflows/sentry-production.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,4 +1,4 @@ | |||||||
| name: New Sentry production release | name: New Sentry Production Release | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
| @@ -6,23 +6,27 @@ on: | |||||||
|       - master |       - master | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   sentry: |   release: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v1.0.0 | ||||||
|       - name: Setup Node.js 14.x |  | ||||||
|         uses: actions/setup-node@v2 |       - name: Setup Node.js 12.x | ||||||
|  |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version: 14.x |           node-version: 12.x | ||||||
|  |  | ||||||
|       - name: Install and build |       - name: Install and build | ||||||
|         run: | |         run: | | ||||||
|           yarn --frozen-lockfile |           npm ci | ||||||
|           yarn build:app |           npm run build:app | ||||||
|         env: |         env: | ||||||
|           CI: true |           CI: true | ||||||
|  |  | ||||||
|       - name: Install Sentry |       - name: Install Sentry | ||||||
|         run: | |         run: | | ||||||
|           curl -sL https://sentry.io/get-cli/ | bash |           curl -sL https://sentry.io/get-cli/ | bash | ||||||
|  |  | ||||||
|       - name: Create new Sentry release |       - name: Create new Sentry release | ||||||
|         run: | |         run: | | ||||||
|           export SENTRY_RELEASE=$(sentry-cli releases propose-version) |           export SENTRY_RELEASE=$(sentry-cli releases propose-version) | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,17 +1,26 @@ | |||||||
| name: Tests | name: Tests | ||||||
|  |  | ||||||
| on: pull_request | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |   pull_request: | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   test: |   test: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v1 | ||||||
|       - name: Setup Node.js 14.x |  | ||||||
|         uses: actions/setup-node@v2 |       - name: Setup Node.js 12.x | ||||||
|  |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version: 14.x |           node-version: 12.x | ||||||
|  |  | ||||||
|       - name: Install and test |       - name: Install and test | ||||||
|         run: | |         run: | | ||||||
|           yarn --frozen-lockfile |           npm ci | ||||||
|           yarn test:app |           npm run test:app | ||||||
|  |         env: | ||||||
|  |           CI: true | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -5,11 +5,9 @@ | |||||||
| .env.test.local | .env.test.local | ||||||
| .envrc | .envrc | ||||||
| .eslintcache | .eslintcache | ||||||
| .history |  | ||||||
| .idea | .idea | ||||||
| .vercel | .vercel | ||||||
| .vscode | .vscode | ||||||
| .yarn |  | ||||||
| *.log | *.log | ||||||
| *.tgz | *.tgz | ||||||
| build | build | ||||||
| @@ -18,8 +16,7 @@ firebase | |||||||
| logs | logs | ||||||
| node_modules | node_modules | ||||||
| npm-debug.log* | npm-debug.log* | ||||||
| package-lock.json |  | ||||||
| static | static | ||||||
| yarn-debug.log* | yarn-debug.log* | ||||||
| yarn-error.log* | yarn-error.log* | ||||||
| src/packages/excalidraw/types | yarn.lock | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.prettierrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "trailingComma": "all" | ||||||
|  | } | ||||||
| @@ -5,10 +5,11 @@ | |||||||
| ### Option 1 - Manual | ### Option 1 - Manual | ||||||
|  |  | ||||||
| 1. Fork and clone the repo | 1. Fork and clone the repo | ||||||
| 1. Run `yarn` to install dependencies | 1. Run `npm install` to install dependencies | ||||||
| 1. Create a branch for your PR with `git checkout -b your-branch-name` | 1. Create a branch for your PR with `git checkout -b your-branch-name` | ||||||
|  |  | ||||||
| > To keep `master` branch pointing to remote repository and make pull requests from branches on your fork. To do this, run: | > To keep `master` branch pointing to remote repository and make | ||||||
|  | > pull requests from branches on your fork. To do this, run: | ||||||
| > | > | ||||||
| > ```sh | > ```sh | ||||||
| > git remote add upstream https://github.com/excalidraw/excalidraw.git | > git remote add upstream https://github.com/excalidraw/excalidraw.git | ||||||
| @@ -19,45 +20,8 @@ | |||||||
| ### Option 2 - CodeSandbox | ### Option 2 - CodeSandbox | ||||||
|  |  | ||||||
| 1. Go to https://codesandbox.io/s/github/excalidraw/excalidraw | 1. Go to https://codesandbox.io/s/github/excalidraw/excalidraw | ||||||
| 1. Connect your GitHub account | 1. Connect your Github account | ||||||
| 1. Go to Git tab on left side | 1. Go to Git tab on left side | ||||||
| 1. Tap on `Fork Sandbox` | 1. Tap on `Fork Sandbox` | ||||||
| 1. Write your code | 1. Write your code | ||||||
| 1. Commit and PR automatically | 1. Commit and PR automatically | ||||||
|  |  | ||||||
| ## Pull Request Guidelines |  | ||||||
|  |  | ||||||
| Don't worry if you get any of the below wrong, or if you don't know how. We'll gladly help out. |  | ||||||
|  |  | ||||||
| ### Title |  | ||||||
|  |  | ||||||
| Make sure the title starts with a semantic prefix: |  | ||||||
|  |  | ||||||
| - **feat**: A new feature |  | ||||||
| - **fix**: A bug fix |  | ||||||
| - **docs**: Documentation only changes |  | ||||||
| - **style**: Changes that do not affect the meaning of the code (white-space, formatting, missing semi-colons, etc) |  | ||||||
| - **refactor**: A code change that neither fixes a bug nor adds a feature |  | ||||||
| - **perf**: A code change that improves performance |  | ||||||
| - **test**: Adding missing tests or correcting existing tests |  | ||||||
| - **build**: Changes that affect the build system or external dependencies (example scopes: gulp, broccoli, npm) |  | ||||||
| - **ci**: Changes to our CI configuration files and scripts (example scopes: Travis, Circle, BrowserStack, SauceLabs) |  | ||||||
| - **chore**: Other changes that don't modify src or test files |  | ||||||
| - **revert**: Reverts a previous commit |  | ||||||
|  |  | ||||||
| ### Changelog |  | ||||||
|  |  | ||||||
| Add a brief description of your pull request to the changelog located here: [`src/packages/excalidraw/CHANGELOG.md`](src/packages/excalidraw/CHANGELOG.md) |  | ||||||
|  |  | ||||||
| Notes: |  | ||||||
|  |  | ||||||
| - Make sure to prepend to the section corresponding with the semantic prefix you selected in the title |  | ||||||
| - Link to your pull request - this will require updating the CHANGELOG _after_ creating the pull request |  | ||||||
|  |  | ||||||
| ### Testing |  | ||||||
|  |  | ||||||
| Once you submit your pull request it will automatically be tested. Be sure to check the results of the test and fix any issues that arise. |  | ||||||
|  |  | ||||||
| It's also a good idea to consider if your change should include additional tests. This is highly recommended for new features or bug-fixes. For example, it's good practice to create a test for each bug you fix which ensures that we don't regress the code in the future. |  | ||||||
|  |  | ||||||
| Finally - always manually test your changes using the convenient staging environment deployed for each pull request. As much as local development attempts to replicate production, there can still be subtle differences in behavior. For larger features consider testing your change in multiple browsers as well. |  | ||||||
|   | |||||||
| @@ -2,15 +2,16 @@ FROM node:14-alpine AS build | |||||||
|  |  | ||||||
| WORKDIR /opt/node_app | WORKDIR /opt/node_app | ||||||
|  |  | ||||||
| COPY package.json yarn.lock ./ | COPY package.json package-lock.json ./ | ||||||
| RUN yarn --ignore-optional | RUN npm i --no-optional | ||||||
|  |  | ||||||
|  | ARG REACT_APP_INCLUDE_GTAG=false | ||||||
| ARG NODE_ENV=production | ARG NODE_ENV=production | ||||||
|  |  | ||||||
| COPY . . | COPY . . | ||||||
| RUN yarn build:app:docker | RUN npm run build:app:docker | ||||||
|  |  | ||||||
| FROM nginx:1.21-alpine | FROM nginx:1.17-alpine | ||||||
|  |  | ||||||
| COPY --from=build /opt/node_app/build /usr/share/nginx/html | COPY --from=build /opt/node_app/build /usr/share/nginx/html | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										164
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,8 +1,8 @@ | |||||||
| <div align="center" style="display:flex;flex-direction:column;"> | <div align="center" style="display:flex;flex-direction:column;"> | ||||||
|   <a href="https://excalidraw.com"> |   <a href="https://excalidraw.com"> | ||||||
|     <img width="540" src="./public/og-image-sm.png" alt="Excalidraw logo: Sketch handrawn like diagrams." /> |     <img src="./public/og-image.png" alt="Excalidraw logo: Sketch handrawn like diagrams." /> | ||||||
|   </a> |   </a> | ||||||
|   <h3>Virtual whiteboard for sketching hand-drawn like diagrams.<br>Collaborative and end-to-end encrypted.</h3> |   <h3>Virtual whiteboard for sketching hand-drawn like diagrams.</h3> | ||||||
|   <p> |   <p> | ||||||
|     <a href="https://twitter.com/Excalidraw"> |     <a href="https://twitter.com/Excalidraw"> | ||||||
|       <img alt="Follow Excalidraw on Twitter" src="https://img.shields.io/twitter/follow/excalidraw.svg?label=follow+excalidraw&style=social&logo=twitter"> |       <img alt="Follow Excalidraw on Twitter" src="https://img.shields.io/twitter/follow/excalidraw.svg?label=follow+excalidraw&style=social&logo=twitter"> | ||||||
| @@ -10,136 +10,62 @@ | |||||||
|     <a target="_blank" href="https://crowdin.com/project/excalidraw"> |     <a target="_blank" href="https://crowdin.com/project/excalidraw"> | ||||||
|       <img src="https://badges.crowdin.net/excalidraw/localized.svg"> |       <img src="https://badges.crowdin.net/excalidraw/localized.svg"> | ||||||
|     </a> |     </a> | ||||||
|  |     <a target="_blank" href="https://hub.docker.com/r/excalidraw/excalidraw"> | ||||||
|  |       <img src="https://img.shields.io/docker/pulls/excalidraw/excalidraw"> | ||||||
|  |     </a> | ||||||
|   </p> |   </p> | ||||||
|   <p>Ask questions or hang out on our <a target="_blank" href="https://discord.gg/UexuTaE">discord.gg/UexuTaE</a>.</p> |  | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| ## Try it now | ## Try it now | ||||||
|  |  | ||||||
| Go to [excalidraw.com](https://excalidraw.com) to start sketching. | Go to [excalidraw.com](https://excalidraw.com) to start sketching. | ||||||
|  |  | ||||||
| Read the latest news and updates on our [blog](https://blog.excalidraw.com). A good start is to see all the updates of [One Year of Excalidraw](https://blog.excalidraw.com/one-year-of-excalidraw/). | Read our [blog](https://blog.excalidraw.com) and follow the [guides](https://howto.excalidraw.com) to learn more about Excalidraw and how to use it effectively. | ||||||
|  |  | ||||||
| ## Supporting Excalidraw |  | ||||||
|  |  | ||||||
| If you like the project, you can become a sponsor at [Open Collective](https://opencollective.com/excalidraw). |  | ||||||
|  |  | ||||||
| [<img src="https://opencollective.com/excalidraw/tiers/sponsors/0/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/0/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/1/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/1/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/2/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/2/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/3/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/3/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/4/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/4/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/5/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/5/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/6/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/6/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/7/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/7/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/8/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/8/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/9/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/9/website) [<img src="https://opencollective.com/excalidraw/tiers/sponsors/10/avatar.svg?avatarHeight=120">](https://opencollective.com/excalidraw/tiers/sponsors/10/website) |  | ||||||
|  |  | ||||||
| <a href="https://opencollective.com/excalidraw#category-CONTRIBUTE" target="_blank"><img src="https://opencollective.com/excalidraw/tiers/backers.svg?avatarHeight=32"/></a> |  | ||||||
|  |  | ||||||
| Last but not least, we're thankful to these companies for offering their services for free: |  | ||||||
|  |  | ||||||
| [](https://vercel.com) [](https://sentry.io) [](https://crowdin.com) |  | ||||||
|  |  | ||||||
| ## Documentation |  | ||||||
|  |  | ||||||
| ### Shortcuts |  | ||||||
|  |  | ||||||
| You can almost do anything with shortcuts. Click on the help icon on the bottom right corner to see them all. |  | ||||||
|  |  | ||||||
| ### Curved lines and arrows |  | ||||||
|  |  | ||||||
| Choose line or arrow and click click click instead of drag. |  | ||||||
|  |  | ||||||
| ### Charts |  | ||||||
|  |  | ||||||
| You can easily create charts by copy pasting data from Excel or just plain comma separated text. |  | ||||||
|  |  | ||||||
| ### Translating |  | ||||||
|  |  | ||||||
| To translate Excalidraw into other languages, please visit [our Crowdin page](https://crowdin.com/project/excalidraw). To add a new language, [open an issue](https://github.com/excalidraw/excalidraw/issues/new) so we can get things set up on our end first. |  | ||||||
|  |  | ||||||
| Translations will be available on the app if they exceed a certain threshold of completion (currently 85%). |  | ||||||
|  |  | ||||||
| ### Create a collaboration session manually |  | ||||||
|  |  | ||||||
| In order to create a session manually, you just need to generate a link of this form: |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| https://excalidraw.com/#room=[0-9a-f]{20},[a-zA-Z0-9_-]{22} |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| #### Example |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| https://excalidraw.com/#room=91bd46ae3aa84dff9d20,pfLqgEoY1c2ioq8LmGwsFA |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| The first set of digits is the room. This is visible from the server that’s going to dispatch messages to everyone that knows this number. |  | ||||||
|  |  | ||||||
| The second set of digits is the encryption key. The Excalidraw server doesn’t know about it. This is what all the participants use to encrypt/decrypt the messages. |  | ||||||
|  |  | ||||||
| > Note: Please ensure that the encryption key is 22 characters long. |  | ||||||
|  |  | ||||||
| ## Shape libraries | ## Shape libraries | ||||||
|  |  | ||||||
| Find a growing list of libraries containing assets for your drawings at [libraries.excalidraw.com](https://libraries.excalidraw.com). | Find a growing list of libraries containing assets for your drawings at [libraries.excalidraw.com](https://libraries.excalidraw.com). | ||||||
|  |  | ||||||
| ## Embedding Excalidraw in your App? | ## Run the code | ||||||
|  |  | ||||||
| Try out [`@excalidraw/excalidraw`](https://www.npmjs.com/package/@excalidraw/excalidraw). This package allows you to easily embed Excalidraw as a React component into your apps. |  | ||||||
|  |  | ||||||
| ## Development |  | ||||||
|  |  | ||||||
| ### Code Sandbox | ### Code Sandbox | ||||||
|  |  | ||||||
| - Go to https://codesandbox.io/s/github/excalidraw/excalidraw | - Go to https://codesandbox.io/s/github/excalidraw/excalidraw | ||||||
|   - You may need to sign in with GitHub and reload the page |   - You may need to sign in with Github and reload the page | ||||||
| - You can start coding instantly, and even send PRs from there! | - You can start coding instantly, and even send PRs from there! | ||||||
|  |  | ||||||
| ### Local Installation | ### Local Installation | ||||||
|  |  | ||||||
| These instructions will get you a copy of the project up and running on your local machine for development and testing purposes. | These instructions will get you a copy of the project up and running on your local machine for development and testing purposes. | ||||||
|  |  | ||||||
| #### Requirements |  | ||||||
|  |  | ||||||
| - [Node.js](https://nodejs.org/en/) |  | ||||||
| - [Yarn](https://yarnpkg.com/getting-started/install) (v1 or v2.4.2+) |  | ||||||
| - [Git](https://git-scm.com/downloads) |  | ||||||
|  |  | ||||||
| #### Clone the repo | #### Clone the repo | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| git clone https://github.com/excalidraw/excalidraw.git | git clone https://github.com/excalidraw/excalidraw.git | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| #### Install the dependencies |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| yarn |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| #### Start the server |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| yarn start |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Now you can open [http://localhost:3000](http://localhost:3000) and start coding in your favorite code editor. |  | ||||||
|  |  | ||||||
| #### Commands | #### Commands | ||||||
|  |  | ||||||
| | Command            | Description                       | | | Command               | Description                       | | ||||||
| | ------------------ | --------------------------------- | | | --------------------- | --------------------------------- | | ||||||
| | `yarn`             | Install the dependencies          | | | `npm install`         | Install the dependencies          | | ||||||
| | `yarn start`       | Run the project                   | | | `npm start`           | Run the project                   | | ||||||
| | `yarn fix`         | Reformat all files with Prettier  | | | `npm run fix`         | Reformat all files with Prettier  | | ||||||
| | `yarn test`        | Run tests                         | | | `npm test`            | Run tests                         | | ||||||
| | `yarn test:update` | Update test snapshots             | | | `npm run test:update` | Update test snapshots             | | ||||||
| | `yarn test:code`   | Test for formatting with Prettier | | | `npm run test:code`   | Test for formatting with Prettier | | ||||||
|  |  | ||||||
| #### Docker Compose | #### Docker Compose | ||||||
|  |  | ||||||
| You can use docker-compose to work on Excalidraw locally if you don't want to setup a Node.js env. | You can use docker-compose to work on excalidraw locally if you don't want to setup a Node.js env. | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| docker-compose up --build -d | docker-compose up --build -d | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Self-hosting | ## Self hosting | ||||||
|  |  | ||||||
| We publish a Docker image with the Excalidraw client at [excalidraw/excalidraw](https://hub.docker.com/r/excalidraw/excalidraw). You can use it to self-host your own client under your own domain, on Kubernetes, AWS ECS, etc. | We publish a Docker image with the Excalidraw client at [excalidraw/excalidraw](https://hub.docker.com/r/excalidraw/excalidraw). You can use it to self host your own client under your own domain, on Kubernetes, AWS ECS, etc. | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| docker build -t excalidraw/excalidraw . | docker build -t excalidraw/excalidraw . | ||||||
| @@ -150,17 +76,63 @@ The Docker image is free of analytics and other tracking libraries. | |||||||
|  |  | ||||||
| **At the moment, self-hosting your own instance doesn't support sharing or collaboration features.** | **At the moment, self-hosting your own instance doesn't support sharing or collaboration features.** | ||||||
|  |  | ||||||
| We are working towards providing a full-fledged solution for self-hosting your own Excalidraw. | We are working towards providing a full-fledged solution for self hosting your own Excalidraw. | ||||||
|  |  | ||||||
| ## Contributing | ## Contributing | ||||||
|  |  | ||||||
| Pull requests are welcome. For major changes, please [open an issue](https://github.com/excalidraw/excalidraw/issues/new) first to discuss what you would like to change. | Pull requests are welcome. For major changes, please [open an issue](https://github.com/excalidraw/excalidraw/issues/new) first to discuss what you would like to change. | ||||||
|  |  | ||||||
| ## Notable used tools | ## Translating | ||||||
|  |  | ||||||
| - [Create React App](https://github.com/facebook/create-react-app) | To translate Excalidraw into other languages, please visit [our Crowdin page](https://crowdin.com/project/excalidraw). To add a new language, [open an issue](https://github.com/excalidraw/excalidraw/issues/new) so we can get things set up on our end first. | ||||||
|  |  | ||||||
|  | Translations will be available on the app if they exceed a certain threshold of completion (currently 85%). | ||||||
|  |  | ||||||
|  | ## Excalidraw is built using these awesome tools | ||||||
|  |  | ||||||
|  | - [React](https://reactjs.org) | ||||||
| - [Rough.js](https://roughjs.com) | - [Rough.js](https://roughjs.com) | ||||||
| - [TypeScript](https://www.typescriptlang.org) | - [TypeScript](https://www.typescriptlang.org) | ||||||
| - [Vercel](https://vercel.com) | - [Vercel](https://vercel.com) | ||||||
|  |  | ||||||
| And the main source of inspiration for starting the project is the awesome [Zwibbler](https://zwibbler.com/demo/) app. | And the main source of inspiration for starting the project is the awesome [Zwibbler](https://zwibbler.com/demo/) app. | ||||||
|  |  | ||||||
|  | ## Testimonials | ||||||
|  |  | ||||||
|  | <a href="https://twitter.com/Lissy_Sykes/status/1213813117177729026"><img width="398" src="https://user-images.githubusercontent.com/197597/71783813-dbf8a600-2fa0-11ea-9c0d-bb3cc45969e6.png"></a> | ||||||
|  | <a href="https://twitter.com/dan_abramov/status/1213762494428262400"><img width="398" src="https://user-images.githubusercontent.com/197597/71783990-4d395880-2fa3-11ea-9ad7-186138db5003.png"></a> | ||||||
|  |  | ||||||
|  | <a href="https://twitter.com/kyehohenberger/status/1214288572037025792"><img width="423" src="https://user-images.githubusercontent.com/197597/71851802-34f13880-308c-11ea-9416-191099e6349c.png"></a> | ||||||
|  | <a href="https://twitter.com/lucasazzola/status/1215126440330416128"><img width="429" src="https://user-images.githubusercontent.com/197597/72039003-48e99580-3258-11ea-8daa-85dd055f2a82.png"> | ||||||
|  |  | ||||||
|  | <a href="https://twitter.com/jordwalke/status/1214858186789806080"><img width="434" src="https://user-images.githubusercontent.com/197597/72036874-07a1b780-3251-11ea-99e8-6bafd93483a0.png"></a> | ||||||
|  |  | ||||||
|  | ## Contributors | ||||||
|  |  | ||||||
|  | ### Code Contributors | ||||||
|  |  | ||||||
|  | This project exists thanks to all the people who contribute. [[Contribute](CONTRIBUTING.md)]. | ||||||
|  | <a href="https://github.com/excalidraw/excalidraw/graphs/contributors"><img src="https://opencollective.com/excalidraw/contributors.svg?width=890&button=false" /></a> | ||||||
|  |  | ||||||
|  | ### Financial Contributors | ||||||
|  |  | ||||||
|  | Become a financial contributor and help us sustain our community. [[Contribute](https://opencollective.com/excalidraw/contribute)] | ||||||
|  |  | ||||||
|  | #### Individuals | ||||||
|  |  | ||||||
|  | <a href="https://opencollective.com/excalidraw"><img src="https://opencollective.com/excalidraw/individuals.svg?width=890"></a> | ||||||
|  |  | ||||||
|  | #### Organizations | ||||||
|  |  | ||||||
|  | Support this project with your organization. Your logo will show up here with a link to your website. [[Contribute](https://opencollective.com/excalidraw/contribute)] | ||||||
|  |  | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/0/website"><img src="https://opencollective.com/excalidraw/organization/0/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/1/website"><img src="https://opencollective.com/excalidraw/organization/1/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/2/website"><img src="https://opencollective.com/excalidraw/organization/2/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/3/website"><img src="https://opencollective.com/excalidraw/organization/3/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/4/website"><img src="https://opencollective.com/excalidraw/organization/4/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/5/website"><img src="https://opencollective.com/excalidraw/organization/5/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/6/website"><img src="https://opencollective.com/excalidraw/organization/6/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/7/website"><img src="https://opencollective.com/excalidraw/organization/7/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/8/website"><img src="https://opencollective.com/excalidraw/organization/8/avatar.svg"></a> | ||||||
|  | <a href="https://opencollective.com/excalidraw/organization/9/website"><img src="https://opencollective.com/excalidraw/organization/9/avatar.svg"></a> | ||||||
|   | |||||||
							
								
								
									
										64
									
								
								analytics.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,64 @@ | |||||||
|  | | Excalidraw              | Category | Name                               | Label                           | Value     | | ||||||
|  | | ----------------------- | -------- | ---------------------------------- | ------------------------------- | --------- | | ||||||
|  | | Shape / Selection       | shape    | selection, rectangle, diamond, etc | `toolbar` or `shortcut`         | | ||||||
|  | | Text on double click    | shape    | text                               | `double-click`                  | | ||||||
|  | | Lock selection          | shape    | lock                               | `on` or `off`                   | | ||||||
|  | | Clear canvas            | action   | clear canvas                       | | ||||||
|  | | Zoom in                 | action   | zoom                               | in                              | `zoom`    | | ||||||
|  | | Zoom out                | action   | zoom                               | out                             | `zoom`    | | ||||||
|  | | Zoom fit                | action   | zoom                               | fit                             | `zoom`    | | ||||||
|  | | Zoom reset              | action   | zoom                               | reset                           | `zoom`    | | ||||||
|  | | Scroll back to content  | action   | scroll to content                  | | ||||||
|  | | Load file               | io       | load                               | `MIME type`                     | | ||||||
|  | | Import from URL         | io       | import                             | | ||||||
|  | | Save                    | io       | save                               | | ||||||
|  | | Save as                 | io       | save as                            | | ||||||
|  | | Export to backend       | io       | export                             | backend                         | | ||||||
|  | | Export as SVG           | io       | export                             | `svg` or `clipboard-svg`        | | ||||||
|  | | Export to PNG           | io       | export                             | `png` or `clipboard-png`        | | ||||||
|  | | Canvas color            | change   | canvas color                       | `color`                         | | ||||||
|  | | Background color        | change   | background color                   | `color`                         | | ||||||
|  | | Stroke color            | change   | stroke color                       | `color`                         | | ||||||
|  | | Stroke width            | change   | stroke                             | width                           | `width`   | | ||||||
|  | | Stroke style            | change   | style                              | `solid` or `dashed` or `dotted` | | ||||||
|  | | Stroke sloppiness       | change   | stroke                             | sloppiness                      | `value`   | | ||||||
|  | | Fill                    | change   | fill                               | `value`                         | | ||||||
|  | | Edge                    | change   | edge                               | `value`                         | | ||||||
|  | | Opacity                 | change   | opacity                            | value                           | `opacity` | | ||||||
|  | | Project name            | change   | title                              | | ||||||
|  | | Theme                   | change   | theme                              | `light` or `dark`               | | ||||||
|  | | Change language         | change   | language                           | `language`                      | | ||||||
|  | | Send to back            | layer    | move                               | `back`                          | | ||||||
|  | | Send backward           | layer    | move                               | `down`                          | | ||||||
|  | | Bring to front          | layer    | move                               | `front`                         | | ||||||
|  | | Bring forward           | layer    | move                               | `up`                            | | ||||||
|  | | Align left              | align    | align                              | `left`                          | | ||||||
|  | | Align right             | align    | align                              | `right`                         | | ||||||
|  | | Align top               | align    | align                              | `top`                           | | ||||||
|  | | Align bottom            | align    | align                              | `bottom`                        | | ||||||
|  | | Center horizontally     | align    | horizontally                       | `center`                        | | ||||||
|  | | Center vertically       | align    | vertically                         | `center`                        | | ||||||
|  | | Distribute horizontally | align    | distribute                         | `horizontally`                  | | ||||||
|  | | Distribute vertically   | align    | distribute                         | `vertically`                    | | ||||||
|  | | Start session           | share    | session start                      | | ||||||
|  | | Join session            | share    | session join                       | | ||||||
|  | | Start end               | share    | session end                        | | ||||||
|  | | Copy room link          | share    | copy link                          | | ||||||
|  | | Go to collaborator      | share    | go to collaborator                 | | ||||||
|  | | Change name             | share    | name                               | | ||||||
|  | | Add to library          | library  | add                                | | ||||||
|  | | Remove from library     | library  | remove                             | | ||||||
|  | | Load library            | library  | load                               | | ||||||
|  | | Save library            | library  | save                               | | ||||||
|  | | Import library          | library  | import                             | | ||||||
|  | | Shortcuts dialog        | dialog   | shortcuts                          | | ||||||
|  | | Collaboration dialog    | dialog   | collaboration                      | | ||||||
|  | | Export dialog           | dialog   | export                             | | ||||||
|  | | Library dialog          | dialog   | library                            | | ||||||
|  | | E2EE shield             | exit     | e2ee shield                        | | ||||||
|  | | GitHub corner           | exit     | github                             | | ||||||
|  | | Excalidraw blog         | exit     | blog                               | | ||||||
|  | | Excalidraw guides       | exit     | guides                             | | ||||||
|  | | File issues             | exit     | issues                             | | ||||||
|  | | First load              | load     | first load                         | | ||||||
|  | | Load from stroage       | load     | storage                            | size                            | `bytes`   | | ||||||
| @@ -18,7 +18,7 @@ services: | |||||||
|     volumes: |     volumes: | ||||||
|       - ./:/opt/node_app/app:delegated |       - ./:/opt/node_app/app:delegated | ||||||
|       - ./package.json:/opt/node_app/package.json |       - ./package.json:/opt/node_app/package.json | ||||||
|       - ./yarn.lock:/opt/node_app/yarn.lock |       - ./package-lock.json:/opt/node_app/package-lock.json | ||||||
|       - notused:/opt/node_app/app/node_modules |       - notused:/opt/node_app/app/node_modules | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   | |||||||
| @@ -2,8 +2,5 @@ | |||||||
|   "firestore": { |   "firestore": { | ||||||
|     "rules": "firestore.rules", |     "rules": "firestore.rules", | ||||||
|     "indexes": "firestore.indexes.json" |     "indexes": "firestore.indexes.json" | ||||||
|   }, |  | ||||||
|   "storage": { |  | ||||||
|     "rules": "storage.rules" |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,12 +0,0 @@ | |||||||
| rules_version = '2'; |  | ||||||
| service firebase.storage { |  | ||||||
|   match /b/{bucket}/o { |  | ||||||
|     match /{migrations} { |  | ||||||
|       match /{scenes}/{scene} { |  | ||||||
|       	allow get, write: if true; |  | ||||||
|         // redundant, but let's be explicit' |  | ||||||
|         allow list: if false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| { | { | ||||||
|   "public": true, |  | ||||||
|   "headers": [ |   "headers": [ | ||||||
|     { |     { | ||||||
|       "source": "/(.*)", |       "source": "/(.*)", | ||||||
| @@ -22,5 +21,12 @@ | |||||||
|         } |         } | ||||||
|       ] |       ] | ||||||
|     } |     } | ||||||
|  |   ], | ||||||
|  |   "redirects": [ | ||||||
|  |     { | ||||||
|  |       "source": "/([^.]+)", | ||||||
|  |       "destination": "/", | ||||||
|  |       "statusCode": 301 | ||||||
|  |     } | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
							
								
								
									
										26607
									
								
								package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										95
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @@ -19,59 +19,52 @@ | |||||||
|     ] |     ] | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@dwelle/browser-fs-access": "0.21.3", |     "@sentry/browser": "5.28.0", | ||||||
|     "@sentry/browser": "6.2.5", |     "@sentry/integrations": "5.28.0", | ||||||
|     "@sentry/integrations": "6.2.5", |     "@testing-library/jest-dom": "5.11.6", | ||||||
|     "@testing-library/jest-dom": "5.11.10", |     "@testing-library/react": "11.2.2", | ||||||
|     "@testing-library/react": "11.2.6", |     "@types/jest": "26.0.16", | ||||||
|     "@tldraw/vec": "0.0.106", |     "@types/nanoid": "2.1.0", | ||||||
|     "@types/jest": "26.0.22", |     "@types/react": "17.0.0", | ||||||
|     "@types/pica": "5.1.3", |     "@types/react-dom": "17.0.0", | ||||||
|     "@types/react": "17.0.3", |     "@types/socket.io-client": "1.4.34", | ||||||
|     "@types/react-dom": "17.0.3", |     "browser-nativefs": "0.11.1", | ||||||
|     "@types/socket.io-client": "1.4.36", |  | ||||||
|     "clsx": "1.1.1", |     "clsx": "1.1.1", | ||||||
|     "fake-indexeddb": "3.1.3", |     "firebase": "8.1.2", | ||||||
|     "firebase": "8.3.3", |     "i18next-browser-languagedetector": "6.0.1", | ||||||
|     "i18next-browser-languagedetector": "6.1.0", |  | ||||||
|     "idb-keyval": "5.1.3", |  | ||||||
|     "image-blob-reduce": "3.0.1", |  | ||||||
|     "lodash.throttle": "4.1.1", |     "lodash.throttle": "4.1.1", | ||||||
|     "nanoid": "3.1.22", |     "nanoid": "2.1.11", | ||||||
|     "open-color": "1.8.0", |     "node-sass": "4.14.1", | ||||||
|  |     "open-color": "1.7.0", | ||||||
|     "pako": "1.0.11", |     "pako": "1.0.11", | ||||||
|     "perfect-freehand": "1.0.15", |  | ||||||
|     "png-chunk-text": "1.0.0", |     "png-chunk-text": "1.0.0", | ||||||
|     "png-chunks-encode": "1.0.0", |     "png-chunks-encode": "1.0.0", | ||||||
|     "png-chunks-extract": "1.0.0", |     "png-chunks-extract": "1.0.0", | ||||||
|     "points-on-curve": "0.2.0", |     "points-on-curve": "0.2.0", | ||||||
|     "pwacompat": "2.0.17", |     "pwacompat": "2.0.17", | ||||||
|     "react": "17.0.2", |     "react": "17.0.1", | ||||||
|     "react-dom": "17.0.2", |     "react-dom": "17.0.1", | ||||||
|     "react-scripts": "4.0.3", |     "react-scripts": "4.0.1", | ||||||
|     "roughjs": "4.4.1", |     "roughjs": "4.3.1", | ||||||
|     "sass": "1.32.10", |  | ||||||
|     "socket.io-client": "2.3.1", |     "socket.io-client": "2.3.1", | ||||||
|     "typescript": "4.2.4" |     "typescript": "4.0.5" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@excalidraw/eslint-config": "1.0.0", |  | ||||||
|     "@excalidraw/prettier-config": "1.0.2", |  | ||||||
|     "@types/lodash.throttle": "4.1.6", |     "@types/lodash.throttle": "4.1.6", | ||||||
|     "@types/pako": "1.0.1", |     "@types/pako": "1.0.1", | ||||||
|     "@types/resize-observer-browser": "0.1.5", |     "asar": "3.0.3", | ||||||
|     "eslint-config-prettier": "8.3.0", |     "eslint-config-prettier": "7.0.0", | ||||||
|     "eslint-plugin-prettier": "3.3.1", |     "eslint-plugin-prettier": "3.1.4", | ||||||
|     "firebase-tools": "9.9.0", |     "firebase-tools": "8.17.0", | ||||||
|     "husky": "4.3.8", |     "husky": "4.3.0", | ||||||
|     "jest-canvas-mock": "2.3.1", |     "jest-canvas-mock": "2.3.0", | ||||||
|     "lint-staged": "10.5.4", |     "lint-staged": "10.5.3", | ||||||
|     "pepjs": "0.5.3", |     "pepjs": "0.5.2", | ||||||
|     "prettier": "2.2.1", |     "prettier": "2.2.1", | ||||||
|     "rewire": "5.0.0" |     "rewire": "5.0.0" | ||||||
|   }, |   }, | ||||||
|   "engines": { |   "engines": { | ||||||
|     "node": ">=14.0.0" |     "node": ">=12.0.0" | ||||||
|   }, |   }, | ||||||
|   "homepage": ".", |   "homepage": ".", | ||||||
|   "husky": { |   "husky": { | ||||||
| @@ -81,35 +74,33 @@ | |||||||
|   }, |   }, | ||||||
|   "jest": { |   "jest": { | ||||||
|     "transformIgnorePatterns": [ |     "transformIgnorePatterns": [ | ||||||
|       "node_modules/(?!(roughjs|points-on-curve|path-data-parser|points-on-path|@dwelle/browser-fs-access)/)" |       "node_modules/(?!(roughjs|points-on-curve|path-data-parser|points-on-path|browser-nativefs)/)" | ||||||
|     ], |     ], | ||||||
|     "resetMocks": false |     "resetMocks": false | ||||||
|   }, |   }, | ||||||
|   "name": "excalidraw", |   "name": "excalidraw", | ||||||
|   "prettier": "@excalidraw/prettier-config", |  | ||||||
|   "private": true, |   "private": true, | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "build-node": "node ./scripts/build-node.js", |     "build-node": "node ./scripts/build-node.js", | ||||||
|     "build:app:docker": "REACT_APP_DISABLE_SENTRY=true react-scripts build", |     "build:app:docker": "REACT_APP_INCLUDE_GTAG=false 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_INCLUDE_GTAG=true REACT_APP_GIT_SHA=$NOW_GITHUB_COMMIT_SHA react-scripts build", | ||||||
|     "build:version": "node ./scripts/build-version.js", |     "build:zip": "node ./scripts/build-version.js", | ||||||
|     "build": "yarn build:app && yarn build:version", |     "build": "npm run build:app && npm run build:zip", | ||||||
|     "eject": "react-scripts eject", |     "eject": "react-scripts eject", | ||||||
|     "fix:code": "yarn test:code --fix", |     "fix:code": "npm run test:code -- --fix", | ||||||
|     "fix:other": "yarn prettier --write", |     "fix:other": "npm run prettier -- --write", | ||||||
|     "fix": "yarn fix:other && yarn fix:code", |     "fix": "npm run fix:other && npm run fix:code", | ||||||
|     "locales-coverage": "node scripts/build-locales-coverage.js", |     "locales-coverage": "node scripts/build-locales-coverage.js", | ||||||
|     "locales-coverage:description": "node scripts/locales-coverage-description.js", |  | ||||||
|     "prettier": "prettier \"**/*.{css,scss,json,md,html,yml}\" --ignore-path=.eslintignore", |     "prettier": "prettier \"**/*.{css,scss,json,md,html,yml}\" --ignore-path=.eslintignore", | ||||||
|     "start": "react-scripts start", |     "start": "react-scripts start", | ||||||
|     "test:all": "yarn test:typecheck && yarn test:code && yarn test:other && yarn test:app --watchAll=false", |     "test:all": "npm run test:typecheck && npm run test:code && npm run test:other && npm run test:app -- --watchAll=false", | ||||||
|     "test:app": "react-scripts test --passWithNoTests", |     "test:app": "react-scripts test --passWithNoTests", | ||||||
|     "test:code": "eslint --max-warnings=0 --ext .js,.ts,.tsx .", |     "test:code": "eslint --max-warnings=0 --ignore-path .gitignore --ext .js,.ts,.tsx .", | ||||||
|     "test:debug": "react-scripts --inspect-brk test --runInBand --no-cache", |     "test:debug": "react-scripts --inspect-brk test --runInBand --no-cache", | ||||||
|     "test:other": "yarn prettier --list-different", |     "test:other": "npm run prettier -- --list-different", | ||||||
|     "test:typecheck": "tsc", |     "test:typecheck": "tsc", | ||||||
|     "test:update": "yarn test:app --updateSnapshot --watchAll=false", |     "test:update": "npm run test:app -- --updateSnapshot --watchAll=false", | ||||||
|     "test": "yarn test:app", |     "test": "npm run test:app", | ||||||
|     "autorelease": "node scripts/autorelease.js" |     "changelog:check": "node ./scripts/changelog-check.js" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								public/FG_Virgil.otf
									
									
									
									
									
										Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 4.3 KiB After Width: | Height: | Size: 9.0 KiB | 
| @@ -1,7 +1,7 @@ | |||||||
| /* http://www.eaglefonts.com/fg-virgil-ttf-131249.htm */ | /* http://www.eaglefonts.com/fg-virgil-ttf-131249.htm */ | ||||||
| @font-face { | @font-face { | ||||||
|   font-family: "Virgil"; |   font-family: "Virgil"; | ||||||
|   src: url("Virgil.woff2"); |   src: url("FG_Virgil.woff2"); | ||||||
|   font-display: swap; |   font-display: swap; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,18 +13,6 @@ | |||||||
|  |  | ||||||
|     <meta name="theme-color" content="#000" /> |     <meta name="theme-color" content="#000" /> | ||||||
|  |  | ||||||
|     <!-- Declarative Link Capturing (https://web.dev/declarative-link-capturing/) --> |  | ||||||
|     <meta |  | ||||||
|       http-equiv="origin-trial" |  | ||||||
|       content="Ak3VyzTheARtX2CnxBZ3ZKxImB0mNyvDakmMxeAChgxrWFMZ3IGN64VP+uj36VxM5OegsbLmrP258b1xvqp7+Q8AAABbeyJvcmlnaW4iOiJodHRwczovL2V4Y2FsaWRyYXcuY29tOjQ0MyIsImZlYXR1cmUiOiJXZWJBcHBMaW5rQ2FwdHVyaW5nIiwiZXhwaXJ5IjoxNjM0MDgzMTk5fQ==" |  | ||||||
|     /> |  | ||||||
|  |  | ||||||
|     <!-- File Handling (https://web.dev/file-handling/) --> |  | ||||||
|     <meta |  | ||||||
|       http-equiv="origin-trial" |  | ||||||
|       content="AkMQsAnFmKfRfPKQHNCv2WmZREqgwkqhyt2M7aOwQiCStB+hPYnGnv+mNbkPDAsGXrwsj/waFi76wPzTDUaEeQ0AAABUeyJvcmlnaW4iOiJodHRwczovL2V4Y2FsaWRyYXcuY29tOjQ0MyIsImZlYXR1cmUiOiJGaWxlSGFuZGxpbmciLCJleHBpcnkiOjE2MzQwODMxOTl9" |  | ||||||
|     /> |  | ||||||
|  |  | ||||||
|     <!-- General tags --> |     <!-- General tags --> | ||||||
|     <meta |     <meta | ||||||
|       name="description" |       name="description" | ||||||
| @@ -63,15 +51,13 @@ | |||||||
|       name="twitter:description" |       name="twitter:description" | ||||||
|       content="Excalidraw is a whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them." |       content="Excalidraw is a whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them." | ||||||
|     /> |     /> | ||||||
|  |     <!-- OG tags require absolute url for images --> | ||||||
|  |     <meta name="twitter:image" content="https://excalidraw.com/og-image.png" /> | ||||||
|     <link rel="shortcut icon" href="favicon.ico" type="image/x-icon" /> |     <link rel="shortcut icon" href="favicon.ico" type="image/x-icon" /> | ||||||
|  |  | ||||||
|     <!-- Excalidraw version --> |  | ||||||
|     <meta name="version" content="{version}" /> |  | ||||||
|  |  | ||||||
|     <link |     <link | ||||||
|       rel="preload" |       rel="preload" | ||||||
|       href="Virgil.woff2" |       href="FG_Virgil.woff2" | ||||||
|       as="font" |       as="font" | ||||||
|       type="font/woff2" |       type="font/woff2" | ||||||
|       crossorigin="anonymous" |       crossorigin="anonymous" | ||||||
| @@ -97,15 +83,11 @@ | |||||||
|     /> |     /> | ||||||
|  |  | ||||||
|     <link rel="stylesheet" href="fonts.css" type="text/css" /> |     <link rel="stylesheet" href="fonts.css" type="text/css" /> | ||||||
|     <script> |  | ||||||
|       window.EXCALIDRAW_ASSET_PATH = "/"; |     <% if (process.env.REACT_APP_INCLUDE_GTAG === 'true') { %> | ||||||
|       // setting this so that libraries installation reuses this window tab. |  | ||||||
|       window.name = "_excalidraw"; |  | ||||||
|     </script> |  | ||||||
|     <% if (process.env.REACT_APP_GOOGLE_ANALYTICS_ID) { %> |  | ||||||
|     <script |     <script | ||||||
|       async |       async | ||||||
|       src="https://www.googletagmanager.com/gtag/js?id=%REACT_APP_GOOGLE_ANALYTICS_ID%" |       src="https://www.googletagmanager.com/gtag/js?id=UA-387204-13" | ||||||
|     ></script> |     ></script> | ||||||
|     <script> |     <script> | ||||||
|       window.dataLayer = window.dataLayer || []; |       window.dataLayer = window.dataLayer || []; | ||||||
| @@ -113,23 +95,22 @@ | |||||||
|         dataLayer.push(arguments); |         dataLayer.push(arguments); | ||||||
|       } |       } | ||||||
|       gtag("js", new Date()); |       gtag("js", new Date()); | ||||||
|       gtag("config", "%REACT_APP_GOOGLE_ANALYTICS_ID%"); |       gtag("config", "UA-387204-13"); | ||||||
|     </script> |     </script> | ||||||
|     <% } %> |     <% } %> | ||||||
|  |  | ||||||
|     <!-- FIXME: remove this when we update CRA (fix SW caching) --> |     <!-- FIXME: remove this when we update CRA (fix SW caching) --> | ||||||
|     <style> |     <style> | ||||||
|       body, |       body { | ||||||
|       html { |  | ||||||
|         margin: 0; |         margin: 0; | ||||||
|         --ui-font: system-ui, BlinkMacSystemFont, -apple-system, Segoe UI, |         --ui-font: system-ui, BlinkMacSystemFont, -apple-system, Segoe UI, | ||||||
|           Roboto, Helvetica, Arial, sans-serif; |           Roboto, Helvetica, Arial, sans-serif; | ||||||
|         font-family: var(--ui-font); |         font-family: var(--ui-font); | ||||||
|         -webkit-text-size-adjust: 100%; |         -webkit-text-size-adjust: 100%; | ||||||
|  |         -webkit-user-select: none; | ||||||
|         width: 100%; |         user-select: none; | ||||||
|         height: 100%; |         width: 100vw; | ||||||
|         overflow: hidden; |         height: 100vh; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .visually-hidden { |       .visually-hidden { | ||||||
| @@ -139,7 +120,6 @@ | |||||||
|         overflow: hidden; |         overflow: hidden; | ||||||
|         clip: rect(1px, 1px, 1px, 1px); |         clip: rect(1px, 1px, 1px, 1px); | ||||||
|         white-space: nowrap; /* added line */ |         white-space: nowrap; /* added line */ | ||||||
|         user-select: none; |  | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .LoadingMessage { |       .LoadingMessage { | ||||||
| @@ -162,24 +142,6 @@ | |||||||
|         color: var(--popup-text-color); |         color: var(--popup-text-color); | ||||||
|         font-size: 1.3em; |         font-size: 1.3em; | ||||||
|       } |       } | ||||||
|       #root { |  | ||||||
|         height: 100%; |  | ||||||
|         -webkit-touch-callout: none; |  | ||||||
|         -webkit-user-select: none; |  | ||||||
|         -khtml-user-select: none; |  | ||||||
|         -moz-user-select: none; |  | ||||||
|         -ms-user-select: none; |  | ||||||
|         user-select: none; |  | ||||||
|  |  | ||||||
|         @media screen and (min-width: 1200px) { |  | ||||||
|           -webkit-touch-callout: default; |  | ||||||
|           -webkit-user-select: auto; |  | ||||||
|           -khtml-user-select: auto; |  | ||||||
|           -moz-user-select: auto; |  | ||||||
|           -ms-user-select: auto; |  | ||||||
|           user-select: auto; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     </style> |     </style> | ||||||
|   </head> |   </head> | ||||||
|  |  | ||||||
|   | |||||||
| Before Width: | Height: | Size: 2.6 KiB After Width: | Height: | Size: 6.4 KiB | 
| @@ -25,51 +25,5 @@ | |||||||
|         "application/vnd.excalidraw+json": [".excalidraw"] |         "application/vnd.excalidraw+json": [".excalidraw"] | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   ], |  | ||||||
|   "capture_links": "new-client", |  | ||||||
|   "share_target": { |  | ||||||
|     "action": "/web-share-target", |  | ||||||
|     "method": "POST", |  | ||||||
|     "enctype": "multipart/form-data", |  | ||||||
|     "params": { |  | ||||||
|       "files": [ |  | ||||||
|         { |  | ||||||
|           "name": "file", |  | ||||||
|           "accept": ["application/vnd.excalidraw+json", "application/json", ".excalidraw"] |  | ||||||
|         } |  | ||||||
|       ] |  | ||||||
|     } |  | ||||||
|   }, |  | ||||||
|   "screenshots": [ |  | ||||||
|     { |  | ||||||
|       "src": "/screenshots/virtual-whiteboard.png", |  | ||||||
|       "type": "image/png", |  | ||||||
|       "sizes": "462x945" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "src": "/screenshots/wireframe.png", |  | ||||||
|       "type": "image/png", |  | ||||||
|       "sizes": "462x945" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "src": "/screenshots/illustration.png", |  | ||||||
|       "type": "image/png", |  | ||||||
|       "sizes": "462x945" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "src": "/screenshots/shapes.png", |  | ||||||
|       "type": "image/png", |  | ||||||
|       "sizes": "462x945" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "src": "/screenshots/collaboration.png", |  | ||||||
|       "type": "image/png", |  | ||||||
|       "sizes": "462x945" |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "src": "/screenshots/export.png", |  | ||||||
|       "type": "image/png", |  | ||||||
|       "sizes": "462x945" |  | ||||||
|     } |  | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
| Before Width: | Height: | Size: 76 KiB | 
| Before Width: | Height: | Size: 70 KiB After Width: | Height: | Size: 83 KiB | 
| Before Width: | Height: | Size: 28 KiB | 
| Before Width: | Height: | Size: 25 KiB | 
| Before Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 25 KiB | 
| Before Width: | Height: | Size: 27 KiB | 
| Before Width: | Height: | Size: 27 KiB | 
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.backgroundSync=function(t,e,s){"use strict";try{self["workbox:background-sync:4.3.1"]&&_()}catch(t){}const i=3,n="workbox-background-sync",a="requests",r="queueName";class c{constructor(t){this.t=t,this.s=new s.DBWrapper(n,i,{onupgradeneeded:this.i})}async pushEntry(t){delete t.id,t.queueName=this.t,await this.s.add(a,t)}async unshiftEntry(t){const[e]=await this.s.getAllMatching(a,{count:1});e?t.id=e.id-1:delete t.id,t.queueName=this.t,await this.s.add(a,t)}async popEntry(){return this.h({direction:"prev"})}async shiftEntry(){return this.h({direction:"next"})}async getAll(){return await this.s.getAllMatching(a,{index:r,query:IDBKeyRange.only(this.t)})}async deleteEntry(t){await this.s.delete(a,t)}async h({direction:t}){const[e]=await this.s.getAllMatching(a,{direction:t,index:r,query:IDBKeyRange.only(this.t),count:1});if(e)return await this.deleteEntry(e.id),e}i(t){const e=t.target.result;t.oldVersion>0&&t.oldVersion<i&&e.objectStoreNames.contains(a)&&e.deleteObjectStore(a),e.createObjectStore(a,{autoIncrement:!0,keyPath:"id"}).createIndex(r,r,{unique:!1})}}const h=["method","referrer","referrerPolicy","mode","credentials","cache","redirect","integrity","keepalive"];class o{static async fromRequest(t){const e={url:t.url,headers:{}};"GET"!==t.method&&(e.body=await t.clone().arrayBuffer());for(const[s,i]of t.headers.entries())e.headers[s]=i;for(const s of h)void 0!==t[s]&&(e[s]=t[s]);return new o(e)}constructor(t){"navigate"===t.mode&&(t.mode="same-origin"),this.o=t}toObject(){const t=Object.assign({},this.o);return t.headers=Object.assign({},this.o.headers),t.body&&(t.body=t.body.slice(0)),t}toRequest(){return new Request(this.o.url,this.o)}clone(){return new o(this.toObject())}}const u="workbox-background-sync",y=10080,w=new Set;class d{constructor(t,{onSync:s,maxRetentionTime:i}={}){if(w.has(t))throw new e.WorkboxError("duplicate-queue-name",{name:t});w.add(t),this.u=t,this.l=s||this.replayRequests,this.q=i||y,this.m=new c(this.u),this.p()}get name(){return this.u}async pushRequest(t){await this.g(t,"push")}async unshiftRequest(t){await this.g(t,"unshift")}async popRequest(){return this.R("pop")}async shiftRequest(){return this.R("shift")}async getAll(){const t=await this.m.getAll(),e=Date.now(),s=[];for(const i of t){const t=60*this.q*1e3;e-i.timestamp>t?await this.m.deleteEntry(i.id):s.push(f(i))}return s}async g({request:t,metadata:e,timestamp:s=Date.now()},i){const n={requestData:(await o.fromRequest(t.clone())).toObject(),timestamp:s};e&&(n.metadata=e),await this.m[`${i}Entry`](n),this.k?this.D=!0:await this.registerSync()}async R(t){const e=Date.now(),s=await this.m[`${t}Entry`]();if(s){const i=60*this.q*1e3;return e-s.timestamp>i?this.R(t):f(s)}}async replayRequests(){let t;for(;t=await this.shiftRequest();)try{await fetch(t.request.clone())}catch(s){throw await this.unshiftRequest(t),new e.WorkboxError("queue-replay-failed",{name:this.u})}}async registerSync(){if("sync"in registration)try{await registration.sync.register(`${u}:${this.u}`)}catch(t){}}p(){"sync"in registration?self.addEventListener("sync",t=>{if(t.tag===`${u}:${this.u}`){const e=async()=>{let e;this.k=!0;try{await this.l({queue:this})}catch(t){throw e=t}finally{!this.D||e&&!t.lastChance||await this.registerSync(),this.k=!1,this.D=!1}};t.waitUntil(e())}}):this.l({queue:this})}static get _(){return w}}const f=t=>{const e={request:new o(t.requestData).toRequest(),timestamp:t.timestamp};return t.metadata&&(e.metadata=t.metadata),e};return t.Queue=d,t.Plugin=class{constructor(...t){this.v=new d(...t),this.fetchDidFail=this.fetchDidFail.bind(this)}async fetchDidFail({request:t}){await this.v.pushRequest({request:t})}},t}({},workbox.core._private,workbox.core._private); |  | ||||||
| //# sourceMappingURL=workbox-background-sync.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.broadcastUpdate=function(e,t){"use strict";try{self["workbox:broadcast-update:4.3.1"]&&_()}catch(e){}const s=(e,t,s)=>{return!s.some(s=>e.headers.has(s)&&t.headers.has(s))||s.every(s=>{const n=e.headers.has(s)===t.headers.has(s),a=e.headers.get(s)===t.headers.get(s);return n&&a})},n="workbox",a=1e4,i=["content-length","etag","last-modified"],o=async({channel:e,cacheName:t,url:s})=>{const n={type:"CACHE_UPDATED",meta:"workbox-broadcast-update",payload:{cacheName:t,updatedURL:s}};if(e)e.postMessage(n);else{const e=await clients.matchAll({type:"window"});for(const t of e)t.postMessage(n)}};class c{constructor({headersToCheck:e,channelName:t,deferNoticationTimeout:s}={}){this.t=e||i,this.s=t||n,this.i=s||a,this.o()}notifyIfUpdated({oldResponse:e,newResponse:t,url:n,cacheName:a,event:i}){if(!s(e,t,this.t)){const e=(async()=>{i&&i.request&&"navigate"===i.request.mode&&await this.h(i),await this.l({channel:this.u(),cacheName:a,url:n})})();if(i)try{i.waitUntil(e)}catch(e){}return e}}async l(e){await o(e)}u(){return"BroadcastChannel"in self&&!this.p&&(this.p=new BroadcastChannel(this.s)),this.p}h(e){if(!this.m.has(e)){const s=new t.Deferred;this.m.set(e,s);const n=setTimeout(()=>{s.resolve()},this.i);s.promise.then(()=>clearTimeout(n))}return this.m.get(e).promise}o(){this.m=new Map,self.addEventListener("message",e=>{if("WINDOW_READY"===e.data.type&&"workbox-window"===e.data.meta&&this.m.size>0){for(const e of this.m.values())e.resolve();this.m.clear()}})}}return e.BroadcastCacheUpdate=c,e.Plugin=class{constructor(e){this.l=new c(e)}cacheDidUpdate({cacheName:e,oldResponse:t,newResponse:s,request:n,event:a}){t&&this.l.notifyIfUpdated({cacheName:e,oldResponse:t,newResponse:s,event:a,url:n.url})}},e.broadcastUpdate=o,e.responsesAreSame=s,e}({},workbox.core._private); |  | ||||||
| //# sourceMappingURL=workbox-broadcast-update.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.cacheableResponse=function(t){"use strict";try{self["workbox:cacheable-response:4.3.1"]&&_()}catch(t){}class s{constructor(t={}){this.t=t.statuses,this.s=t.headers}isResponseCacheable(t){let s=!0;return this.t&&(s=this.t.includes(t.status)),this.s&&s&&(s=Object.keys(this.s).some(s=>t.headers.get(s)===this.s[s])),s}}return t.CacheableResponse=s,t.Plugin=class{constructor(t){this.i=new s(t)}cacheWillUpdate({response:t}){return this.i.isResponseCacheable(t)?t:null}},t}({}); |  | ||||||
| //# sourceMappingURL=workbox-cacheable-response.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.expiration=function(t,e,s,i,a,n){"use strict";try{self["workbox:expiration:4.3.1"]&&_()}catch(t){}const h="workbox-expiration",c="cache-entries",r=t=>{const e=new URL(t,location);return e.hash="",e.href};class o{constructor(t){this.t=t,this.s=new e.DBWrapper(h,1,{onupgradeneeded:t=>this.i(t)})}i(t){const e=t.target.result.createObjectStore(c,{keyPath:"id"});e.createIndex("cacheName","cacheName",{unique:!1}),e.createIndex("timestamp","timestamp",{unique:!1}),s.deleteDatabase(this.t)}async setTimestamp(t,e){t=r(t),await this.s.put(c,{url:t,timestamp:e,cacheName:this.t,id:this.h(t)})}async getTimestamp(t){return(await this.s.get(c,this.h(t))).timestamp}async expireEntries(t,e){const s=await this.s.transaction(c,"readwrite",(s,i)=>{const a=s.objectStore(c),n=[];let h=0;a.index("timestamp").openCursor(null,"prev").onsuccess=(({target:s})=>{const a=s.result;if(a){const s=a.value;s.cacheName===this.t&&(t&&s.timestamp<t||e&&h>=e?n.push(a.value):h++),a.continue()}else i(n)})}),i=[];for(const t of s)await this.s.delete(c,t.id),i.push(t.url);return i}h(t){return this.t+"|"+r(t)}}class u{constructor(t,e={}){this.o=!1,this.u=!1,this.l=e.maxEntries,this.p=e.maxAgeSeconds,this.t=t,this.m=new o(t)}async expireEntries(){if(this.o)return void(this.u=!0);this.o=!0;const t=this.p?Date.now()-1e3*this.p:void 0,e=await this.m.expireEntries(t,this.l),s=await caches.open(this.t);for(const t of e)await s.delete(t);this.o=!1,this.u&&(this.u=!1,this.expireEntries())}async updateTimestamp(t){await this.m.setTimestamp(t,Date.now())}async isURLExpired(t){return await this.m.getTimestamp(t)<Date.now()-1e3*this.p}async delete(){this.u=!1,await this.m.expireEntries(1/0)}}return t.CacheExpiration=u,t.Plugin=class{constructor(t={}){this.D=t,this.p=t.maxAgeSeconds,this.g=new Map,t.purgeOnQuotaError&&n.registerQuotaErrorCallback(()=>this.deleteCacheAndMetadata())}k(t){if(t===a.cacheNames.getRuntimeName())throw new i.WorkboxError("expire-custom-caches-only");let e=this.g.get(t);return e||(e=new u(t,this.D),this.g.set(t,e)),e}cachedResponseWillBeUsed({event:t,request:e,cacheName:s,cachedResponse:i}){if(!i)return null;let a=this.N(i);const n=this.k(s);n.expireEntries();const h=n.updateTimestamp(e.url);if(t)try{t.waitUntil(h)}catch(t){}return a?i:null}N(t){if(!this.p)return!0;const e=this._(t);return null===e||e>=Date.now()-1e3*this.p}_(t){if(!t.headers.has("date"))return null;const e=t.headers.get("date"),s=new Date(e).getTime();return isNaN(s)?null:s}async cacheDidUpdate({cacheName:t,request:e}){const s=this.k(t);await s.updateTimestamp(e.url),await s.expireEntries()}async deleteCacheAndMetadata(){for(const[t,e]of this.g)await caches.delete(t),await e.delete();this.g=new Map}},t}({},workbox.core._private,workbox.core._private,workbox.core._private,workbox.core._private,workbox.core); |  | ||||||
| //# sourceMappingURL=workbox-expiration.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.navigationPreload=function(t){"use strict";try{self["workbox:navigation-preload:4.3.1"]&&_()}catch(t){}function e(){return Boolean(self.registration&&self.registration.navigationPreload)}return t.disable=function(){e()&&self.addEventListener("activate",t=>{t.waitUntil(self.registration.navigationPreload.disable().then(()=>{}))})},t.enable=function(t){e()&&self.addEventListener("activate",e=>{e.waitUntil(self.registration.navigationPreload.enable().then(()=>{t&&self.registration.navigationPreload.setHeaderValue(t)}))})},t.isSupported=e,t}({}); |  | ||||||
| //# sourceMappingURL=workbox-navigation-preload.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.googleAnalytics=function(e,t,o,n,a,c,w){"use strict";try{self["workbox:google-analytics:4.3.1"]&&_()}catch(e){}const r=/^\/(\w+\/)?collect/,s=e=>async({queue:t})=>{let o;for(;o=await t.shiftRequest();){const{request:n,timestamp:a}=o,c=new URL(n.url);try{const w="POST"===n.method?new URLSearchParams(await n.clone().text()):c.searchParams,r=a-(Number(w.get("qt"))||0),s=Date.now()-r;if(w.set("qt",s),e.parameterOverrides)for(const t of Object.keys(e.parameterOverrides)){const o=e.parameterOverrides[t];w.set(t,o)}"function"==typeof e.hitFilter&&e.hitFilter.call(null,w),await fetch(new Request(c.origin+c.pathname,{body:w.toString(),method:"POST",mode:"cors",credentials:"omit",headers:{"Content-Type":"text/plain"}}))}catch(e){throw await t.unshiftRequest(o),e}}},i=e=>{const t=({url:e})=>"www.google-analytics.com"===e.hostname&&r.test(e.pathname),o=new w.NetworkOnly({plugins:[e]});return[new n.Route(t,o,"GET"),new n.Route(t,o,"POST")]},l=e=>{const t=new c.NetworkFirst({cacheName:e});return new n.Route(({url:e})=>"www.google-analytics.com"===e.hostname&&"/analytics.js"===e.pathname,t,"GET")},m=e=>{const t=new c.NetworkFirst({cacheName:e});return new n.Route(({url:e})=>"www.googletagmanager.com"===e.hostname&&"/gtag/js"===e.pathname,t,"GET")},u=e=>{const t=new c.NetworkFirst({cacheName:e});return new n.Route(({url:e})=>"www.googletagmanager.com"===e.hostname&&"/gtm.js"===e.pathname,t,"GET")};return e.initialize=((e={})=>{const n=o.cacheNames.getGoogleAnalyticsName(e.cacheName),c=new t.Plugin("workbox-google-analytics",{maxRetentionTime:2880,onSync:s(e)}),w=[u(n),l(n),m(n),...i(c)],r=new a.Router;for(const e of w)r.registerRoute(e);r.addFetchListener()}),e}({},workbox.backgroundSync,workbox.core._private,workbox.routing,workbox.routing,workbox.strategies,workbox.strategies); |  | ||||||
| //# sourceMappingURL=workbox-offline-ga.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.precaching=function(t,e,n,s,c){"use strict";try{self["workbox:precaching:4.3.1"]&&_()}catch(t){}const o=[],i={get:()=>o,add(t){o.push(...t)}};const a="__WB_REVISION__";function r(t){if(!t)throw new c.WorkboxError("add-to-cache-list-unexpected-type",{entry:t});if("string"==typeof t){const e=new URL(t,location);return{cacheKey:e.href,url:e.href}}const{revision:e,url:n}=t;if(!n)throw new c.WorkboxError("add-to-cache-list-unexpected-type",{entry:t});if(!e){const t=new URL(n,location);return{cacheKey:t.href,url:t.href}}const s=new URL(n,location),o=new URL(n,location);return o.searchParams.set(a,e),{cacheKey:o.href,url:s.href}}class l{constructor(t){this.t=e.cacheNames.getPrecacheName(t),this.s=new Map}addToCacheList(t){for(const e of t){const{cacheKey:t,url:n}=r(e);if(this.s.has(n)&&this.s.get(n)!==t)throw new c.WorkboxError("add-to-cache-list-conflicting-entries",{firstEntry:this.s.get(n),secondEntry:t});this.s.set(n,t)}}async install({event:t,plugins:e}={}){const n=[],s=[],c=await caches.open(this.t),o=await c.keys(),i=new Set(o.map(t=>t.url));for(const t of this.s.values())i.has(t)?s.push(t):n.push(t);const a=n.map(n=>this.o({event:t,plugins:e,url:n}));return await Promise.all(a),{updatedURLs:n,notUpdatedURLs:s}}async activate(){const t=await caches.open(this.t),e=await t.keys(),n=new Set(this.s.values()),s=[];for(const c of e)n.has(c.url)||(await t.delete(c),s.push(c.url));return{deletedURLs:s}}async o({url:t,event:e,plugins:o}){const i=new Request(t,{credentials:"same-origin"});let a,r=await s.fetchWrapper.fetch({event:e,plugins:o,request:i});for(const t of o||[])"cacheWillUpdate"in t&&(a=t.cacheWillUpdate.bind(t));if(!(a?a({event:e,request:i,response:r}):r.status<400))throw new c.WorkboxError("bad-precaching-response",{url:t,status:r.status});r.redirected&&(r=await async function(t){const e=t.clone(),n="body"in e?Promise.resolve(e.body):e.blob(),s=await n;return new Response(s,{headers:e.headers,status:e.status,statusText:e.statusText})}(r)),await n.cacheWrapper.put({event:e,plugins:o,request:i,response:r,cacheName:this.t,matchOptions:{ignoreSearch:!0}})}getURLsToCacheKeys(){return this.s}getCachedURLs(){return[...this.s.keys()]}getCacheKeyForURL(t){const e=new URL(t,location);return this.s.get(e.href)}}let u;const h=()=>(u||(u=new l),u);const d=(t,e)=>{const n=h().getURLsToCacheKeys();for(const s of function*(t,{ignoreURLParametersMatching:e,directoryIndex:n,cleanURLs:s,urlManipulation:c}={}){const o=new URL(t,location);o.hash="",yield o.href;const i=function(t,e){for(const n of[...t.searchParams.keys()])e.some(t=>t.test(n))&&t.searchParams.delete(n);return t}(o,e);if(yield i.href,n&&i.pathname.endsWith("/")){const t=new URL(i);t.pathname+=n,yield t.href}if(s){const t=new URL(i);t.pathname+=".html",yield t.href}if(c){const t=c({url:o});for(const e of t)yield e.href}}(t,e)){const t=n.get(s);if(t)return t}};let w=!1;const f=t=>{w||((({ignoreURLParametersMatching:t=[/^utm_/],directoryIndex:n="index.html",cleanURLs:s=!0,urlManipulation:c=null}={})=>{const o=e.cacheNames.getPrecacheName();addEventListener("fetch",e=>{const i=d(e.request.url,{cleanURLs:s,directoryIndex:n,ignoreURLParametersMatching:t,urlManipulation:c});if(!i)return;let a=caches.open(o).then(t=>t.match(i)).then(t=>t||fetch(i));e.respondWith(a)})})(t),w=!0)},y=t=>{const e=h(),n=i.get();t.waitUntil(e.install({event:t,plugins:n}).catch(t=>{throw t}))},p=t=>{const e=h(),n=i.get();t.waitUntil(e.activate({event:t,plugins:n}))},L=t=>{h().addToCacheList(t),t.length>0&&(addEventListener("install",y),addEventListener("activate",p))};return t.addPlugins=(t=>{i.add(t)}),t.addRoute=f,t.cleanupOutdatedCaches=(()=>{addEventListener("activate",t=>{const n=e.cacheNames.getPrecacheName();t.waitUntil((async(t,e="-precache-")=>{const n=(await caches.keys()).filter(n=>n.includes(e)&&n.includes(self.registration.scope)&&n!==t);return await Promise.all(n.map(t=>caches.delete(t))),n})(n).then(t=>{}))})}),t.getCacheKeyForURL=(t=>{return h().getCacheKeyForURL(t)}),t.precache=L,t.precacheAndRoute=((t,e)=>{L(t),f(e)}),t.PrecacheController=l,t}({},workbox.core._private,workbox.core._private,workbox.core._private,workbox.core._private); |  | ||||||
| //# sourceMappingURL=workbox-precaching.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.rangeRequests=function(e,n){"use strict";try{self["workbox:range-requests:4.3.1"]&&_()}catch(e){}async function t(e,t){try{if(206===t.status)return t;const s=e.headers.get("range");if(!s)throw new n.WorkboxError("no-range-header");const a=function(e){const t=e.trim().toLowerCase();if(!t.startsWith("bytes="))throw new n.WorkboxError("unit-must-be-bytes",{normalizedRangeHeader:t});if(t.includes(","))throw new n.WorkboxError("single-range-only",{normalizedRangeHeader:t});const s=/(\d*)-(\d*)/.exec(t);if(null===s||!s[1]&&!s[2])throw new n.WorkboxError("invalid-range-values",{normalizedRangeHeader:t});return{start:""===s[1]?null:Number(s[1]),end:""===s[2]?null:Number(s[2])}}(s),r=await t.blob(),i=function(e,t,s){const a=e.size;if(s>a||t<0)throw new n.WorkboxError("range-not-satisfiable",{size:a,end:s,start:t});let r,i;return null===t?(r=a-s,i=a):null===s?(r=t,i=a):(r=t,i=s+1),{start:r,end:i}}(r,a.start,a.end),o=r.slice(i.start,i.end),u=o.size,l=new Response(o,{status:206,statusText:"Partial Content",headers:t.headers});return l.headers.set("Content-Length",u),l.headers.set("Content-Range",`bytes ${i.start}-${i.end-1}/`+r.size),l}catch(e){return new Response("",{status:416,statusText:"Range Not Satisfiable"})}}return e.createPartialResponse=t,e.Plugin=class{async cachedResponseWillBeUsed({request:e,cachedResponse:n}){return n&&e.headers.has("range")?await t(e,n):n}},e}({},workbox.core._private); |  | ||||||
| //# sourceMappingURL=workbox-range-requests.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.routing=function(t,e,r){"use strict";try{self["workbox:routing:4.3.1"]&&_()}catch(t){}const s="GET",n=t=>t&&"object"==typeof t?t:{handle:t};class o{constructor(t,e,r){this.handler=n(e),this.match=t,this.method=r||s}}class i extends o{constructor(t,{whitelist:e=[/./],blacklist:r=[]}={}){super(t=>this.t(t),t),this.s=e,this.o=r}t({url:t,request:e}){if("navigate"!==e.mode)return!1;const r=t.pathname+t.search;for(const t of this.o)if(t.test(r))return!1;return!!this.s.some(t=>t.test(r))}}class u extends o{constructor(t,e,r){super(({url:e})=>{const r=t.exec(e.href);return r?e.origin!==location.origin&&0!==r.index?null:r.slice(1):null},e,r)}}class c{constructor(){this.i=new Map}get routes(){return this.i}addFetchListener(){self.addEventListener("fetch",t=>{const{request:e}=t,r=this.handleRequest({request:e,event:t});r&&t.respondWith(r)})}addCacheListener(){self.addEventListener("message",async t=>{if(t.data&&"CACHE_URLS"===t.data.type){const{payload:e}=t.data,r=Promise.all(e.urlsToCache.map(t=>{"string"==typeof t&&(t=[t]);const e=new Request(...t);return this.handleRequest({request:e})}));t.waitUntil(r),t.ports&&t.ports[0]&&(await r,t.ports[0].postMessage(!0))}})}handleRequest({request:t,event:e}){const r=new URL(t.url,location);if(!r.protocol.startsWith("http"))return;let s,{params:n,route:o}=this.findMatchingRoute({url:r,request:t,event:e}),i=o&&o.handler;if(!i&&this.u&&(i=this.u),i){try{s=i.handle({url:r,request:t,event:e,params:n})}catch(t){s=Promise.reject(t)}return s&&this.h&&(s=s.catch(t=>this.h.handle({url:r,event:e,err:t}))),s}}findMatchingRoute({url:t,request:e,event:r}){const s=this.i.get(e.method)||[];for(const n of s){let s,o=n.match({url:t,request:e,event:r});if(o)return Array.isArray(o)&&o.length>0?s=o:o.constructor===Object&&Object.keys(o).length>0&&(s=o),{route:n,params:s}}return{}}setDefaultHandler(t){this.u=n(t)}setCatchHandler(t){this.h=n(t)}registerRoute(t){this.i.has(t.method)||this.i.set(t.method,[]),this.i.get(t.method).push(t)}unregisterRoute(t){if(!this.i.has(t.method))throw new r.WorkboxError("unregister-route-but-not-found-with-method",{method:t.method});const e=this.i.get(t.method).indexOf(t);if(!(e>-1))throw new r.WorkboxError("unregister-route-route-not-registered");this.i.get(t.method).splice(e,1)}}let a;const h=()=>(a||((a=new c).addFetchListener(),a.addCacheListener()),a);return t.NavigationRoute=i,t.RegExpRoute=u,t.registerNavigationRoute=((t,r={})=>{const s=e.cacheNames.getPrecacheName(r.cacheName),n=new i(async()=>{try{const e=await caches.match(t,{cacheName:s});if(e)return e;throw new Error(`The cache ${s} did not have an entry for `+`${t}.`)}catch(e){return fetch(t)}},{whitelist:r.whitelist,blacklist:r.blacklist});return h().registerRoute(n),n}),t.registerRoute=((t,e,s="GET")=>{let n;if("string"==typeof t){const r=new URL(t,location);n=new o(({url:t})=>t.href===r.href,e,s)}else if(t instanceof RegExp)n=new u(t,e,s);else if("function"==typeof t)n=new o(t,e,s);else{if(!(t instanceof o))throw new r.WorkboxError("unsupported-route-type",{moduleName:"workbox-routing",funcName:"registerRoute",paramName:"capture"});n=t}return h().registerRoute(n),n}),t.Route=o,t.Router=c,t.setCatchHandler=(t=>{h().setCatchHandler(t)}),t.setDefaultHandler=(t=>{h().setDefaultHandler(t)}),t}({},workbox.core._private,workbox.core._private); |  | ||||||
| //# sourceMappingURL=workbox-routing.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.strategies=function(e,t,s,n,r){"use strict";try{self["workbox:strategies:4.3.1"]&&_()}catch(e){}class i{constructor(e={}){this.t=t.cacheNames.getRuntimeName(e.cacheName),this.s=e.plugins||[],this.i=e.fetchOptions||null,this.h=e.matchOptions||null}async handle({event:e,request:t}){return this.makeRequest({event:e,request:t||e.request})}async makeRequest({event:e,request:t}){"string"==typeof t&&(t=new Request(t));let n,i=await s.cacheWrapper.match({cacheName:this.t,request:t,event:e,matchOptions:this.h,plugins:this.s});if(!i)try{i=await this.u(t,e)}catch(e){n=e}if(!i)throw new r.WorkboxError("no-response",{url:t.url,error:n});return i}async u(e,t){const r=await n.fetchWrapper.fetch({request:e,event:t,fetchOptions:this.i,plugins:this.s}),i=r.clone(),h=s.cacheWrapper.put({cacheName:this.t,request:e,response:i,event:t,plugins:this.s});if(t)try{t.waitUntil(h)}catch(e){}return r}}class h{constructor(e={}){this.t=t.cacheNames.getRuntimeName(e.cacheName),this.s=e.plugins||[],this.h=e.matchOptions||null}async handle({event:e,request:t}){return this.makeRequest({event:e,request:t||e.request})}async makeRequest({event:e,request:t}){"string"==typeof t&&(t=new Request(t));const n=await s.cacheWrapper.match({cacheName:this.t,request:t,event:e,matchOptions:this.h,plugins:this.s});if(!n)throw new r.WorkboxError("no-response",{url:t.url});return n}}const u={cacheWillUpdate:({response:e})=>200===e.status||0===e.status?e:null};class a{constructor(e={}){if(this.t=t.cacheNames.getRuntimeName(e.cacheName),e.plugins){let t=e.plugins.some(e=>!!e.cacheWillUpdate);this.s=t?e.plugins:[u,...e.plugins]}else this.s=[u];this.o=e.networkTimeoutSeconds,this.i=e.fetchOptions||null,this.h=e.matchOptions||null}async handle({event:e,request:t}){return this.makeRequest({event:e,request:t||e.request})}async makeRequest({event:e,request:t}){const s=[];"string"==typeof t&&(t=new Request(t));const n=[];let i;if(this.o){const{id:r,promise:h}=this.l({request:t,event:e,logs:s});i=r,n.push(h)}const h=this.q({timeoutId:i,request:t,event:e,logs:s});n.push(h);let u=await Promise.race(n);if(u||(u=await h),!u)throw new r.WorkboxError("no-response",{url:t.url});return u}l({request:e,logs:t,event:s}){let n;return{promise:new Promise(t=>{n=setTimeout(async()=>{t(await this.p({request:e,event:s}))},1e3*this.o)}),id:n}}async q({timeoutId:e,request:t,logs:r,event:i}){let h,u;try{u=await n.fetchWrapper.fetch({request:t,event:i,fetchOptions:this.i,plugins:this.s})}catch(e){h=e}if(e&&clearTimeout(e),h||!u)u=await this.p({request:t,event:i});else{const e=u.clone(),n=s.cacheWrapper.put({cacheName:this.t,request:t,response:e,event:i,plugins:this.s});if(i)try{i.waitUntil(n)}catch(e){}}return u}p({event:e,request:t}){return s.cacheWrapper.match({cacheName:this.t,request:t,event:e,matchOptions:this.h,plugins:this.s})}}class c{constructor(e={}){this.t=t.cacheNames.getRuntimeName(e.cacheName),this.s=e.plugins||[],this.i=e.fetchOptions||null}async handle({event:e,request:t}){return this.makeRequest({event:e,request:t||e.request})}async makeRequest({event:e,request:t}){let s,i;"string"==typeof t&&(t=new Request(t));try{i=await n.fetchWrapper.fetch({request:t,event:e,fetchOptions:this.i,plugins:this.s})}catch(e){s=e}if(!i)throw new r.WorkboxError("no-response",{url:t.url,error:s});return i}}class o{constructor(e={}){if(this.t=t.cacheNames.getRuntimeName(e.cacheName),this.s=e.plugins||[],e.plugins){let t=e.plugins.some(e=>!!e.cacheWillUpdate);this.s=t?e.plugins:[u,...e.plugins]}else this.s=[u];this.i=e.fetchOptions||null,this.h=e.matchOptions||null}async handle({event:e,request:t}){return this.makeRequest({event:e,request:t||e.request})}async makeRequest({event:e,request:t}){"string"==typeof t&&(t=new Request(t));const n=this.u({request:t,event:e});let i,h=await s.cacheWrapper.match({cacheName:this.t,request:t,event:e,matchOptions:this.h,plugins:this.s});if(h){if(e)try{e.waitUntil(n)}catch(i){}}else try{h=await n}catch(e){i=e}if(!h)throw new r.WorkboxError("no-response",{url:t.url,error:i});return h}async u({request:e,event:t}){const r=await n.fetchWrapper.fetch({request:e,event:t,fetchOptions:this.i,plugins:this.s}),i=s.cacheWrapper.put({cacheName:this.t,request:e,response:r.clone(),event:t,plugins:this.s});if(t)try{t.waitUntil(i)}catch(e){}return r}}const l={cacheFirst:i,cacheOnly:h,networkFirst:a,networkOnly:c,staleWhileRevalidate:o},q=e=>{const t=l[e];return e=>new t(e)},w=q("cacheFirst"),p=q("cacheOnly"),v=q("networkFirst"),y=q("networkOnly"),m=q("staleWhileRevalidate");return e.CacheFirst=i,e.CacheOnly=h,e.NetworkFirst=a,e.NetworkOnly=c,e.StaleWhileRevalidate=o,e.cacheFirst=w,e.cacheOnly=p,e.networkFirst=v,e.networkOnly=y,e.staleWhileRevalidate=m,e}({},workbox.core._private,workbox.core._private,workbox.core._private,workbox.core._private); |  | ||||||
| //# sourceMappingURL=workbox-strategies.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| this.workbox=this.workbox||{},this.workbox.streams=function(e){"use strict";try{self["workbox:streams:4.3.1"]&&_()}catch(e){}function n(e){const n=e.map(e=>Promise.resolve(e).then(e=>(function(e){return e.body&&e.body.getReader?e.body.getReader():e.getReader?e.getReader():new Response(e).body.getReader()})(e)));let t,r;const s=new Promise((e,n)=>{t=e,r=n});let o=0;return{done:s,stream:new ReadableStream({pull(e){return n[o].then(e=>e.read()).then(r=>{if(r.done)return++o>=n.length?(e.close(),void t()):this.pull(e);e.enqueue(r.value)}).catch(e=>{throw r(e),e})},cancel(){t()}})}}function t(e={}){const n=new Headers(e);return n.has("content-type")||n.set("content-type","text/html"),n}function r(e,r){const{done:s,stream:o}=n(e),a=t(r);return{done:s,response:new Response(o,{headers:a})}}let s=void 0;function o(){if(void 0===s)try{new ReadableStream({start(){}}),s=!0}catch(e){s=!1}return s}return e.concatenate=n,e.concatenateToResponse=r,e.isSupported=o,e.strategy=function(e,n){return async({event:s,url:a,params:c})=>{if(o()){const{done:t,response:o}=r(e.map(e=>e({event:s,url:a,params:c})),n);return s.waitUntil(t),o}const i=await Promise.all(e.map(e=>e({event:s,url:a,params:c})).map(async e=>{const n=await e;return n instanceof Response?n.blob():n})),u=t(n);return new Response(new Blob(i),{headers:u})}},e}({}); |  | ||||||
| //# sourceMappingURL=workbox-streams.prod.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| !function(){"use strict";try{self["workbox:sw:4.3.1"]&&_()}catch(t){}const t="https://storage.googleapis.com/workbox-cdn/releases/4.3.1",e={backgroundSync:"background-sync",broadcastUpdate:"broadcast-update",cacheableResponse:"cacheable-response",core:"core",expiration:"expiration",googleAnalytics:"offline-ga",navigationPreload:"navigation-preload",precaching:"precaching",rangeRequests:"range-requests",routing:"routing",strategies:"strategies",streams:"streams"};self.workbox=new class{constructor(){return this.v={},this.t={debug:"localhost"===self.location.hostname,modulePathPrefix:null,modulePathCb:null},this.s=this.t.debug?"dev":"prod",this.o=!1,new Proxy(this,{get(t,s){if(t[s])return t[s];const o=e[s];return o&&t.loadModule(`workbox-${o}`),t[s]}})}setConfig(t={}){if(this.o)throw new Error("Config must be set before accessing workbox.* modules");Object.assign(this.t,t),this.s=this.t.debug?"dev":"prod"}loadModule(t){const e=this.i(t);try{importScripts(e),this.o=!0}catch(s){throw console.error(`Unable to import module '${t}' from '${e}'.`),s}}i(e){if(this.t.modulePathCb)return this.t.modulePathCb(e,this.t.debug);let s=[t];const o=`${e}.${this.s}.js`,r=this.t.modulePathPrefix;return r&&""===(s=r.split("/"))[s.length-1]&&s.splice(s.length-1,1),s.push(o),s.join("/")}}}(); |  | ||||||
| //# sourceMappingURL=workbox-sw.js.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| try{self["workbox:window:4.3.1"]&&_()}catch(n){}var n=function(n,t){return new Promise(function(i){var e=new MessageChannel;e.port1.onmessage=function(n){return i(n.data)},n.postMessage(t,[e.port2])})};function t(n,t){for(var i=0;i<t.length;i++){var e=t[i];e.enumerable=e.enumerable||!1,e.configurable=!0,"value"in e&&(e.writable=!0),Object.defineProperty(n,e.key,e)}}function i(n){if(void 0===n)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return n}try{self["workbox:core:4.3.1"]&&_()}catch(n){}var e=function(){var n=this;this.promise=new Promise(function(t,i){n.resolve=t,n.reject=i})},r=function(n,t){return new URL(n,location).href===new URL(t,location).href},o=function(n,t){Object.assign(this,t,{type:n})};function u(n){return function(){for(var t=[],i=0;i<arguments.length;i++)t[i]=arguments[i];try{return Promise.resolve(n.apply(this,t))}catch(n){return Promise.reject(n)}}}function a(n,t,i){return i?t?t(n):n:(n&&n.then||(n=Promise.resolve(n)),t?n.then(t):n)}function s(){}var c=function(c){var f,h;function v(n,t){var r;return void 0===t&&(t={}),(r=c.call(this)||this).t=n,r.i=t,r.o=0,r.u=new e,r.s=new e,r.h=new e,r.v=r.v.bind(i(i(r))),r.l=r.l.bind(i(i(r))),r.g=r.g.bind(i(i(r))),r.m=r.m.bind(i(i(r))),r}h=c,(f=v).prototype=Object.create(h.prototype),f.prototype.constructor=f,f.__proto__=h;var l,w,g,d=v.prototype;return d.register=u(function(n){var t,i,e=this,u=(void 0===n?{}:n).immediate,c=void 0!==u&&u;return t=function(){return e.p=Boolean(navigator.serviceWorker.controller),e.P=e.R(),a(e.k(),function(n){e.B=n,e.P&&(e.O=e.P,e.s.resolve(e.P),e.h.resolve(e.P),e.j(e.P),e.P.addEventListener("statechange",e.l,{once:!0}));var t=e.B.waiting;return t&&r(t.scriptURL,e.t)&&(e.O=t,Promise.resolve().then(function(){e.dispatchEvent(new o("waiting",{sw:t,wasWaitingBeforeRegister:!0}))})),e.O&&e.u.resolve(e.O),e.B.addEventListener("updatefound",e.g),navigator.serviceWorker.addEventListener("controllerchange",e.m,{once:!0}),"BroadcastChannel"in self&&(e.C=new BroadcastChannel("workbox"),e.C.addEventListener("message",e.v)),navigator.serviceWorker.addEventListener("message",e.v),e.B})},(i=function(){if(!c&&"complete"!==document.readyState)return function(n,t){if(!t)return n&&n.then?n.then(s):Promise.resolve()}(new Promise(function(n){return addEventListener("load",n)}))}())&&i.then?i.then(t):t(i)}),d.getSW=u(function(){return this.O||this.u.promise}),d.messageSW=u(function(t){return a(this.getSW(),function(i){return n(i,t)})}),d.R=function(){var n=navigator.serviceWorker.controller;if(n&&r(n.scriptURL,this.t))return n},d.k=u(function(){var n=this;return function(n,t){try{var i=n()}catch(n){return t(n)}return i&&i.then?i.then(void 0,t):i}(function(){return a(navigator.serviceWorker.register(n.t,n.i),function(t){return n.L=performance.now(),t})},function(n){throw n})}),d.j=function(t){n(t,{type:"WINDOW_READY",meta:"workbox-window"})},d.g=function(){var n=this.B.installing;this.o>0||!r(n.scriptURL,this.t)||performance.now()>this.L+6e4?(this.W=n,this.B.removeEventListener("updatefound",this.g)):(this.O=n,this.u.resolve(n)),++this.o,n.addEventListener("statechange",this.l)},d.l=function(n){var t=this,i=n.target,e=i.state,r=i===this.W,u=r?"external":"",a={sw:i,originalEvent:n};!r&&this.p&&(a.isUpdate=!0),this.dispatchEvent(new o(u+e,a)),"installed"===e?this._=setTimeout(function(){"installed"===e&&t.B.waiting===i&&t.dispatchEvent(new o(u+"waiting",a))},200):"activating"===e&&(clearTimeout(this._),r||this.s.resolve(i))},d.m=function(n){var t=this.O;t===navigator.serviceWorker.controller&&(this.dispatchEvent(new o("controlling",{sw:t,originalEvent:n})),this.h.resolve(t))},d.v=function(n){var t=n.data;this.dispatchEvent(new o("message",{data:t,originalEvent:n}))},l=v,(w=[{key:"active",get:function(){return this.s.promise}},{key:"controlling",get:function(){return this.h.promise}}])&&t(l.prototype,w),g&&t(l,g),v}(function(){function n(){this.D={}}var t=n.prototype;return t.addEventListener=function(n,t){this.T(n).add(t)},t.removeEventListener=function(n,t){this.T(n).delete(t)},t.dispatchEvent=function(n){n.target=this,this.T(n.type).forEach(function(t){return t(n)})},t.T=function(n){return this.D[n]=this.D[n]||new Set},n}());export{c as Workbox,n as messageSW}; |  | ||||||
| //# sourceMappingURL=workbox-window.prod.es5.mjs.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| try{self["workbox:window:4.3.1"]&&_()}catch(t){}const t=(t,s)=>new Promise(i=>{let e=new MessageChannel;e.port1.onmessage=(t=>i(t.data)),t.postMessage(s,[e.port2])});try{self["workbox:core:4.3.1"]&&_()}catch(t){}class s{constructor(){this.promise=new Promise((t,s)=>{this.resolve=t,this.reject=s})}}class i{constructor(){this.t={}}addEventListener(t,s){this.s(t).add(s)}removeEventListener(t,s){this.s(t).delete(s)}dispatchEvent(t){t.target=this,this.s(t.type).forEach(s=>s(t))}s(t){return this.t[t]=this.t[t]||new Set}}const e=(t,s)=>new URL(t,location).href===new URL(s,location).href;class n{constructor(t,s){Object.assign(this,s,{type:t})}}const h=200,a=6e4;class o extends i{constructor(t,i={}){super(),this.i=t,this.h=i,this.o=0,this.l=new s,this.g=new s,this.u=new s,this.m=this.m.bind(this),this.v=this.v.bind(this),this.p=this.p.bind(this),this._=this._.bind(this)}async register({immediate:t=!1}={}){t||"complete"===document.readyState||await new Promise(t=>addEventListener("load",t)),this.C=Boolean(navigator.serviceWorker.controller),this.W=this.L(),this.S=await this.B(),this.W&&(this.R=this.W,this.g.resolve(this.W),this.u.resolve(this.W),this.P(this.W),this.W.addEventListener("statechange",this.v,{once:!0}));const s=this.S.waiting;return s&&e(s.scriptURL,this.i)&&(this.R=s,Promise.resolve().then(()=>{this.dispatchEvent(new n("waiting",{sw:s,wasWaitingBeforeRegister:!0}))})),this.R&&this.l.resolve(this.R),this.S.addEventListener("updatefound",this.p),navigator.serviceWorker.addEventListener("controllerchange",this._,{once:!0}),"BroadcastChannel"in self&&(this.T=new BroadcastChannel("workbox"),this.T.addEventListener("message",this.m)),navigator.serviceWorker.addEventListener("message",this.m),this.S}get active(){return this.g.promise}get controlling(){return this.u.promise}async getSW(){return this.R||this.l.promise}async messageSW(s){const i=await this.getSW();return t(i,s)}L(){const t=navigator.serviceWorker.controller;if(t&&e(t.scriptURL,this.i))return t}async B(){try{const t=await navigator.serviceWorker.register(this.i,this.h);return this.U=performance.now(),t}catch(t){throw t}}P(s){t(s,{type:"WINDOW_READY",meta:"workbox-window"})}p(){const t=this.S.installing;this.o>0||!e(t.scriptURL,this.i)||performance.now()>this.U+a?(this.k=t,this.S.removeEventListener("updatefound",this.p)):(this.R=t,this.l.resolve(t)),++this.o,t.addEventListener("statechange",this.v)}v(t){const s=t.target,{state:i}=s,e=s===this.k,a=e?"external":"",o={sw:s,originalEvent:t};!e&&this.C&&(o.isUpdate=!0),this.dispatchEvent(new n(a+i,o)),"installed"===i?this.D=setTimeout(()=>{"installed"===i&&this.S.waiting===s&&this.dispatchEvent(new n(a+"waiting",o))},h):"activating"===i&&(clearTimeout(this.D),e||this.g.resolve(s))}_(t){const s=this.R;s===navigator.serviceWorker.controller&&(this.dispatchEvent(new n("controlling",{sw:s,originalEvent:t})),this.u.resolve(s))}m(t){const{data:s}=t;this.dispatchEvent(new n("message",{data:s,originalEvent:t}))}}export{o as Workbox,t as messageSW}; |  | ||||||
| //# sourceMappingURL=workbox-window.prod.mjs.map |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| !function(n,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((n=n||self).workbox={})}(this,function(n){"use strict";try{self["workbox:window:4.3.1"]&&_()}catch(n){}var t=function(n,t){return new Promise(function(i){var e=new MessageChannel;e.port1.onmessage=function(n){return i(n.data)},n.postMessage(t,[e.port2])})};function i(n,t){for(var i=0;i<t.length;i++){var e=t[i];e.enumerable=e.enumerable||!1,e.configurable=!0,"value"in e&&(e.writable=!0),Object.defineProperty(n,e.key,e)}}function e(n){if(void 0===n)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return n}try{self["workbox:core:4.3.1"]&&_()}catch(n){}var r=function(){var n=this;this.promise=new Promise(function(t,i){n.resolve=t,n.reject=i})},o=function(n,t){return new URL(n,location).href===new URL(t,location).href},u=function(n,t){Object.assign(this,t,{type:n})};function s(n){return function(){for(var t=[],i=0;i<arguments.length;i++)t[i]=arguments[i];try{return Promise.resolve(n.apply(this,t))}catch(n){return Promise.reject(n)}}}function a(n,t,i){return i?t?t(n):n:(n&&n.then||(n=Promise.resolve(n)),t?n.then(t):n)}function c(){}var f=function(n){var f,h;function v(t,i){var o;return void 0===i&&(i={}),(o=n.call(this)||this).t=t,o.i=i,o.o=0,o.u=new r,o.s=new r,o.h=new r,o.v=o.v.bind(e(e(o))),o.l=o.l.bind(e(e(o))),o.g=o.g.bind(e(e(o))),o.m=o.m.bind(e(e(o))),o}h=n,(f=v).prototype=Object.create(h.prototype),f.prototype.constructor=f,f.__proto__=h;var l,w,d,g=v.prototype;return g.register=s(function(n){var t,i,e=this,r=(void 0===n?{}:n).immediate,s=void 0!==r&&r;return t=function(){return e.p=Boolean(navigator.serviceWorker.controller),e.P=e.j(),a(e.O(),function(n){e.R=n,e.P&&(e._=e.P,e.s.resolve(e.P),e.h.resolve(e.P),e.k(e.P),e.P.addEventListener("statechange",e.l,{once:!0}));var t=e.R.waiting;return t&&o(t.scriptURL,e.t)&&(e._=t,Promise.resolve().then(function(){e.dispatchEvent(new u("waiting",{sw:t,wasWaitingBeforeRegister:!0}))})),e._&&e.u.resolve(e._),e.R.addEventListener("updatefound",e.g),navigator.serviceWorker.addEventListener("controllerchange",e.m,{once:!0}),"BroadcastChannel"in self&&(e.B=new BroadcastChannel("workbox"),e.B.addEventListener("message",e.v)),navigator.serviceWorker.addEventListener("message",e.v),e.R})},(i=function(){if(!s&&"complete"!==document.readyState)return function(n,t){if(!t)return n&&n.then?n.then(c):Promise.resolve()}(new Promise(function(n){return addEventListener("load",n)}))}())&&i.then?i.then(t):t(i)}),g.getSW=s(function(){return this._||this.u.promise}),g.messageSW=s(function(n){return a(this.getSW(),function(i){return t(i,n)})}),g.j=function(){var n=navigator.serviceWorker.controller;if(n&&o(n.scriptURL,this.t))return n},g.O=s(function(){var n=this;return function(n,t){try{var i=n()}catch(n){return t(n)}return i&&i.then?i.then(void 0,t):i}(function(){return a(navigator.serviceWorker.register(n.t,n.i),function(t){return n.C=performance.now(),t})},function(n){throw n})}),g.k=function(n){t(n,{type:"WINDOW_READY",meta:"workbox-window"})},g.g=function(){var n=this.R.installing;this.o>0||!o(n.scriptURL,this.t)||performance.now()>this.C+6e4?(this.L=n,this.R.removeEventListener("updatefound",this.g)):(this._=n,this.u.resolve(n)),++this.o,n.addEventListener("statechange",this.l)},g.l=function(n){var t=this,i=n.target,e=i.state,r=i===this.L,o=r?"external":"",s={sw:i,originalEvent:n};!r&&this.p&&(s.isUpdate=!0),this.dispatchEvent(new u(o+e,s)),"installed"===e?this.W=setTimeout(function(){"installed"===e&&t.R.waiting===i&&t.dispatchEvent(new u(o+"waiting",s))},200):"activating"===e&&(clearTimeout(this.W),r||this.s.resolve(i))},g.m=function(n){var t=this._;t===navigator.serviceWorker.controller&&(this.dispatchEvent(new u("controlling",{sw:t,originalEvent:n})),this.h.resolve(t))},g.v=function(n){var t=n.data;this.dispatchEvent(new u("message",{data:t,originalEvent:n}))},l=v,(w=[{key:"active",get:function(){return this.s.promise}},{key:"controlling",get:function(){return this.h.promise}}])&&i(l.prototype,w),d&&i(l,d),v}(function(){function n(){this.D={}}var t=n.prototype;return t.addEventListener=function(n,t){this.M(n).add(t)},t.removeEventListener=function(n,t){this.M(n).delete(t)},t.dispatchEvent=function(n){n.target=this,this.M(n.type).forEach(function(t){return t(n)})},t.M=function(n){return this.D[n]=this.D[n]||new Set},n}());n.Workbox=f,n.messageSW=t,Object.defineProperty(n,"__esModule",{value:!0})}); |  | ||||||
| //# sourceMappingURL=workbox-window.prod.umd.js.map |  | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| const fs = require("fs"); |  | ||||||
| const { exec, execSync } = require("child_process"); |  | ||||||
|  |  | ||||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; |  | ||||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; |  | ||||||
| const pkg = require(excalidrawPackage); |  | ||||||
|  |  | ||||||
| const getShortCommitHash = () => { |  | ||||||
|   return execSync("git rev-parse --short HEAD").toString().trim(); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const publish = () => { |  | ||||||
|   try { |  | ||||||
|     execSync(`yarn  --frozen-lockfile`); |  | ||||||
|     execSync(`yarn --frozen-lockfile`, { cwd: excalidrawDir }); |  | ||||||
|     execSync(`yarn run build:umd`, { cwd: excalidrawDir }); |  | ||||||
|     execSync(`yarn --cwd ${excalidrawDir} publish`); |  | ||||||
|   } catch (e) { |  | ||||||
|     console.error(e); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // get files changed between prev and head commit |  | ||||||
| exec(`git diff --name-only HEAD^ HEAD`, async (error, stdout, stderr) => { |  | ||||||
|   if (error || stderr) { |  | ||||||
|     console.error(error); |  | ||||||
|     process.exit(1); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const changedFiles = stdout.trim().split("\n"); |  | ||||||
|   const filesToIgnoreRegex = /src\/excalidraw-app|packages\/utils/; |  | ||||||
|  |  | ||||||
|   const excalidrawPackageFiles = changedFiles.filter((file) => { |  | ||||||
|     return ( |  | ||||||
|       (file.indexOf("src") >= 0 || file.indexOf("package.json")) >= 0 && |  | ||||||
|       !filesToIgnoreRegex.test(file) |  | ||||||
|     ); |  | ||||||
|   }); |  | ||||||
|   if (!excalidrawPackageFiles.length) { |  | ||||||
|     process.exit(0); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // update package.json |  | ||||||
|   pkg.version = `${pkg.version}-${getShortCommitHash()}`; |  | ||||||
|   pkg.name = "@excalidraw/excalidraw-next"; |  | ||||||
|   fs.writeFileSync(excalidrawPackage, JSON.stringify(pkg, null, 2), "utf8"); |  | ||||||
|  |  | ||||||
|   // update readme |  | ||||||
|   const data = fs.readFileSync(`${excalidrawDir}/README_NEXT.md`, "utf8"); |  | ||||||
|   fs.writeFileSync(`${excalidrawDir}/README.md`, data, "utf8"); |  | ||||||
|   publish(); |  | ||||||
| }); |  | ||||||
| @@ -5,7 +5,7 @@ | |||||||
|  |  | ||||||
| // In order to run: | // In order to run: | ||||||
| //   npm install canvas # please do not check it in | //   npm install canvas # please do not check it in | ||||||
| //   yarn build-node | //   npm run build-node | ||||||
| //   node build/static/js/build-node.js | //   node build/static/js/build-node.js | ||||||
| //   open test.png | //   open test.png | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,60 +2,36 @@ | |||||||
|  |  | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const path = require("path"); | const path = require("path"); | ||||||
| const versionFile = path.join("build", "version.json"); | const asar = require("asar"); | ||||||
| const indexFile = path.join("build", "index.html"); |  | ||||||
|  |  | ||||||
| const versionDate = (date) => date.toISOString().replace(".000", ""); | const zero = (digit) => `0${digit}`.slice(-2); | ||||||
|  |  | ||||||
| const commitHash = () => { | const versionDate = (date) => { | ||||||
|   try { |   const date_ = `${date.getFullYear()}-${zero(date.getMonth() + 1)}-${zero( | ||||||
|     return require("child_process") |     date.getDate(), | ||||||
|       .execSync("git rev-parse --short HEAD") |   )}`; | ||||||
|       .toString() |   const time = `${zero(date.getHours())}-${zero(date.getMinutes())}-${zero( | ||||||
|       .trim(); |     date.getSeconds(), | ||||||
|   } catch { |   )}`; | ||||||
|     return "none"; |   return `${date_}-${time}`; | ||||||
|   } |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const commitDate = (hash) => { | const now = new Date(); | ||||||
|   try { |  | ||||||
|     const unix = require("child_process") |  | ||||||
|       .execSync(`git show -s --format=%ct ${hash}`) |  | ||||||
|       .toString() |  | ||||||
|       .trim(); |  | ||||||
|     const date = new Date(parseInt(unix) * 1000); |  | ||||||
|     return versionDate(date); |  | ||||||
|   } catch { |  | ||||||
|     return versionDate(new Date()); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const getFullVersion = () => { |  | ||||||
|   const hash = commitHash(); |  | ||||||
|   return `${commitDate(hash)}-${hash}`; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const data = JSON.stringify( | const data = JSON.stringify( | ||||||
|   { |   { | ||||||
|     version: getFullVersion(), |     asar: "excalidraw.asar", | ||||||
|  |     version: versionDate(now), | ||||||
|   }, |   }, | ||||||
|   undefined, |   undefined, | ||||||
|   2, |   2, | ||||||
| ); | ); | ||||||
|  |  | ||||||
| fs.writeFileSync(versionFile, data); | fs.writeFileSync(path.join("build", "version.json"), data); | ||||||
|  |  | ||||||
| // https://stackoverflow.com/a/14181136/8418 | (async () => { | ||||||
| fs.readFile(indexFile, "utf8", (error, data) => { |   const src = "build/"; | ||||||
|   if (error) { |   const dest = path.join("build", `excalidraw.asar`); | ||||||
|     return console.error(error); |  | ||||||
|   } |  | ||||||
|   const result = data.replace(/{version}/g, getFullVersion()); |  | ||||||
|  |  | ||||||
|   fs.writeFile(indexFile, result, "utf8", (error) => { |   await asar.createPackage(src, dest); | ||||||
|     if (error) { | })(); | ||||||
|       return console.error(error); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								scripts/changelog-check.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | |||||||
|  | const { exec } = require("child_process"); | ||||||
|  |  | ||||||
|  | const changeLogCheck = () => { | ||||||
|  |   exec( | ||||||
|  |     "git diff origin/master --cached --name-only", | ||||||
|  |     (error, stdout, stderr) => { | ||||||
|  |       if (error || stderr) { | ||||||
|  |         process.exit(1); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (!stdout || stdout.includes("packages/excalidraw/CHANGELOG.MD")) { | ||||||
|  |         process.exit(0); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const onlyNonSrcFilesUpdated = stdout.indexOf("src") < 0; | ||||||
|  |       if (onlyNonSrcFilesUpdated) { | ||||||
|  |         process.exit(0); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const changedFiles = stdout.trim().split("\n"); | ||||||
|  |       const filesToIgnoreRegex = /src\/excalidraw-app|packages\/utils/; | ||||||
|  |  | ||||||
|  |       const excalidrawPackageFiles = changedFiles.filter((file) => { | ||||||
|  |         return file.indexOf("src") >= 0 && !filesToIgnoreRegex.test(file); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       if (excalidrawPackageFiles.length) { | ||||||
|  |         process.exit(1); | ||||||
|  |       } | ||||||
|  |       process.exit(0); | ||||||
|  |     }, | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  | changeLogCheck(); | ||||||
| @@ -1,173 +0,0 @@ | |||||||
| const fs = require("fs"); |  | ||||||
|  |  | ||||||
| const THRESSHOLD = 85; |  | ||||||
|  |  | ||||||
| const crowdinMap = { |  | ||||||
|   "ar-SA": "en-ar", |  | ||||||
|   "bg-BG": "en-bg", |  | ||||||
|   "ca-ES": "en-ca", |  | ||||||
|   "de-DE": "en-de", |  | ||||||
|   "el-GR": "en-el", |  | ||||||
|   "es-ES": "en-es", |  | ||||||
|   "fa-IR": "en-fa", |  | ||||||
|   "fi-FI": "en-fi", |  | ||||||
|   "fr-FR": "en-fr", |  | ||||||
|   "he-IL": "en-he", |  | ||||||
|   "hi-IN": "en-hi", |  | ||||||
|   "hu-HU": "en-hu", |  | ||||||
|   "id-ID": "en-id", |  | ||||||
|   "it-IT": "en-it", |  | ||||||
|   "ja-JP": "en-ja", |  | ||||||
|   "kab-KAB": "en-kab", |  | ||||||
|   "ko-KR": "en-ko", |  | ||||||
|   "my-MM": "en-my", |  | ||||||
|   "nb-NO": "en-nb", |  | ||||||
|   "nl-NL": "en-nl", |  | ||||||
|   "nn-NO": "en-nnno", |  | ||||||
|   "oc-FR": "en-oc", |  | ||||||
|   "pa-IN": "en-pain", |  | ||||||
|   "pl-PL": "en-pl", |  | ||||||
|   "pt-BR": "en-ptbr", |  | ||||||
|   "pt-PT": "en-pt", |  | ||||||
|   "ro-RO": "en-ro", |  | ||||||
|   "ru-RU": "en-ru", |  | ||||||
|   "sk-SK": "en-sk", |  | ||||||
|   "sv-SE": "en-sv", |  | ||||||
|   "tr-TR": "en-tr", |  | ||||||
|   "uk-UA": "en-uk", |  | ||||||
|   "zh-CN": "en-zhcn", |  | ||||||
|   "zh-TW": "en-zhtw", |  | ||||||
|   "lv-LV": "en-lv", |  | ||||||
|   "cs-CZ": "en-cs", |  | ||||||
|   "kk-KZ": "en-kk", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const flags = { |  | ||||||
|   "ar-SA": "🇸🇦", |  | ||||||
|   "bg-BG": "🇧🇬", |  | ||||||
|   "ca-ES": "🏳", |  | ||||||
|   "de-DE": "🇩🇪", |  | ||||||
|   "el-GR": "🇬🇷", |  | ||||||
|   "es-ES": "🇪🇸", |  | ||||||
|   "fa-IR": "🇮🇷", |  | ||||||
|   "fi-FI": "🇫🇮", |  | ||||||
|   "fr-FR": "🇫🇷", |  | ||||||
|   "he-IL": "🇮🇱", |  | ||||||
|   "hi-IN": "🇮🇳", |  | ||||||
|   "hu-HU": "🇭🇺", |  | ||||||
|   "id-ID": "🇮🇩", |  | ||||||
|   "it-IT": "🇮🇹", |  | ||||||
|   "ja-JP": "🇯🇵", |  | ||||||
|   "kab-KAB": "🏳", |  | ||||||
|   "ko-KR": "🇰🇷", |  | ||||||
|   "my-MM": "🇲🇲", |  | ||||||
|   "nb-NO": "🇳🇴", |  | ||||||
|   "nl-NL": "🇳🇱", |  | ||||||
|   "nn-NO": "🇳🇴", |  | ||||||
|   "oc-FR": "🏳", |  | ||||||
|   "pa-IN": "🇮🇳", |  | ||||||
|   "pl-PL": "🇵🇱", |  | ||||||
|   "pt-BR": "🇧🇷", |  | ||||||
|   "pt-PT": "🇵🇹", |  | ||||||
|   "ro-RO": "🇷🇴", |  | ||||||
|   "ru-RU": "🇷🇺", |  | ||||||
|   "sk-SK": "🇸🇰", |  | ||||||
|   "sv-SE": "🇸🇪", |  | ||||||
|   "tr-TR": "🇹🇷", |  | ||||||
|   "uk-UA": "🇺🇦", |  | ||||||
|   "zh-CN": "🇨🇳", |  | ||||||
|   "zh-TW": "🇹🇼", |  | ||||||
|   "lv-LV": "🇱🇻", |  | ||||||
|   "cs-CZ": "🇨🇿", |  | ||||||
|   "kk-KZ": "🇰🇿", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const languages = { |  | ||||||
|   "ar-SA": "العربية", |  | ||||||
|   "bg-BG": "Български", |  | ||||||
|   "ca-ES": "Català", |  | ||||||
|   "de-DE": "Deutsch", |  | ||||||
|   "el-GR": "Ελληνικά", |  | ||||||
|   "es-ES": "Español", |  | ||||||
|   "fa-IR": "فارسی", |  | ||||||
|   "fi-FI": "Suomi", |  | ||||||
|   "fr-FR": "Français", |  | ||||||
|   "he-IL": "עברית", |  | ||||||
|   "hi-IN": "हिन्दी", |  | ||||||
|   "hu-HU": "Magyar", |  | ||||||
|   "id-ID": "Bahasa Indonesia", |  | ||||||
|   "it-IT": "Italiano", |  | ||||||
|   "ja-JP": "日本語", |  | ||||||
|   "kab-KAB": "Taqbaylit", |  | ||||||
|   "ko-KR": "한국어", |  | ||||||
|   "my-MM": "Burmese", |  | ||||||
|   "nb-NO": "Norsk bokmål", |  | ||||||
|   "nl-NL": "Nederlands", |  | ||||||
|   "nn-NO": "Norsk nynorsk", |  | ||||||
|   "oc-FR": "Occitan", |  | ||||||
|   "pa-IN": "ਪੰਜਾਬੀ", |  | ||||||
|   "pl-PL": "Polski", |  | ||||||
|   "pt-BR": "Português Brasileiro", |  | ||||||
|   "pt-PT": "Português", |  | ||||||
|   "ro-RO": "Română", |  | ||||||
|   "ru-RU": "Русский", |  | ||||||
|   "sk-SK": "Slovenčina", |  | ||||||
|   "sv-SE": "Svenska", |  | ||||||
|   "tr-TR": "Türkçe", |  | ||||||
|   "uk-UA": "Українська", |  | ||||||
|   "zh-CN": "简体中文", |  | ||||||
|   "zh-TW": "繁體中文", |  | ||||||
|   "lv-LV": "Latviešu", |  | ||||||
|   "cs-CZ": "Česky", |  | ||||||
|   "kk-KZ": "Қазақ тілі", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const percentages = fs.readFileSync( |  | ||||||
|   `${__dirname}/../src/locales/percentages.json`, |  | ||||||
| ); |  | ||||||
| const rowData = JSON.parse(percentages); |  | ||||||
|  |  | ||||||
| const coverages = Object.entries(rowData) |  | ||||||
|   .sort(([, a], [, b]) => b - a) |  | ||||||
|   .reduce((r, [k, v]) => ({ ...r, [k]: v }), {}); |  | ||||||
|  |  | ||||||
| const boldIf = (text, condition) => (condition ? `**${text}**` : text); |  | ||||||
|  |  | ||||||
| const printHeader = () => { |  | ||||||
|   let result = "| | Flag | Locale | % |\n"; |  | ||||||
|   result += "| :--: | :--: | -- | :--: |"; |  | ||||||
|   return result; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const printRow = (id, locale, coverage) => { |  | ||||||
|   const isOver = coverage >= THRESSHOLD; |  | ||||||
|   let result = `| ${isOver ? id : "..."} | `; |  | ||||||
|   result += `${locale in flags ? flags[locale] : ""} | `; |  | ||||||
|   const language = locale in languages ? languages[locale] : locale; |  | ||||||
|   if (locale in crowdinMap && crowdinMap[locale]) { |  | ||||||
|     result += `[${boldIf( |  | ||||||
|       language, |  | ||||||
|       isOver, |  | ||||||
|     )}](https://crowdin.com/translate/excalidraw/10/${crowdinMap[locale]}) | `; |  | ||||||
|   } else { |  | ||||||
|     result += `${boldIf(language, isOver)} | `; |  | ||||||
|   } |  | ||||||
|   result += `${coverage === 100 ? "💯" : boldIf(coverage, isOver)} |`; |  | ||||||
|   return result; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| console.info( |  | ||||||
|   `Each language must be at least **${THRESSHOLD}%** translated in order to appear on Excalidraw. Join us on [Crowdin](https://crowdin.com/project/excalidraw) and help us translate your own language. **Can't find yours yet?** Open an [issue](https://github.com/excalidraw/excalidraw/issues/new) and we'll add it to the list.`, |  | ||||||
| ); |  | ||||||
| console.info("\n\r"); |  | ||||||
| console.info(printHeader()); |  | ||||||
| let index = 1; |  | ||||||
| for (const coverage in coverages) { |  | ||||||
|   if (coverage === "en") { |  | ||||||
|     continue; |  | ||||||
|   } |  | ||||||
|   console.info(printRow(index, coverage, coverages[coverage])); |  | ||||||
|   index++; |  | ||||||
| } |  | ||||||
| console.info("\n\r"); |  | ||||||
| console.info("\\* Languages in **bold** are going to appear on production."); |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| const fs = require("fs"); |  | ||||||
| const util = require("util"); |  | ||||||
| const exec = util.promisify(require("child_process").exec); |  | ||||||
| const updateReadme = require("./updateReadme"); |  | ||||||
| const updateChangelog = require("./updateChangelog"); |  | ||||||
|  |  | ||||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; |  | ||||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; |  | ||||||
|  |  | ||||||
| const updatePackageVersion = (nextVersion) => { |  | ||||||
|   const pkg = require(excalidrawPackage); |  | ||||||
|   pkg.version = nextVersion; |  | ||||||
|   const content = `${JSON.stringify(pkg, null, 2)}\n`; |  | ||||||
|   fs.writeFileSync(excalidrawPackage, content, "utf-8"); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const release = async (nextVersion) => { |  | ||||||
|   try { |  | ||||||
|     updateReadme(); |  | ||||||
|     await updateChangelog(nextVersion); |  | ||||||
|     updatePackageVersion(nextVersion); |  | ||||||
|     await exec(`git add -u`); |  | ||||||
|     await exec( |  | ||||||
|       `git commit -m "docs: release @excalidraw/excalidraw@${nextVersion}  🎉"`, |  | ||||||
|     ); |  | ||||||
|     /* eslint-disable no-console */ |  | ||||||
|     console.log("Done!"); |  | ||||||
|   } catch (e) { |  | ||||||
|     console.error(e); |  | ||||||
|     process.exit(1); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const nextVersion = process.argv.slice(2)[0]; |  | ||||||
| if (!nextVersion) { |  | ||||||
|   console.error("Pass the next version to release!"); |  | ||||||
|   process.exit(1); |  | ||||||
| } |  | ||||||
| release(nextVersion); |  | ||||||
| @@ -1,97 +0,0 @@ | |||||||
| const fs = require("fs"); |  | ||||||
| const util = require("util"); |  | ||||||
| const exec = util.promisify(require("child_process").exec); |  | ||||||
|  |  | ||||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; |  | ||||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; |  | ||||||
| const pkg = require(excalidrawPackage); |  | ||||||
| const lastVersion = pkg.version; |  | ||||||
| const existingChangeLog = fs.readFileSync( |  | ||||||
|   `${excalidrawDir}/CHANGELOG.md`, |  | ||||||
|   "utf8", |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| const supportedTypes = ["feat", "fix", "style", "refactor", "perf", "build"]; |  | ||||||
| const headerForType = { |  | ||||||
|   feat: "Features", |  | ||||||
|   fix: "Fixes", |  | ||||||
|   style: "Styles", |  | ||||||
|   refactor: " Refactor", |  | ||||||
|   perf: "Performance", |  | ||||||
|   build: "Build", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const getCommitHashForLastVersion = async () => { |  | ||||||
|   try { |  | ||||||
|     const commitMessage = `"release @excalidraw/excalidraw@${lastVersion}"`; |  | ||||||
|     const { stdout } = await exec( |  | ||||||
|       `git log --format=format:"%H" --grep=${commitMessage}`, |  | ||||||
|     ); |  | ||||||
|     return stdout; |  | ||||||
|   } catch (e) { |  | ||||||
|     console.error(e); |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const getLibraryCommitsSinceLastRelease = async () => { |  | ||||||
|   const commitHash = await getCommitHashForLastVersion(); |  | ||||||
|   const { stdout } = await exec( |  | ||||||
|     `git log --pretty=format:%s ${commitHash}...master`, |  | ||||||
|   ); |  | ||||||
|   const commitsSinceLastRelease = stdout.split("\n"); |  | ||||||
|   const commitList = {}; |  | ||||||
|   supportedTypes.forEach((type) => { |  | ||||||
|     commitList[type] = []; |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   commitsSinceLastRelease.forEach((commit) => { |  | ||||||
|     const indexOfColon = commit.indexOf(":"); |  | ||||||
|     const type = commit.slice(0, indexOfColon); |  | ||||||
|     if (!supportedTypes.includes(type)) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     const messageWithoutType = commit.slice(indexOfColon + 1).trim(); |  | ||||||
|     const messageWithCapitalizeFirst = |  | ||||||
|       messageWithoutType.charAt(0).toUpperCase() + messageWithoutType.slice(1); |  | ||||||
|     const prNumber = commit.match(/\(#([0-9]*)\)/)[1]; |  | ||||||
|  |  | ||||||
|     // return if the changelog already contains the pr number which would happen for package updates |  | ||||||
|     if (existingChangeLog.includes(prNumber)) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     const prMarkdown = `[#${prNumber}](https://github.com/excalidraw/excalidraw/pull/${prNumber})`; |  | ||||||
|     const messageWithPRLink = messageWithCapitalizeFirst.replace( |  | ||||||
|       /\(#[0-9]*\)/, |  | ||||||
|       prMarkdown, |  | ||||||
|     ); |  | ||||||
|     commitList[type].push(messageWithPRLink); |  | ||||||
|   }); |  | ||||||
|   return commitList; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const updateChangelog = async (nextVersion) => { |  | ||||||
|   const commitList = await getLibraryCommitsSinceLastRelease(); |  | ||||||
|   let changelogForLibrary = |  | ||||||
|     "## Excalidraw Library\n\n**_This section lists the updates made to the excalidraw library and will not affect the integration._**\n\n"; |  | ||||||
|   supportedTypes.forEach((type) => { |  | ||||||
|     if (commitList[type].length) { |  | ||||||
|       changelogForLibrary += `### ${headerForType[type]}\n\n`; |  | ||||||
|       const commits = commitList[type]; |  | ||||||
|       commits.forEach((commit) => { |  | ||||||
|         changelogForLibrary += `- ${commit}\n\n`; |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|   changelogForLibrary += "---\n"; |  | ||||||
|   const lastVersionIndex = existingChangeLog.indexOf(`## ${lastVersion}`); |  | ||||||
|   let updatedContent = |  | ||||||
|     existingChangeLog.slice(0, lastVersionIndex) + |  | ||||||
|     changelogForLibrary + |  | ||||||
|     existingChangeLog.slice(lastVersionIndex); |  | ||||||
|   const currentDate = new Date().toISOString().slice(0, 10); |  | ||||||
|   const newVersion = `## ${nextVersion} (${currentDate})`; |  | ||||||
|   updatedContent = updatedContent.replace(`## Unreleased`, newVersion); |  | ||||||
|   fs.writeFileSync(`${excalidrawDir}/CHANGELOG.md`, updatedContent, "utf8"); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = updateChangelog; |  | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| const fs = require("fs"); |  | ||||||
|  |  | ||||||
| const updateReadme = () => { |  | ||||||
|   const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; |  | ||||||
|   let data = fs.readFileSync(`${excalidrawDir}/README_NEXT.md`, "utf8"); |  | ||||||
|  |  | ||||||
|   // remove note for unstable release |  | ||||||
|   data = data.replace( |  | ||||||
|     /<!-- unstable-readme-start-->[\s\S]*?<!-- unstable-readme-end-->/, |  | ||||||
|     "", |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   // replace "excalidraw-next" with "excalidraw" |  | ||||||
|   data = data.replace(/excalidraw-next/g, "excalidraw"); |  | ||||||
|   data = data.trim(); |  | ||||||
|  |  | ||||||
|   const demoIndex = data.indexOf("### Demo"); |  | ||||||
|   const excalidrawNextNote = |  | ||||||
|     "#### Note\n\n**If you don't want to wait for the next stable release and try out the unreleased changes you can use [@excalidraw/excalidraw-next](https://www.npmjs.com/package/@excalidraw/excalidraw-next).**\n\n"; |  | ||||||
|   // Add excalidraw next note to try out for unreleased changes |  | ||||||
|   data = data.slice(0, demoIndex) + excalidrawNextNote + data.slice(demoIndex); |  | ||||||
|  |  | ||||||
|   // update readme |  | ||||||
|   fs.writeFileSync(`${excalidrawDir}/README.md`, data, "utf8"); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = updateReadme; |  | ||||||
| @@ -2,22 +2,23 @@ import { register } from "./register"; | |||||||
| import { getSelectedElements } from "../scene"; | import { getSelectedElements } from "../scene"; | ||||||
| import { getNonDeletedElements } from "../element"; | import { getNonDeletedElements } from "../element"; | ||||||
| import { deepCopyElement } from "../element/newElement"; | import { deepCopyElement } from "../element/newElement"; | ||||||
|  | import { Library } from "../data/library"; | ||||||
|  | import { EVENT_LIBRARY, trackEvent } from "../analytics"; | ||||||
|  |  | ||||||
| export const actionAddToLibrary = register({ | export const actionAddToLibrary = register({ | ||||||
|   name: "addToLibrary", |   name: "addToLibrary", | ||||||
|   perform: (elements, appState, _, app) => { |   perform: (elements, appState) => { | ||||||
|     const selectedElements = getSelectedElements( |     const selectedElements = getSelectedElements( | ||||||
|       getNonDeletedElements(elements), |       getNonDeletedElements(elements), | ||||||
|       appState, |       appState, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     app.library.loadLibrary().then((items) => { |     Library.loadLibrary().then((items) => { | ||||||
|       app.library.saveLibrary([ |       Library.saveLibrary([...items, selectedElements.map(deepCopyElement)]); | ||||||
|         ...items, |  | ||||||
|         selectedElements.map(deepCopyElement), |  | ||||||
|       ]); |  | ||||||
|     }); |     }); | ||||||
|  |     trackEvent(EVENT_LIBRARY, "add"); | ||||||
|     return false; |     return false; | ||||||
|   }, |   }, | ||||||
|  |   contextMenuOrder: 6, | ||||||
|   contextItemLabel: "labels.addToLibrary", |   contextItemLabel: "labels.addToLibrary", | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,4 +1,7 @@ | |||||||
| import { alignElements, Alignment } from "../align"; | import React from "react"; | ||||||
|  | import { KEYS } from "../keys"; | ||||||
|  | import { t } from "../i18n"; | ||||||
|  | import { register } from "./register"; | ||||||
| import { | import { | ||||||
|   AlignBottomIcon, |   AlignBottomIcon, | ||||||
|   AlignLeftIcon, |   AlignLeftIcon, | ||||||
| @@ -7,15 +10,14 @@ import { | |||||||
|   CenterHorizontallyIcon, |   CenterHorizontallyIcon, | ||||||
|   CenterVerticallyIcon, |   CenterVerticallyIcon, | ||||||
| } from "../components/icons"; | } from "../components/icons"; | ||||||
| import { ToolButton } from "../components/ToolButton"; |  | ||||||
| import { getElementMap, getNonDeletedElements } from "../element"; |  | ||||||
| import { ExcalidrawElement } from "../element/types"; |  | ||||||
| import { t } from "../i18n"; |  | ||||||
| import { KEYS } from "../keys"; |  | ||||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||||
|  | import { getElementMap, getNonDeletedElements } from "../element"; | ||||||
|  | import { ToolButton } from "../components/ToolButton"; | ||||||
|  | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
|  | import { alignElements, Alignment } from "../align"; | ||||||
| import { getShortcutKey } from "../utils"; | import { getShortcutKey } from "../utils"; | ||||||
| import { register } from "./register"; | import { trackEvent, EVENT_ALIGN } from "../analytics"; | ||||||
|  |  | ||||||
| const enableActionGroup = ( | const enableActionGroup = ( | ||||||
|   elements: readonly ExcalidrawElement[], |   elements: readonly ExcalidrawElement[], | ||||||
| @@ -42,6 +44,7 @@ const alignSelectedElements = ( | |||||||
| export const actionAlignTop = register({ | export const actionAlignTop = register({ | ||||||
|   name: "alignTop", |   name: "alignTop", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "align", "top"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: alignSelectedElements(elements, appState, { |       elements: alignSelectedElements(elements, appState, { | ||||||
| @@ -57,7 +60,7 @@ export const actionAlignTop = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<AlignTopIcon theme={appState.theme} />} |       icon={<AlignTopIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.alignTop")} — ${getShortcutKey( |       title={`${t("labels.alignTop")} — ${getShortcutKey( | ||||||
|         "CtrlOrCmd+Shift+Up", |         "CtrlOrCmd+Shift+Up", | ||||||
| @@ -71,6 +74,7 @@ export const actionAlignTop = register({ | |||||||
| export const actionAlignBottom = register({ | export const actionAlignBottom = register({ | ||||||
|   name: "alignBottom", |   name: "alignBottom", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "align", "bottom"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: alignSelectedElements(elements, appState, { |       elements: alignSelectedElements(elements, appState, { | ||||||
| @@ -86,7 +90,7 @@ export const actionAlignBottom = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<AlignBottomIcon theme={appState.theme} />} |       icon={<AlignBottomIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.alignBottom")} — ${getShortcutKey( |       title={`${t("labels.alignBottom")} — ${getShortcutKey( | ||||||
|         "CtrlOrCmd+Shift+Down", |         "CtrlOrCmd+Shift+Down", | ||||||
| @@ -100,6 +104,7 @@ export const actionAlignBottom = register({ | |||||||
| export const actionAlignLeft = register({ | export const actionAlignLeft = register({ | ||||||
|   name: "alignLeft", |   name: "alignLeft", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "align", "left"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: alignSelectedElements(elements, appState, { |       elements: alignSelectedElements(elements, appState, { | ||||||
| @@ -115,7 +120,7 @@ export const actionAlignLeft = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<AlignLeftIcon theme={appState.theme} />} |       icon={<AlignLeftIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.alignLeft")} — ${getShortcutKey( |       title={`${t("labels.alignLeft")} — ${getShortcutKey( | ||||||
|         "CtrlOrCmd+Shift+Left", |         "CtrlOrCmd+Shift+Left", | ||||||
| @@ -129,6 +134,7 @@ export const actionAlignLeft = register({ | |||||||
| export const actionAlignRight = register({ | export const actionAlignRight = register({ | ||||||
|   name: "alignRight", |   name: "alignRight", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "align", "right"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: alignSelectedElements(elements, appState, { |       elements: alignSelectedElements(elements, appState, { | ||||||
| @@ -144,7 +150,7 @@ export const actionAlignRight = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<AlignRightIcon theme={appState.theme} />} |       icon={<AlignRightIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.alignRight")} — ${getShortcutKey( |       title={`${t("labels.alignRight")} — ${getShortcutKey( | ||||||
|         "CtrlOrCmd+Shift+Right", |         "CtrlOrCmd+Shift+Right", | ||||||
| @@ -158,6 +164,7 @@ export const actionAlignRight = register({ | |||||||
| export const actionAlignVerticallyCentered = register({ | export const actionAlignVerticallyCentered = register({ | ||||||
|   name: "alignVerticallyCentered", |   name: "alignVerticallyCentered", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "vertically", "center"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: alignSelectedElements(elements, appState, { |       elements: alignSelectedElements(elements, appState, { | ||||||
| @@ -171,7 +178,7 @@ export const actionAlignVerticallyCentered = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<CenterVerticallyIcon theme={appState.theme} />} |       icon={<CenterVerticallyIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={t("labels.centerVertically")} |       title={t("labels.centerVertically")} | ||||||
|       aria-label={t("labels.centerVertically")} |       aria-label={t("labels.centerVertically")} | ||||||
| @@ -183,6 +190,7 @@ export const actionAlignVerticallyCentered = register({ | |||||||
| export const actionAlignHorizontallyCentered = register({ | export const actionAlignHorizontallyCentered = register({ | ||||||
|   name: "alignHorizontallyCentered", |   name: "alignHorizontallyCentered", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "horizontally", "center"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: alignSelectedElements(elements, appState, { |       elements: alignSelectedElements(elements, appState, { | ||||||
| @@ -196,7 +204,7 @@ export const actionAlignHorizontallyCentered = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<CenterHorizontallyIcon theme={appState.theme} />} |       icon={<CenterHorizontallyIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={t("labels.centerHorizontally")} |       title={t("labels.centerHorizontally")} | ||||||
|       aria-label={t("labels.centerHorizontally")} |       aria-label={t("labels.centerHorizontally")} | ||||||
|   | |||||||
| @@ -1,29 +1,37 @@ | |||||||
|  | import React from "react"; | ||||||
| import { ColorPicker } from "../components/ColorPicker"; | import { ColorPicker } from "../components/ColorPicker"; | ||||||
| import { zoomIn, zoomOut } from "../components/icons"; |  | ||||||
| import { ToolButton } from "../components/ToolButton"; |  | ||||||
| import { DarkModeToggle } from "../components/DarkModeToggle"; |  | ||||||
| import { THEME, ZOOM_STEP } from "../constants"; |  | ||||||
| import { getCommonBounds, getNonDeletedElements } from "../element"; |  | ||||||
| import { ExcalidrawElement } from "../element/types"; |  | ||||||
| import { t } from "../i18n"; |  | ||||||
| import { CODES, KEYS } from "../keys"; |  | ||||||
| import { getNormalizedZoom, getSelectedElements } from "../scene"; |  | ||||||
| import { centerScrollOn } from "../scene/scroll"; |  | ||||||
| import { getNewZoom } from "../scene/zoom"; |  | ||||||
| import { AppState, NormalizedZoomValue } from "../types"; |  | ||||||
| import { getShortcutKey } from "../utils"; |  | ||||||
| import { register } from "./register"; |  | ||||||
| import { Tooltip } from "../components/Tooltip"; |  | ||||||
| import { newElementWith } from "../element/mutateElement"; |  | ||||||
| import { getDefaultAppState } from "../appState"; | import { getDefaultAppState } from "../appState"; | ||||||
| import ClearCanvas from "../components/ClearCanvas"; | import { trash, zoomIn, zoomOut, resetZoom } from "../components/icons"; | ||||||
|  | import { ToolButton } from "../components/ToolButton"; | ||||||
|  | import { t } from "../i18n"; | ||||||
|  | import { getNormalizedZoom } from "../scene"; | ||||||
|  | import { CODES, KEYS } from "../keys"; | ||||||
|  | import { getShortcutKey } from "../utils"; | ||||||
|  | import useIsMobile from "../is-mobile"; | ||||||
|  | import { register } from "./register"; | ||||||
|  | import { newElementWith } from "../element/mutateElement"; | ||||||
|  | import { AppState, NormalizedZoomValue } from "../types"; | ||||||
|  | import { getCommonBounds } from "../element"; | ||||||
|  | import { getNewZoom } from "../scene/zoom"; | ||||||
|  | import { centerScrollOn } from "../scene/scroll"; | ||||||
|  | import { EVENT_ACTION, EVENT_CHANGE, trackEvent } from "../analytics"; | ||||||
|  | import colors from "../colors"; | ||||||
|  |  | ||||||
| export const actionChangeViewBackgroundColor = register({ | export const actionChangeViewBackgroundColor = register({ | ||||||
|   name: "changeViewBackgroundColor", |   name: "changeViewBackgroundColor", | ||||||
|   perform: (_, appState, value) => { |   perform: (_, appState, value) => { | ||||||
|  |     if (value !== appState.viewBackgroundColor) { | ||||||
|  |       trackEvent( | ||||||
|  |         EVENT_CHANGE, | ||||||
|  |         "canvas color", | ||||||
|  |         colors.canvasBackground.includes(value) | ||||||
|  |           ? `${value} (picker ${colors.canvasBackground.indexOf(value)})` | ||||||
|  |           : value, | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|     return { |     return { | ||||||
|       appState: { ...appState, ...value }, |       appState: { ...appState, viewBackgroundColor: value }, | ||||||
|       commitToHistory: !!value.viewBackgroundColor, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData }) => { |   PanelComponent: ({ appState, updateData }) => { | ||||||
| @@ -33,12 +41,7 @@ export const actionChangeViewBackgroundColor = register({ | |||||||
|           label={t("labels.canvasBackground")} |           label={t("labels.canvasBackground")} | ||||||
|           type="canvasBackground" |           type="canvasBackground" | ||||||
|           color={appState.viewBackgroundColor} |           color={appState.viewBackgroundColor} | ||||||
|           onChange={(color) => updateData({ viewBackgroundColor: color })} |           onChange={(color) => updateData(color)} | ||||||
|           isActive={appState.openPopup === "canvasColorPicker"} |  | ||||||
|           setActive={(active) => |  | ||||||
|             updateData({ openPopup: active ? "canvasColorPicker" : null }) |  | ||||||
|           } |  | ||||||
|           data-testid="canvas-background-picker" |  | ||||||
|         /> |         /> | ||||||
|       </div> |       </div> | ||||||
|     ); |     ); | ||||||
| @@ -47,39 +50,52 @@ export const actionChangeViewBackgroundColor = register({ | |||||||
|  |  | ||||||
| export const actionClearCanvas = register({ | export const actionClearCanvas = register({ | ||||||
|   name: "clearCanvas", |   name: "clearCanvas", | ||||||
|   perform: (elements, appState, _, app) => { |   perform: (elements, appState: AppState) => { | ||||||
|     app.imageCache.clear(); |     trackEvent(EVENT_ACTION, "clear canvas"); | ||||||
|     return { |     return { | ||||||
|       elements: elements.map((element) => |       elements: elements.map((element) => | ||||||
|         newElementWith(element, { isDeleted: true }), |         newElementWith(element, { isDeleted: true }), | ||||||
|       ), |       ), | ||||||
|       appState: { |       appState: { | ||||||
|         ...getDefaultAppState(), |         ...getDefaultAppState(), | ||||||
|         files: {}, |         appearance: appState.appearance, | ||||||
|         theme: appState.theme, |  | ||||||
|         elementLocked: appState.elementLocked, |         elementLocked: appState.elementLocked, | ||||||
|         exportBackground: appState.exportBackground, |         exportBackground: appState.exportBackground, | ||||||
|         exportEmbedScene: appState.exportEmbedScene, |         exportEmbedScene: appState.exportEmbedScene, | ||||||
|         gridSize: appState.gridSize, |         gridSize: appState.gridSize, | ||||||
|  |         shouldAddWatermark: appState.shouldAddWatermark, | ||||||
|         showStats: appState.showStats, |         showStats: appState.showStats, | ||||||
|         pasteDialog: appState.pasteDialog, |  | ||||||
|       }, |       }, | ||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|  |   PanelComponent: ({ updateData }) => ( | ||||||
|   PanelComponent: ({ updateData }) => <ClearCanvas onConfirm={updateData} />, |     <ToolButton | ||||||
|  |       type="button" | ||||||
|  |       icon={trash} | ||||||
|  |       title={t("buttons.clearReset")} | ||||||
|  |       aria-label={t("buttons.clearReset")} | ||||||
|  |       showAriaLabel={useIsMobile()} | ||||||
|  |       onClick={() => { | ||||||
|  |         if (window.confirm(t("alerts.clearReset"))) { | ||||||
|  |           updateData(null); | ||||||
|  |         } | ||||||
|  |       }} | ||||||
|  |     /> | ||||||
|  |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | const ZOOM_STEP = 0.1; | ||||||
|  |  | ||||||
| export const actionZoomIn = register({ | export const actionZoomIn = register({ | ||||||
|   name: "zoomIn", |   name: "zoomIn", | ||||||
|   perform: (_elements, appState) => { |   perform: (_elements, appState) => { | ||||||
|     const zoom = getNewZoom( |     const zoom = getNewZoom( | ||||||
|       getNormalizedZoom(appState.zoom.value + ZOOM_STEP), |       getNormalizedZoom(appState.zoom.value + ZOOM_STEP), | ||||||
|       appState.zoom, |       appState.zoom, | ||||||
|       { left: appState.offsetLeft, top: appState.offsetTop }, |  | ||||||
|       { x: appState.width / 2, y: appState.height / 2 }, |       { x: appState.width / 2, y: appState.height / 2 }, | ||||||
|     ); |     ); | ||||||
|  |     trackEvent(EVENT_ACTION, "zoom", "in", zoom.value * 100); | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
| @@ -97,7 +113,6 @@ export const actionZoomIn = register({ | |||||||
|       onClick={() => { |       onClick={() => { | ||||||
|         updateData(null); |         updateData(null); | ||||||
|       }} |       }} | ||||||
|       size="small" |  | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
| @@ -111,10 +126,10 @@ export const actionZoomOut = register({ | |||||||
|     const zoom = getNewZoom( |     const zoom = getNewZoom( | ||||||
|       getNormalizedZoom(appState.zoom.value - ZOOM_STEP), |       getNormalizedZoom(appState.zoom.value - ZOOM_STEP), | ||||||
|       appState.zoom, |       appState.zoom, | ||||||
|       { left: appState.offsetLeft, top: appState.offsetTop }, |  | ||||||
|       { x: appState.width / 2, y: appState.height / 2 }, |       { x: appState.width / 2, y: appState.height / 2 }, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     trackEvent(EVENT_ACTION, "zoom", "out", zoom.value * 100); | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
| @@ -132,7 +147,6 @@ export const actionZoomOut = register({ | |||||||
|       onClick={() => { |       onClick={() => { | ||||||
|         updateData(null); |         updateData(null); | ||||||
|       }} |       }} | ||||||
|       size="small" |  | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
| @@ -143,37 +157,28 @@ export const actionZoomOut = register({ | |||||||
| export const actionResetZoom = register({ | export const actionResetZoom = register({ | ||||||
|   name: "resetZoom", |   name: "resetZoom", | ||||||
|   perform: (_elements, appState) => { |   perform: (_elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ACTION, "zoom", "reset", 100); | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|         zoom: getNewZoom( |         zoom: getNewZoom(1 as NormalizedZoomValue, appState.zoom, { | ||||||
|           1 as NormalizedZoomValue, |           x: appState.width / 2, | ||||||
|           appState.zoom, |           y: appState.height / 2, | ||||||
|           { left: appState.offsetLeft, top: appState.offsetTop }, |         }), | ||||||
|           { |  | ||||||
|             x: appState.width / 2, |  | ||||||
|             y: appState.height / 2, |  | ||||||
|           }, |  | ||||||
|         ), |  | ||||||
|       }, |       }, | ||||||
|       commitToHistory: false, |       commitToHistory: false, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ updateData, appState }) => ( |   PanelComponent: ({ updateData }) => ( | ||||||
|     <Tooltip label={t("buttons.resetZoom")}> |     <ToolButton | ||||||
|       <ToolButton |       type="button" | ||||||
|         type="button" |       icon={resetZoom} | ||||||
|         className="reset-zoom-button" |       title={t("buttons.resetZoom")} | ||||||
|         title={t("buttons.resetZoom")} |       aria-label={t("buttons.resetZoom")} | ||||||
|         aria-label={t("buttons.resetZoom")} |       onClick={() => { | ||||||
|         onClick={() => { |         updateData(null); | ||||||
|           updateData(null); |       }} | ||||||
|         }} |     /> | ||||||
|         size="small" |  | ||||||
|       > |  | ||||||
|         {(appState.zoom.value * 100).toFixed(0)}% |  | ||||||
|       </ToolButton> |  | ||||||
|     </Tooltip> |  | ||||||
|   ), |   ), | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     (event.code === CODES.ZERO || event.code === CODES.NUM_ZERO) && |     (event.code === CODES.ZERO || event.code === CODES.NUM_ZERO) && | ||||||
| @@ -199,89 +204,41 @@ const zoomValueToFitBoundsOnViewport = ( | |||||||
|   return clampedZoomValueToFitElements as NormalizedZoomValue; |   return clampedZoomValueToFitElements as NormalizedZoomValue; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const zoomToFitElements = ( |  | ||||||
|   elements: readonly ExcalidrawElement[], |  | ||||||
|   appState: Readonly<AppState>, |  | ||||||
|   zoomToSelection: boolean, |  | ||||||
| ) => { |  | ||||||
|   const nonDeletedElements = getNonDeletedElements(elements); |  | ||||||
|   const selectedElements = getSelectedElements(nonDeletedElements, appState); |  | ||||||
|  |  | ||||||
|   const commonBounds = |  | ||||||
|     zoomToSelection && selectedElements.length > 0 |  | ||||||
|       ? getCommonBounds(selectedElements) |  | ||||||
|       : getCommonBounds(nonDeletedElements); |  | ||||||
|  |  | ||||||
|   const zoomValue = zoomValueToFitBoundsOnViewport(commonBounds, { |  | ||||||
|     width: appState.width, |  | ||||||
|     height: appState.height, |  | ||||||
|   }); |  | ||||||
|   const newZoom = getNewZoom(zoomValue, appState.zoom, { |  | ||||||
|     left: appState.offsetLeft, |  | ||||||
|     top: appState.offsetTop, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   const [x1, y1, x2, y2] = commonBounds; |  | ||||||
|   const centerX = (x1 + x2) / 2; |  | ||||||
|   const centerY = (y1 + y2) / 2; |  | ||||||
|   return { |  | ||||||
|     appState: { |  | ||||||
|       ...appState, |  | ||||||
|       ...centerScrollOn({ |  | ||||||
|         scenePoint: { x: centerX, y: centerY }, |  | ||||||
|         viewportDimensions: { |  | ||||||
|           width: appState.width, |  | ||||||
|           height: appState.height, |  | ||||||
|         }, |  | ||||||
|         zoom: newZoom, |  | ||||||
|       }), |  | ||||||
|       zoom: newZoom, |  | ||||||
|     }, |  | ||||||
|     commitToHistory: false, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const actionZoomToSelected = register({ |  | ||||||
|   name: "zoomToSelection", |  | ||||||
|   perform: (elements, appState) => zoomToFitElements(elements, appState, true), |  | ||||||
|   keyTest: (event) => |  | ||||||
|     event.code === CODES.TWO && |  | ||||||
|     event.shiftKey && |  | ||||||
|     !event.altKey && |  | ||||||
|     !event[KEYS.CTRL_OR_CMD], |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export const actionZoomToFit = register({ | export const actionZoomToFit = register({ | ||||||
|   name: "zoomToFit", |   name: "zoomToFit", | ||||||
|   perform: (elements, appState) => zoomToFitElements(elements, appState, false), |   perform: (elements, appState) => { | ||||||
|  |     const nonDeletedElements = elements.filter((element) => !element.isDeleted); | ||||||
|  |     const commonBounds = getCommonBounds(nonDeletedElements); | ||||||
|  |  | ||||||
|  |     const zoomValue = zoomValueToFitBoundsOnViewport(commonBounds, { | ||||||
|  |       width: appState.width, | ||||||
|  |       height: appState.height, | ||||||
|  |     }); | ||||||
|  |     const newZoom = getNewZoom(zoomValue, appState.zoom); | ||||||
|  |  | ||||||
|  |     const [x1, y1, x2, y2] = commonBounds; | ||||||
|  |     const centerX = (x1 + x2) / 2; | ||||||
|  |     const centerY = (y1 + y2) / 2; | ||||||
|  |     trackEvent(EVENT_ACTION, "zoom", "fit", newZoom.value * 100); | ||||||
|  |     return { | ||||||
|  |       appState: { | ||||||
|  |         ...appState, | ||||||
|  |         ...centerScrollOn({ | ||||||
|  |           scenePoint: { x: centerX, y: centerY }, | ||||||
|  |           viewportDimensions: { | ||||||
|  |             width: appState.width, | ||||||
|  |             height: appState.height, | ||||||
|  |           }, | ||||||
|  |           zoom: newZoom, | ||||||
|  |         }), | ||||||
|  |         zoom: newZoom, | ||||||
|  |       }, | ||||||
|  |       commitToHistory: false, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event.code === CODES.ONE && |     event.code === CODES.ONE && | ||||||
|     event.shiftKey && |     event.shiftKey && | ||||||
|     !event.altKey && |     !event.altKey && | ||||||
|     !event[KEYS.CTRL_OR_CMD], |     !event[KEYS.CTRL_OR_CMD], | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionToggleTheme = register({ |  | ||||||
|   name: "toggleTheme", |  | ||||||
|   perform: (_, appState, value) => { |  | ||||||
|     return { |  | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         theme: |  | ||||||
|           value || (appState.theme === THEME.LIGHT ? THEME.DARK : THEME.LIGHT), |  | ||||||
|       }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   PanelComponent: ({ appState, updateData }) => ( |  | ||||||
|     <div style={{ marginInlineStart: "0.25rem" }}> |  | ||||||
|       <DarkModeToggle |  | ||||||
|         value={appState.theme} |  | ||||||
|         onChange={(theme) => { |  | ||||||
|           updateData(theme); |  | ||||||
|         }} |  | ||||||
|       /> |  | ||||||
|     </div> |  | ||||||
|   ), |  | ||||||
|   keyTest: (event) => event.altKey && event.shiftKey && event.code === CODES.D, |  | ||||||
| }); |  | ||||||
|   | |||||||
| @@ -1,122 +0,0 @@ | |||||||
| import { CODES, KEYS } from "../keys"; |  | ||||||
| import { register } from "./register"; |  | ||||||
| import { copyToClipboard } from "../clipboard"; |  | ||||||
| import { actionDeleteSelected } from "./actionDeleteSelected"; |  | ||||||
| import { getSelectedElements } from "../scene/selection"; |  | ||||||
| import { exportCanvas } from "../data/index"; |  | ||||||
| import { getNonDeletedElements } from "../element"; |  | ||||||
| import { t } from "../i18n"; |  | ||||||
|  |  | ||||||
| export const actionCopy = register({ |  | ||||||
|   name: "copy", |  | ||||||
|   perform: (elements, appState, _, app) => { |  | ||||||
|     copyToClipboard(getNonDeletedElements(elements), appState, app.files); |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   contextItemLabel: "labels.copy", |  | ||||||
|   // don't supply a shortcut since we handle this conditionally via onCopy event |  | ||||||
|   keyTest: undefined, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export const actionCut = register({ |  | ||||||
|   name: "cut", |  | ||||||
|   perform: (elements, appState, data, app) => { |  | ||||||
|     actionCopy.perform(elements, appState, data, app); |  | ||||||
|     return actionDeleteSelected.perform(elements, appState, data, app); |  | ||||||
|   }, |  | ||||||
|   contextItemLabel: "labels.cut", |  | ||||||
|   keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.code === CODES.X, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export const actionCopyAsSvg = register({ |  | ||||||
|   name: "copyAsSvg", |  | ||||||
|   perform: async (elements, appState, _data, app) => { |  | ||||||
|     if (!app.canvas) { |  | ||||||
|       return { |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     const selectedElements = getSelectedElements( |  | ||||||
|       getNonDeletedElements(elements), |  | ||||||
|       appState, |  | ||||||
|     ); |  | ||||||
|     try { |  | ||||||
|       await exportCanvas( |  | ||||||
|         "clipboard-svg", |  | ||||||
|         selectedElements.length |  | ||||||
|           ? selectedElements |  | ||||||
|           : getNonDeletedElements(elements), |  | ||||||
|         appState, |  | ||||||
|         app.files, |  | ||||||
|         appState, |  | ||||||
|       ); |  | ||||||
|       return { |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } catch (error) { |  | ||||||
|       console.error(error); |  | ||||||
|       return { |  | ||||||
|         appState: { |  | ||||||
|           ...appState, |  | ||||||
|           errorMessage: error.message, |  | ||||||
|         }, |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|   }, |  | ||||||
|   contextItemLabel: "labels.copyAsSvg", |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export const actionCopyAsPng = register({ |  | ||||||
|   name: "copyAsPng", |  | ||||||
|   perform: async (elements, appState, _data, app) => { |  | ||||||
|     if (!app.canvas) { |  | ||||||
|       return { |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|     const selectedElements = getSelectedElements( |  | ||||||
|       getNonDeletedElements(elements), |  | ||||||
|       appState, |  | ||||||
|     ); |  | ||||||
|     try { |  | ||||||
|       await exportCanvas( |  | ||||||
|         "clipboard", |  | ||||||
|         selectedElements.length |  | ||||||
|           ? selectedElements |  | ||||||
|           : getNonDeletedElements(elements), |  | ||||||
|         appState, |  | ||||||
|         app.files, |  | ||||||
|         appState, |  | ||||||
|       ); |  | ||||||
|       return { |  | ||||||
|         appState: { |  | ||||||
|           ...appState, |  | ||||||
|           toastMessage: t("toast.copyToClipboardAsPng", { |  | ||||||
|             exportSelection: selectedElements.length |  | ||||||
|               ? t("toast.selection") |  | ||||||
|               : t("toast.canvas"), |  | ||||||
|             exportColorScheme: appState.exportWithDarkMode |  | ||||||
|               ? t("buttons.darkMode") |  | ||||||
|               : t("buttons.lightMode"), |  | ||||||
|           }), |  | ||||||
|         }, |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } catch (error) { |  | ||||||
|       console.error(error); |  | ||||||
|       return { |  | ||||||
|         appState: { |  | ||||||
|           ...appState, |  | ||||||
|           errorMessage: error.message, |  | ||||||
|         }, |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|   }, |  | ||||||
|   contextItemLabel: "labels.copyAsPng", |  | ||||||
|   keyTest: (event) => event.code === CODES.C && event.altKey && event.shiftKey, |  | ||||||
| }); |  | ||||||
| @@ -1,6 +1,7 @@ | |||||||
| import { isSomeElementSelected } from "../scene"; | import { isSomeElementSelected } from "../scene"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
|  | import React from "react"; | ||||||
| import { trash } from "../components/icons"; | import { trash } from "../components/icons"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| @@ -135,6 +136,7 @@ export const actionDeleteSelected = register({ | |||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   contextItemLabel: "labels.delete", |   contextItemLabel: "labels.delete", | ||||||
|  |   contextMenuOrder: 3, | ||||||
|   keyTest: (event) => event.key === KEYS.BACKSPACE || event.key === KEYS.DELETE, |   keyTest: (event) => event.key === KEYS.BACKSPACE || event.key === KEYS.DELETE, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
|     <ToolButton |     <ToolButton | ||||||
|   | |||||||
| @@ -1,17 +1,19 @@ | |||||||
|  | import React from "react"; | ||||||
|  | import { CODES } from "../keys"; | ||||||
|  | import { t } from "../i18n"; | ||||||
|  | import { register } from "./register"; | ||||||
| import { | import { | ||||||
|   DistributeHorizontallyIcon, |   DistributeHorizontallyIcon, | ||||||
|   DistributeVerticallyIcon, |   DistributeVerticallyIcon, | ||||||
| } from "../components/icons"; | } from "../components/icons"; | ||||||
| import { ToolButton } from "../components/ToolButton"; |  | ||||||
| import { distributeElements, Distribution } from "../disitrubte"; |  | ||||||
| import { getElementMap, getNonDeletedElements } from "../element"; |  | ||||||
| import { ExcalidrawElement } from "../element/types"; |  | ||||||
| import { t } from "../i18n"; |  | ||||||
| import { CODES } from "../keys"; |  | ||||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||||
|  | import { getElementMap, getNonDeletedElements } from "../element"; | ||||||
|  | import { ToolButton } from "../components/ToolButton"; | ||||||
|  | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
|  | import { distributeElements, Distribution } from "../disitrubte"; | ||||||
| import { getShortcutKey } from "../utils"; | import { getShortcutKey } from "../utils"; | ||||||
| import { register } from "./register"; | import { EVENT_ALIGN, trackEvent } from "../analytics"; | ||||||
|  |  | ||||||
| const enableActionGroup = ( | const enableActionGroup = ( | ||||||
|   elements: readonly ExcalidrawElement[], |   elements: readonly ExcalidrawElement[], | ||||||
| @@ -38,6 +40,7 @@ const distributeSelectedElements = ( | |||||||
| export const distributeHorizontally = register({ | export const distributeHorizontally = register({ | ||||||
|   name: "distributeHorizontally", |   name: "distributeHorizontally", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "distribute", "horizontally"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: distributeSelectedElements(elements, appState, { |       elements: distributeSelectedElements(elements, appState, { | ||||||
| @@ -52,7 +55,7 @@ export const distributeHorizontally = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<DistributeHorizontallyIcon theme={appState.theme} />} |       icon={<DistributeHorizontallyIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.distributeHorizontally")} — ${getShortcutKey( |       title={`${t("labels.distributeHorizontally")} — ${getShortcutKey( | ||||||
|         "Alt+H", |         "Alt+H", | ||||||
| @@ -66,6 +69,7 @@ export const distributeHorizontally = register({ | |||||||
| export const distributeVertically = register({ | export const distributeVertically = register({ | ||||||
|   name: "distributeVertically", |   name: "distributeVertically", | ||||||
|   perform: (elements, appState) => { |   perform: (elements, appState) => { | ||||||
|  |     trackEvent(EVENT_ALIGN, "distribute", "vertically"); | ||||||
|     return { |     return { | ||||||
|       appState, |       appState, | ||||||
|       elements: distributeSelectedElements(elements, appState, { |       elements: distributeSelectedElements(elements, appState, { | ||||||
| @@ -80,7 +84,7 @@ export const distributeVertically = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<DistributeVerticallyIcon theme={appState.theme} />} |       icon={<DistributeVerticallyIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.distributeVertically")} — ${getShortcutKey("Alt+V")}`} |       title={`${t("labels.distributeVertically")} — ${getShortcutKey("Alt+V")}`} | ||||||
|       aria-label={t("labels.distributeVertically")} |       aria-label={t("labels.distributeVertically")} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import React from "react"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { ExcalidrawElement } from "../element/types"; | import { ExcalidrawElement } from "../element/types"; | ||||||
|   | |||||||
| @@ -1,92 +1,30 @@ | |||||||
| import { trackEvent } from "../analytics"; | import React from "react"; | ||||||
| import { load, questionCircle, saveAs } from "../components/icons"; | import { EVENT_CHANGE, EVENT_IO, trackEvent } from "../analytics"; | ||||||
|  | import { load, save, saveAs } from "../components/icons"; | ||||||
| import { ProjectName } from "../components/ProjectName"; | import { ProjectName } from "../components/ProjectName"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| import "../components/ToolIcon.scss"; |  | ||||||
| import { Tooltip } from "../components/Tooltip"; |  | ||||||
| import { DarkModeToggle } from "../components/DarkModeToggle"; |  | ||||||
| import { loadFromJSON, saveAsJSON } from "../data"; | import { loadFromJSON, saveAsJSON } from "../data"; | ||||||
| import { resaveAsImageWithScene } from "../data/resave"; |  | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { useIsMobile } from "../components/App"; | import useIsMobile from "../is-mobile"; | ||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
|  | import { muteFSAbortError } from "../utils"; | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { CheckboxItem } from "../components/CheckboxItem"; |  | ||||||
| import { getExportSize } from "../scene/export"; |  | ||||||
| import { DEFAULT_EXPORT_PADDING, EXPORT_SCALES, THEME } from "../constants"; |  | ||||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; |  | ||||||
| import { getNonDeletedElements } from "../element"; |  | ||||||
| import { ActiveFile } from "../components/ActiveFile"; |  | ||||||
| import { isImageFileHandle } from "../data/blob"; |  | ||||||
| import { nativeFileSystemSupported } from "../data/filesystem"; |  | ||||||
| import { Theme } from "../element/types"; |  | ||||||
|  |  | ||||||
| export const actionChangeProjectName = register({ | export const actionChangeProjectName = register({ | ||||||
|   name: "changeProjectName", |   name: "changeProjectName", | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     trackEvent("change", "title"); |     trackEvent(EVENT_CHANGE, "title"); | ||||||
|     return { appState: { ...appState, name: value }, commitToHistory: false }; |     return { appState: { ...appState, name: value }, commitToHistory: false }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData, appProps }) => ( |   PanelComponent: ({ appState, updateData }) => ( | ||||||
|     <ProjectName |     <ProjectName | ||||||
|       label={t("labels.fileTitle")} |       label={t("labels.fileTitle")} | ||||||
|       value={appState.name || "Unnamed"} |       value={appState.name || "Unnamed"} | ||||||
|       onChange={(name: string) => updateData(name)} |       onChange={(name: string) => updateData(name)} | ||||||
|       isNameEditable={ |  | ||||||
|         typeof appProps.name === "undefined" && !appState.viewModeEnabled |  | ||||||
|       } |  | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionChangeExportScale = register({ |  | ||||||
|   name: "changeExportScale", |  | ||||||
|   perform: (_elements, appState, value) => { |  | ||||||
|     return { |  | ||||||
|       appState: { ...appState, exportScale: value }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   PanelComponent: ({ elements: allElements, appState, updateData }) => { |  | ||||||
|     const elements = getNonDeletedElements(allElements); |  | ||||||
|     const exportSelected = isSomeElementSelected(elements, appState); |  | ||||||
|     const exportedElements = exportSelected |  | ||||||
|       ? getSelectedElements(elements, appState) |  | ||||||
|       : elements; |  | ||||||
|  |  | ||||||
|     return ( |  | ||||||
|       <> |  | ||||||
|         {EXPORT_SCALES.map((s) => { |  | ||||||
|           const [width, height] = getExportSize( |  | ||||||
|             exportedElements, |  | ||||||
|             DEFAULT_EXPORT_PADDING, |  | ||||||
|             s, |  | ||||||
|           ); |  | ||||||
|  |  | ||||||
|           const scaleButtonTitle = `${t( |  | ||||||
|             "buttons.scale", |  | ||||||
|           )} ${s}x (${width}x${height})`; |  | ||||||
|  |  | ||||||
|           return ( |  | ||||||
|             <ToolButton |  | ||||||
|               key={s} |  | ||||||
|               size="small" |  | ||||||
|               type="radio" |  | ||||||
|               icon={`${s}x`} |  | ||||||
|               name="export-canvas-scale" |  | ||||||
|               title={scaleButtonTitle} |  | ||||||
|               aria-label={scaleButtonTitle} |  | ||||||
|               id="export-canvas-scale" |  | ||||||
|               checked={s === appState.exportScale} |  | ||||||
|               onChange={() => updateData(s)} |  | ||||||
|             /> |  | ||||||
|           ); |  | ||||||
|         })} |  | ||||||
|       </> |  | ||||||
|     ); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export const actionChangeExportBackground = register({ | export const actionChangeExportBackground = register({ | ||||||
|   name: "changeExportBackground", |   name: "changeExportBackground", | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
| @@ -96,12 +34,14 @@ export const actionChangeExportBackground = register({ | |||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData }) => ( |   PanelComponent: ({ appState, updateData }) => ( | ||||||
|     <CheckboxItem |     <label> | ||||||
|       checked={appState.exportBackground} |       <input | ||||||
|       onChange={(checked) => updateData(checked)} |         type="checkbox" | ||||||
|     > |         checked={appState.exportBackground} | ||||||
|  |         onChange={(event) => updateData(event.target.checked)} | ||||||
|  |       />{" "} | ||||||
|       {t("labels.withBackground")} |       {t("labels.withBackground")} | ||||||
|     </CheckboxItem> |     </label> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -114,43 +54,44 @@ export const actionChangeExportEmbedScene = register({ | |||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData }) => ( |   PanelComponent: ({ appState, updateData }) => ( | ||||||
|     <CheckboxItem |     <label title={t("labels.exportEmbedScene_details")}> | ||||||
|       checked={appState.exportEmbedScene} |       <input | ||||||
|       onChange={(checked) => updateData(checked)} |         type="checkbox" | ||||||
|     > |         checked={appState.exportEmbedScene} | ||||||
|  |         onChange={(event) => updateData(event.target.checked)} | ||||||
|  |       />{" "} | ||||||
|       {t("labels.exportEmbedScene")} |       {t("labels.exportEmbedScene")} | ||||||
|       <Tooltip label={t("labels.exportEmbedScene_details")} long={true}> |     </label> | ||||||
|         <div className="excalidraw-tooltip-icon">{questionCircle}</div> |  | ||||||
|       </Tooltip> |  | ||||||
|     </CheckboxItem> |  | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionSaveToActiveFile = register({ | export const actionChangeShouldAddWatermark = register({ | ||||||
|   name: "saveToActiveFile", |   name: "changeShouldAddWatermark", | ||||||
|   perform: async (elements, appState, value, app) => { |   perform: (_elements, appState, value) => { | ||||||
|     const fileHandleExists = !!appState.fileHandle; |     return { | ||||||
|  |       appState: { ...appState, shouldAddWatermark: value }, | ||||||
|  |       commitToHistory: false, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |   PanelComponent: ({ appState, updateData }) => ( | ||||||
|  |     <label> | ||||||
|  |       <input | ||||||
|  |         type="checkbox" | ||||||
|  |         checked={appState.shouldAddWatermark} | ||||||
|  |         onChange={(event) => updateData(event.target.checked)} | ||||||
|  |       />{" "} | ||||||
|  |       {t("labels.addWatermark")} | ||||||
|  |     </label> | ||||||
|  |   ), | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | export const actionSaveScene = register({ | ||||||
|  |   name: "saveScene", | ||||||
|  |   perform: async (elements, appState, value) => { | ||||||
|     try { |     try { | ||||||
|       const { fileHandle } = isImageFileHandle(appState.fileHandle) |       const { fileHandle } = await saveAsJSON(elements, appState); | ||||||
|         ? await resaveAsImageWithScene(elements, appState, app.files) |       trackEvent(EVENT_IO, "save"); | ||||||
|         : await saveAsJSON(elements, appState, app.files); |       return { commitToHistory: false, appState: { ...appState, fileHandle } }; | ||||||
|  |  | ||||||
|       return { |  | ||||||
|         commitToHistory: false, |  | ||||||
|         appState: { |  | ||||||
|           ...appState, |  | ||||||
|           fileHandle, |  | ||||||
|           toastMessage: fileHandleExists |  | ||||||
|             ? fileHandle?.name |  | ||||||
|               ? t("toast.fileSavedToFilename").replace( |  | ||||||
|                   "{filename}", |  | ||||||
|                   `"${fileHandle.name}"`, |  | ||||||
|                 ) |  | ||||||
|               : t("toast.fileSaved") |  | ||||||
|             : null, |  | ||||||
|         }, |  | ||||||
|       }; |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       if (error?.name !== "AbortError") { |       if (error?.name !== "AbortError") { | ||||||
|         console.error(error); |         console.error(error); | ||||||
| @@ -160,26 +101,27 @@ export const actionSaveToActiveFile = register({ | |||||||
|   }, |   }, | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event.key === KEYS.S && event[KEYS.CTRL_OR_CMD] && !event.shiftKey, |     event.key === KEYS.S && event[KEYS.CTRL_OR_CMD] && !event.shiftKey, | ||||||
|   PanelComponent: ({ updateData, appState }) => ( |   PanelComponent: ({ updateData }) => ( | ||||||
|     <ActiveFile |     <ToolButton | ||||||
|       onSave={() => updateData(null)} |       type="button" | ||||||
|       fileName={appState.fileHandle?.name} |       icon={save} | ||||||
|  |       title={t("buttons.save")} | ||||||
|  |       aria-label={t("buttons.save")} | ||||||
|  |       showAriaLabel={useIsMobile()} | ||||||
|  |       onClick={() => updateData(null)} | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionSaveFileToDisk = register({ | export const actionSaveAsScene = register({ | ||||||
|   name: "saveFileToDisk", |   name: "saveAsScene", | ||||||
|   perform: async (elements, appState, value, app) => { |   perform: async (elements, appState, value) => { | ||||||
|     try { |     try { | ||||||
|       const { fileHandle } = await saveAsJSON( |       const { fileHandle } = await saveAsJSON(elements, { | ||||||
|         elements, |         ...appState, | ||||||
|         { |         fileHandle: null, | ||||||
|           ...appState, |       }); | ||||||
|           fileHandle: null, |       trackEvent(EVENT_IO, "save as"); | ||||||
|         }, |  | ||||||
|         app.files, |  | ||||||
|       ); |  | ||||||
|       return { commitToHistory: false, appState: { ...appState, fileHandle } }; |       return { commitToHistory: false, appState: { ...appState, fileHandle } }; | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       if (error?.name !== "AbortError") { |       if (error?.name !== "AbortError") { | ||||||
| @@ -197,41 +139,28 @@ export const actionSaveFileToDisk = register({ | |||||||
|       title={t("buttons.saveAs")} |       title={t("buttons.saveAs")} | ||||||
|       aria-label={t("buttons.saveAs")} |       aria-label={t("buttons.saveAs")} | ||||||
|       showAriaLabel={useIsMobile()} |       showAriaLabel={useIsMobile()} | ||||||
|       hidden={!nativeFileSystemSupported} |       hidden={ | ||||||
|  |         !("chooseFileSystemEntries" in window || "showOpenFilePicker" in window) | ||||||
|  |       } | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       data-testid="save-as-button" |  | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionLoadScene = register({ | export const actionLoadScene = register({ | ||||||
|   name: "loadScene", |   name: "loadScene", | ||||||
|   perform: async (elements, appState, _, app) => { |   perform: ( | ||||||
|     try { |     elements, | ||||||
|       const { |     appState, | ||||||
|         elements: loadedElements, |     { elements: loadedElements, appState: loadedAppState, error }, | ||||||
|         appState: loadedAppState, |   ) => ({ | ||||||
|         files, |     elements: loadedElements, | ||||||
|       } = await loadFromJSON(appState, elements); |     appState: { | ||||||
|       return { |       ...loadedAppState, | ||||||
|         elements: loadedElements, |       errorMessage: error, | ||||||
|         appState: loadedAppState, |     }, | ||||||
|         files, |     commitToHistory: true, | ||||||
|         commitToHistory: true, |   }), | ||||||
|       }; |  | ||||||
|     } catch (error) { |  | ||||||
|       if (error?.name === "AbortError") { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       return { |  | ||||||
|         elements, |  | ||||||
|         appState: { ...appState, errorMessage: error.message }, |  | ||||||
|         files: app.files, |  | ||||||
|         commitToHistory: false, |  | ||||||
|       }; |  | ||||||
|     } |  | ||||||
|   }, |  | ||||||
|   keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.key === KEYS.O, |  | ||||||
|   PanelComponent: ({ updateData, appState }) => ( |   PanelComponent: ({ updateData, appState }) => ( | ||||||
|     <ToolButton |     <ToolButton | ||||||
|       type="button" |       type="button" | ||||||
| @@ -239,36 +168,16 @@ export const actionLoadScene = register({ | |||||||
|       title={t("buttons.load")} |       title={t("buttons.load")} | ||||||
|       aria-label={t("buttons.load")} |       aria-label={t("buttons.load")} | ||||||
|       showAriaLabel={useIsMobile()} |       showAriaLabel={useIsMobile()} | ||||||
|       onClick={updateData} |       onClick={() => { | ||||||
|       data-testid="load-button" |         loadFromJSON(appState) | ||||||
|  |           .then(({ elements, appState }) => { | ||||||
|  |             updateData({ elements, appState }); | ||||||
|  |           }) | ||||||
|  |           .catch(muteFSAbortError) | ||||||
|  |           .catch((error) => { | ||||||
|  |             updateData({ error: error.message }); | ||||||
|  |           }); | ||||||
|  |       }} | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionExportWithDarkMode = register({ |  | ||||||
|   name: "exportWithDarkMode", |  | ||||||
|   perform: (_elements, appState, value) => { |  | ||||||
|     return { |  | ||||||
|       appState: { ...appState, exportWithDarkMode: value }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   PanelComponent: ({ appState, updateData }) => ( |  | ||||||
|     <div |  | ||||||
|       style={{ |  | ||||||
|         display: "flex", |  | ||||||
|         justifyContent: "flex-end", |  | ||||||
|         marginTop: "-45px", |  | ||||||
|         marginBottom: "10px", |  | ||||||
|       }} |  | ||||||
|     > |  | ||||||
|       <DarkModeToggle |  | ||||||
|         value={appState.exportWithDarkMode ? THEME.DARK : THEME.LIGHT} |  | ||||||
|         onChange={(theme: Theme) => { |  | ||||||
|           updateData(theme === THEME.DARK); |  | ||||||
|         }} |  | ||||||
|         title={t("labels.toggleExportColorScheme")} |  | ||||||
|       /> |  | ||||||
|     </div> |  | ||||||
|   ), |  | ||||||
| }); |  | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import { KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { isInvisiblySmallElement } from "../element"; | import { isInvisiblySmallElement } from "../element"; | ||||||
| import { resetCursor } from "../utils"; | import { resetCursor } from "../utils"; | ||||||
|  | import React from "react"; | ||||||
| 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"; | ||||||
| @@ -17,7 +18,7 @@ import { isBindingElement } from "../element/typeChecks"; | |||||||
|  |  | ||||||
| export const actionFinalize = register({ | export const actionFinalize = register({ | ||||||
|   name: "finalize", |   name: "finalize", | ||||||
|   perform: (elements, appState, _, { canvas, focusContainer }) => { |   perform: (elements, appState) => { | ||||||
|     if (appState.editingLinearElement) { |     if (appState.editingLinearElement) { | ||||||
|       const { |       const { | ||||||
|         elementId, |         elementId, | ||||||
| @@ -49,25 +50,20 @@ export const actionFinalize = register({ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     let newElements = elements; |     let newElements = elements; | ||||||
|  |  | ||||||
|     if (appState.pendingImageElement) { |  | ||||||
|       mutateElement(appState.pendingImageElement, { isDeleted: true }, false); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (window.document.activeElement instanceof HTMLElement) { |     if (window.document.activeElement instanceof HTMLElement) { | ||||||
|       focusContainer(); |       window.document.activeElement.blur(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const multiPointElement = appState.multiElement |     const multiPointElement = appState.multiElement | ||||||
|       ? appState.multiElement |       ? appState.multiElement | ||||||
|       : appState.editingElement?.type === "freedraw" |       : appState.editingElement?.type === "draw" | ||||||
|       ? appState.editingElement |       ? appState.editingElement | ||||||
|       : null; |       : null; | ||||||
|  |  | ||||||
|     if (multiPointElement) { |     if (multiPointElement) { | ||||||
|       // pen and mouse have hover |       // pen and mouse have hover | ||||||
|       if ( |       if ( | ||||||
|         multiPointElement.type !== "freedraw" && |         multiPointElement.type !== "draw" && | ||||||
|         appState.lastPointerDownWith !== "touch" |         appState.lastPointerDownWith !== "touch" | ||||||
|       ) { |       ) { | ||||||
|         const { points, lastCommittedPoint } = multiPointElement; |         const { points, lastCommittedPoint } = multiPointElement; | ||||||
| @@ -87,10 +83,10 @@ export const actionFinalize = register({ | |||||||
|       // If the multi point line closes the loop, |       // If the multi point line closes the loop, | ||||||
|       // set the last point to first point. |       // set the last point to first point. | ||||||
|       // This ensures that loop remains closed at different scales. |       // This ensures that loop remains closed at different scales. | ||||||
|       const isLoop = isPathALoop(multiPointElement.points, appState.zoom.value); |       const isLoop = isPathALoop(multiPointElement.points); | ||||||
|       if ( |       if ( | ||||||
|         multiPointElement.type === "line" || |         multiPointElement.type === "line" || | ||||||
|         multiPointElement.type === "freedraw" |         multiPointElement.type === "draw" | ||||||
|       ) { |       ) { | ||||||
|         if (isLoop) { |         if (isLoop) { | ||||||
|           const linePoints = multiPointElement.points; |           const linePoints = multiPointElement.points; | ||||||
| @@ -122,25 +118,19 @@ export const actionFinalize = register({ | |||||||
|         ); |         ); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (!appState.elementLocked && appState.elementType !== "freedraw") { |       if (!appState.elementLocked) { | ||||||
|         appState.selectedElementIds[multiPointElement.id] = true; |         appState.selectedElementIds[multiPointElement.id] = true; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |     if (!appState.elementLocked || !multiPointElement) { | ||||||
|     if ( |       resetCursor(); | ||||||
|       (!appState.elementLocked && appState.elementType !== "freedraw") || |  | ||||||
|       !multiPointElement |  | ||||||
|     ) { |  | ||||||
|       resetCursor(canvas); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|       elements: newElements, |       elements: newElements, | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|         elementType: |         elementType: | ||||||
|           (appState.elementLocked || appState.elementType === "freedraw") && |           appState.elementLocked && multiPointElement | ||||||
|           multiPointElement |  | ||||||
|             ? appState.elementType |             ? appState.elementType | ||||||
|             : "selection", |             : "selection", | ||||||
|         draggingElement: null, |         draggingElement: null, | ||||||
| @@ -149,17 +139,14 @@ export const actionFinalize = register({ | |||||||
|         startBoundElement: null, |         startBoundElement: null, | ||||||
|         suggestedBindings: [], |         suggestedBindings: [], | ||||||
|         selectedElementIds: |         selectedElementIds: | ||||||
|           multiPointElement && |           multiPointElement && !appState.elementLocked | ||||||
|           !appState.elementLocked && |  | ||||||
|           appState.elementType !== "freedraw" |  | ||||||
|             ? { |             ? { | ||||||
|                 ...appState.selectedElementIds, |                 ...appState.selectedElementIds, | ||||||
|                 [multiPointElement.id]: true, |                 [multiPointElement.id]: true, | ||||||
|               } |               } | ||||||
|             : appState.selectedElementIds, |             : appState.selectedElementIds, | ||||||
|         pendingImageElement: null, |  | ||||||
|       }, |       }, | ||||||
|       commitToHistory: appState.elementType === "freedraw", |       commitToHistory: appState.elementType === "draw", | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   keyTest: (event, appState) => |   keyTest: (event, appState) => | ||||||
|   | |||||||
| @@ -1,207 +0,0 @@ | |||||||
| import { register } from "./register"; |  | ||||||
| import { getSelectedElements } from "../scene"; |  | ||||||
| import { getElementMap, getNonDeletedElements } from "../element"; |  | ||||||
| import { mutateElement } from "../element/mutateElement"; |  | ||||||
| import { ExcalidrawElement, NonDeleted } from "../element/types"; |  | ||||||
| import { normalizeAngle, resizeSingleElement } from "../element/resizeElements"; |  | ||||||
| import { AppState } from "../types"; |  | ||||||
| import { getTransformHandles } from "../element/transformHandles"; |  | ||||||
| import { isFreeDrawElement, isLinearElement } from "../element/typeChecks"; |  | ||||||
| import { updateBoundElements } from "../element/binding"; |  | ||||||
| import { LinearElementEditor } from "../element/linearElementEditor"; |  | ||||||
|  |  | ||||||
| const enableActionFlipHorizontal = ( |  | ||||||
|   elements: readonly ExcalidrawElement[], |  | ||||||
|   appState: AppState, |  | ||||||
| ) => { |  | ||||||
|   const eligibleElements = getSelectedElements( |  | ||||||
|     getNonDeletedElements(elements), |  | ||||||
|     appState, |  | ||||||
|   ); |  | ||||||
|   return eligibleElements.length === 1 && eligibleElements[0].type !== "text"; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const enableActionFlipVertical = ( |  | ||||||
|   elements: readonly ExcalidrawElement[], |  | ||||||
|   appState: AppState, |  | ||||||
| ) => { |  | ||||||
|   const eligibleElements = getSelectedElements( |  | ||||||
|     getNonDeletedElements(elements), |  | ||||||
|     appState, |  | ||||||
|   ); |  | ||||||
|   return eligibleElements.length === 1; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export const actionFlipHorizontal = register({ |  | ||||||
|   name: "flipHorizontal", |  | ||||||
|   perform: (elements, appState) => { |  | ||||||
|     return { |  | ||||||
|       elements: flipSelectedElements(elements, appState, "horizontal"), |  | ||||||
|       appState, |  | ||||||
|       commitToHistory: true, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   keyTest: (event) => event.shiftKey && event.code === "KeyH", |  | ||||||
|   contextItemLabel: "labels.flipHorizontal", |  | ||||||
|   contextItemPredicate: (elements, appState) => |  | ||||||
|     enableActionFlipHorizontal(elements, appState), |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| export const actionFlipVertical = register({ |  | ||||||
|   name: "flipVertical", |  | ||||||
|   perform: (elements, appState) => { |  | ||||||
|     return { |  | ||||||
|       elements: flipSelectedElements(elements, appState, "vertical"), |  | ||||||
|       appState, |  | ||||||
|       commitToHistory: true, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   keyTest: (event) => event.shiftKey && event.code === "KeyV", |  | ||||||
|   contextItemLabel: "labels.flipVertical", |  | ||||||
|   contextItemPredicate: (elements, appState) => |  | ||||||
|     enableActionFlipVertical(elements, appState), |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const flipSelectedElements = ( |  | ||||||
|   elements: readonly ExcalidrawElement[], |  | ||||||
|   appState: Readonly<AppState>, |  | ||||||
|   flipDirection: "horizontal" | "vertical", |  | ||||||
| ) => { |  | ||||||
|   const selectedElements = getSelectedElements( |  | ||||||
|     getNonDeletedElements(elements), |  | ||||||
|     appState, |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   // remove once we allow for groups of elements to be flipped |  | ||||||
|   if (selectedElements.length > 1) { |  | ||||||
|     return elements; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const updatedElements = flipElements( |  | ||||||
|     selectedElements, |  | ||||||
|     appState, |  | ||||||
|     flipDirection, |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   const updatedElementsMap = getElementMap(updatedElements); |  | ||||||
|  |  | ||||||
|   return elements.map((element) => updatedElementsMap[element.id] || element); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const flipElements = ( |  | ||||||
|   elements: NonDeleted<ExcalidrawElement>[], |  | ||||||
|   appState: AppState, |  | ||||||
|   flipDirection: "horizontal" | "vertical", |  | ||||||
| ): ExcalidrawElement[] => { |  | ||||||
|   elements.forEach((element) => { |  | ||||||
|     flipElement(element, appState); |  | ||||||
|     // If vertical flip, rotate an extra 180 |  | ||||||
|     if (flipDirection === "vertical") { |  | ||||||
|       rotateElement(element, Math.PI); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|   return elements; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const flipElement = ( |  | ||||||
|   element: NonDeleted<ExcalidrawElement>, |  | ||||||
|   appState: AppState, |  | ||||||
| ) => { |  | ||||||
|   const originalX = element.x; |  | ||||||
|   const originalY = element.y; |  | ||||||
|   const width = element.width; |  | ||||||
|   const height = element.height; |  | ||||||
|   const originalAngle = normalizeAngle(element.angle); |  | ||||||
|  |  | ||||||
|   let finalOffsetX = 0; |  | ||||||
|   if (isLinearElement(element) || isFreeDrawElement(element)) { |  | ||||||
|     finalOffsetX = |  | ||||||
|       element.points.reduce((max, point) => Math.max(max, point[0]), 0) * 2 - |  | ||||||
|       element.width; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // Rotate back to zero, if necessary |  | ||||||
|   mutateElement(element, { |  | ||||||
|     angle: normalizeAngle(0), |  | ||||||
|   }); |  | ||||||
|   // Flip unrotated by pulling TransformHandle to opposite side |  | ||||||
|   const transformHandles = getTransformHandles(element, appState.zoom); |  | ||||||
|   let usingNWHandle = true; |  | ||||||
|   let newNCoordsX = 0; |  | ||||||
|   let nHandle = transformHandles.nw; |  | ||||||
|   if (!nHandle) { |  | ||||||
|     // Use ne handle instead |  | ||||||
|     usingNWHandle = false; |  | ||||||
|     nHandle = transformHandles.ne; |  | ||||||
|     if (!nHandle) { |  | ||||||
|       mutateElement(element, { |  | ||||||
|         angle: originalAngle, |  | ||||||
|       }); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (isLinearElement(element)) { |  | ||||||
|     for (let i = 1; i < element.points.length; i++) { |  | ||||||
|       LinearElementEditor.movePoint(element, i, [ |  | ||||||
|         -element.points[i][0], |  | ||||||
|         element.points[i][1], |  | ||||||
|       ]); |  | ||||||
|     } |  | ||||||
|     LinearElementEditor.normalizePoints(element); |  | ||||||
|   } else { |  | ||||||
|     // calculate new x-coord for transformation |  | ||||||
|     newNCoordsX = usingNWHandle ? element.x + 2 * width : element.x - 2 * width; |  | ||||||
|     resizeSingleElement( |  | ||||||
|       element, |  | ||||||
|       true, |  | ||||||
|       element, |  | ||||||
|       usingNWHandle ? "nw" : "ne", |  | ||||||
|       false, |  | ||||||
|       newNCoordsX, |  | ||||||
|       nHandle[1], |  | ||||||
|     ); |  | ||||||
|     // fix the size to account for handle sizes |  | ||||||
|     mutateElement(element, { |  | ||||||
|       width, |  | ||||||
|       height, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // Rotate by (360 degrees - original angle) |  | ||||||
|   let angle = normalizeAngle(2 * Math.PI - originalAngle); |  | ||||||
|   if (angle < 0) { |  | ||||||
|     // check, probably unnecessary |  | ||||||
|     angle = normalizeAngle(angle + 2 * Math.PI); |  | ||||||
|   } |  | ||||||
|   mutateElement(element, { |  | ||||||
|     angle, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   // Move back to original spot to appear "flipped in place" |  | ||||||
|   mutateElement(element, { |  | ||||||
|     x: originalX + finalOffsetX, |  | ||||||
|     y: originalY, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   updateBoundElements(element); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const rotateElement = (element: ExcalidrawElement, rotationAngle: number) => { |  | ||||||
|   const originalX = element.x; |  | ||||||
|   const originalY = element.y; |  | ||||||
|   let angle = normalizeAngle(element.angle + rotationAngle); |  | ||||||
|   if (angle < 0) { |  | ||||||
|     // check, probably unnecessary |  | ||||||
|     angle = normalizeAngle(2 * Math.PI + angle); |  | ||||||
|   } |  | ||||||
|   mutateElement(element, { |  | ||||||
|     angle, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   // Move back to original spot |  | ||||||
|   mutateElement(element, { |  | ||||||
|     x: originalX, |  | ||||||
|     y: originalY, |  | ||||||
|   }); |  | ||||||
| }; |  | ||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import React from "react"; | ||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import { getShortcutKey } from "../utils"; | import { getShortcutKey } from "../utils"; | ||||||
| @@ -124,6 +125,7 @@ export const actionGroup = register({ | |||||||
|       commitToHistory: true, |       commitToHistory: true, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|  |   contextMenuOrder: 4, | ||||||
|   contextItemLabel: "labels.group", |   contextItemLabel: "labels.group", | ||||||
|   contextItemPredicate: (elements, appState) => |   contextItemPredicate: (elements, appState) => | ||||||
|     enableActionGroup(elements, appState), |     enableActionGroup(elements, appState), | ||||||
| @@ -133,7 +135,7 @@ export const actionGroup = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       hidden={!enableActionGroup(elements, appState)} |       hidden={!enableActionGroup(elements, appState)} | ||||||
|       type="button" |       type="button" | ||||||
|       icon={<GroupIcon theme={appState.theme} />} |       icon={<GroupIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.group")} — ${getShortcutKey("CtrlOrCmd+G")}`} |       title={`${t("labels.group")} — ${getShortcutKey("CtrlOrCmd+G")}`} | ||||||
|       aria-label={t("labels.group")} |       aria-label={t("labels.group")} | ||||||
| @@ -172,6 +174,7 @@ export const actionUngroup = register({ | |||||||
|   }, |   }, | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event.shiftKey && event[KEYS.CTRL_OR_CMD] && event.code === CODES.G, |     event.shiftKey && event[KEYS.CTRL_OR_CMD] && event.code === CODES.G, | ||||||
|  |   contextMenuOrder: 5, | ||||||
|   contextItemLabel: "labels.ungroup", |   contextItemLabel: "labels.ungroup", | ||||||
|   contextItemPredicate: (elements, appState) => |   contextItemPredicate: (elements, appState) => | ||||||
|     getSelectedGroupIds(appState).length > 0, |     getSelectedGroupIds(appState).length > 0, | ||||||
| @@ -180,7 +183,7 @@ export const actionUngroup = register({ | |||||||
|     <ToolButton |     <ToolButton | ||||||
|       type="button" |       type="button" | ||||||
|       hidden={getSelectedGroupIds(appState).length === 0} |       hidden={getSelectedGroupIds(appState).length === 0} | ||||||
|       icon={<UngroupIcon theme={appState.theme} />} |       icon={<UngroupIcon appearance={appState.appearance} />} | ||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.ungroup")} — ${getShortcutKey("CtrlOrCmd+Shift+G")}`} |       title={`${t("labels.ungroup")} — ${getShortcutKey("CtrlOrCmd+Shift+G")}`} | ||||||
|       aria-label={t("labels.ungroup")} |       aria-label={t("labels.ungroup")} | ||||||
|   | |||||||
| @@ -1,11 +1,12 @@ | |||||||
| import { Action, ActionResult } from "./types"; | import { Action, ActionResult } from "./types"; | ||||||
|  | import React from "react"; | ||||||
| import { undo, redo } from "../components/icons"; | import { undo, redo } from "../components/icons"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| import History, { HistoryEntry } from "../history"; | import { SceneHistory, HistoryEntry } from "../history"; | ||||||
| import { ExcalidrawElement } from "../element/types"; | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { AppState } from "../types"; | import { AppState } from "../types"; | ||||||
| import { isWindows, KEYS } from "../keys"; | import { KEYS } from "../keys"; | ||||||
| import { getElementMap } from "../element"; | import { getElementMap } from "../element"; | ||||||
| import { newElementWith } from "../element/mutateElement"; | import { newElementWith } from "../element/mutateElement"; | ||||||
| import { fixBindingsAfterDeletion } from "../element/binding"; | import { fixBindingsAfterDeletion } from "../element/binding"; | ||||||
| @@ -58,23 +59,22 @@ const writeData = ( | |||||||
|   return { commitToHistory }; |   return { commitToHistory }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| type ActionCreator = (history: History) => Action; | const testUndo = (shift: boolean) => (event: KeyboardEvent) => | ||||||
|  |   event[KEYS.CTRL_OR_CMD] && /z/i.test(event.key) && event.shiftKey === shift; | ||||||
|  |  | ||||||
|  | type ActionCreator = (history: SceneHistory) => Action; | ||||||
|  |  | ||||||
| export const createUndoAction: ActionCreator = (history) => ({ | export const createUndoAction: ActionCreator = (history) => ({ | ||||||
|   name: "undo", |   name: "undo", | ||||||
|   perform: (elements, appState) => |   perform: (elements, appState) => | ||||||
|     writeData(elements, appState, () => history.undoOnce()), |     writeData(elements, appState, () => history.undoOnce()), | ||||||
|   keyTest: (event) => |   keyTest: testUndo(false), | ||||||
|     event[KEYS.CTRL_OR_CMD] && |   PanelComponent: ({ updateData }) => ( | ||||||
|     event.key.toLowerCase() === KEYS.Z && |  | ||||||
|     !event.shiftKey, |  | ||||||
|   PanelComponent: ({ updateData, data }) => ( |  | ||||||
|     <ToolButton |     <ToolButton | ||||||
|       type="button" |       type="button" | ||||||
|       icon={undo} |       icon={undo} | ||||||
|       aria-label={t("buttons.undo")} |       aria-label={t("buttons.undo")} | ||||||
|       onClick={updateData} |       onClick={updateData} | ||||||
|       size={data?.size || "medium"} |  | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
|   commitToHistory: () => false, |   commitToHistory: () => false, | ||||||
| @@ -84,18 +84,13 @@ export const createRedoAction: ActionCreator = (history) => ({ | |||||||
|   name: "redo", |   name: "redo", | ||||||
|   perform: (elements, appState) => |   perform: (elements, appState) => | ||||||
|     writeData(elements, appState, () => history.redoOnce()), |     writeData(elements, appState, () => history.redoOnce()), | ||||||
|   keyTest: (event) => |   keyTest: testUndo(true), | ||||||
|     (event[KEYS.CTRL_OR_CMD] && |   PanelComponent: ({ updateData }) => ( | ||||||
|       event.shiftKey && |  | ||||||
|       event.key.toLowerCase() === KEYS.Z) || |  | ||||||
|     (isWindows && event.ctrlKey && !event.shiftKey && event.key === KEYS.Y), |  | ||||||
|   PanelComponent: ({ updateData, data }) => ( |  | ||||||
|     <ToolButton |     <ToolButton | ||||||
|       type="button" |       type="button" | ||||||
|       icon={redo} |       icon={redo} | ||||||
|       aria-label={t("buttons.redo")} |       aria-label={t("buttons.redo")} | ||||||
|       onClick={updateData} |       onClick={updateData} | ||||||
|       size={data?.size || "medium"} |  | ||||||
|     /> |     /> | ||||||
|   ), |   ), | ||||||
|   commitToHistory: () => false, |   commitToHistory: () => false, | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import React from "react"; | ||||||
| import { menu, palette } from "../components/icons"; | import { menu, palette } from "../components/icons"; | ||||||
| import { ToolButton } from "../components/ToolButton"; | import { ToolButton } from "../components/ToolButton"; | ||||||
| import { t } from "../i18n"; | import { t } from "../i18n"; | ||||||
| @@ -6,6 +7,7 @@ import { register } from "./register"; | |||||||
| import { allowFullScreen, exitFullScreen, isFullScreen } from "../utils"; | import { allowFullScreen, exitFullScreen, isFullScreen } from "../utils"; | ||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { HelpIcon } from "../components/HelpIcon"; | import { HelpIcon } from "../components/HelpIcon"; | ||||||
|  | import { EVENT_DIALOG, trackEvent } from "../analytics"; | ||||||
|  |  | ||||||
| export const actionToggleCanvasMenu = register({ | export const actionToggleCanvasMenu = register({ | ||||||
|   name: "toggleCanvasMenu", |   name: "toggleCanvasMenu", | ||||||
| @@ -69,20 +71,18 @@ export const actionFullScreen = register({ | |||||||
|  |  | ||||||
| export const actionShortcuts = register({ | export const actionShortcuts = register({ | ||||||
|   name: "toggleShortcuts", |   name: "toggleShortcuts", | ||||||
|   perform: (_elements, appState, _, { focusContainer }) => { |   perform: (_elements, appState) => { | ||||||
|     if (appState.showHelpDialog) { |     trackEvent(EVENT_DIALOG, "shortcuts"); | ||||||
|       focusContainer(); |  | ||||||
|     } |  | ||||||
|     return { |     return { | ||||||
|       appState: { |       appState: { | ||||||
|         ...appState, |         ...appState, | ||||||
|         showHelpDialog: !appState.showHelpDialog, |         showShortcutsDialog: true, | ||||||
|       }, |       }, | ||||||
|       commitToHistory: false, |       commitToHistory: false, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ updateData }) => ( |   PanelComponent: ({ updateData }) => ( | ||||||
|     <HelpIcon title={t("helpDialog.title")} onClick={updateData} /> |     <HelpIcon title={t("shortcutsDialog.title")} onClick={updateData} /> | ||||||
|   ), |   ), | ||||||
|   keyTest: (event) => event.key === KEYS.QUESTION_MARK, |   keyTest: (event) => event.key === KEYS.QUESTION_MARK, | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,13 +1,16 @@ | |||||||
| import { getClientColors, getClientInitials } from "../clients"; | import React from "react"; | ||||||
| import { Avatar } from "../components/Avatar"; | import { Avatar } from "../components/Avatar"; | ||||||
| import { centerScrollOn } from "../scene/scroll"; |  | ||||||
| import { Collaborator } from "../types"; |  | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
|  | import { getClientColors, getClientInitials } from "../clients"; | ||||||
|  | import { Collaborator } from "../types"; | ||||||
|  | import { centerScrollOn } from "../scene/scroll"; | ||||||
|  | import { EVENT_SHARE, trackEvent } from "../analytics"; | ||||||
|  |  | ||||||
| export const actionGoToCollaborator = register({ | export const actionGoToCollaborator = register({ | ||||||
|   name: "goToCollaborator", |   name: "goToCollaborator", | ||||||
|   perform: (_elements, appState, value) => { |   perform: (_elements, appState, value) => { | ||||||
|     const point = value as Collaborator["pointer"]; |     const point = value as Collaborator["pointer"]; | ||||||
|  |     trackEvent(EVENT_SHARE, "go to collaborator"); | ||||||
|     if (!point) { |     if (!point) { | ||||||
|       return { appState, commitToHistory: false }; |       return { appState, commitToHistory: false }; | ||||||
|     } |     } | ||||||
| @@ -29,8 +32,8 @@ export const actionGoToCollaborator = register({ | |||||||
|       commitToHistory: false, |       commitToHistory: false, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ appState, updateData, data }) => { |   PanelComponent: ({ appState, updateData, id }) => { | ||||||
|     const clientId: string | undefined = data?.id; |     const clientId = id; | ||||||
|     if (!clientId) { |     if (!clientId) { | ||||||
|       return null; |       return null; | ||||||
|     } |     } | ||||||
| @@ -41,7 +44,7 @@ export const actionGoToCollaborator = register({ | |||||||
|       return null; |       return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const { background, stroke } = getClientColors(clientId, appState); |     const { background, stroke } = getClientColors(clientId); | ||||||
|     const shortName = getClientInitials(collaborator.username); |     const shortName = getClientInitials(collaborator.username); | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|   | |||||||
| @@ -1,66 +1,56 @@ | |||||||
| import { AppState } from "../../src/types"; | import React from "react"; | ||||||
|  | import { getLanguage } from "../i18n"; | ||||||
|  | import { | ||||||
|  |   ExcalidrawElement, | ||||||
|  |   ExcalidrawTextElement, | ||||||
|  |   TextAlign, | ||||||
|  |   FontFamily, | ||||||
|  |   ExcalidrawLinearElement, | ||||||
|  |   Arrowhead, | ||||||
|  | } from "../element/types"; | ||||||
|  | import { | ||||||
|  |   getCommonAttributeOfSelectedElements, | ||||||
|  |   isSomeElementSelected, | ||||||
|  |   getTargetElements, | ||||||
|  |   canChangeSharpness, | ||||||
|  |   canHaveArrowheads, | ||||||
|  | } from "../scene"; | ||||||
|  | import { ButtonSelect } from "../components/ButtonSelect"; | ||||||
| import { ButtonIconSelect } from "../components/ButtonIconSelect"; | import { ButtonIconSelect } from "../components/ButtonIconSelect"; | ||||||
| import { ColorPicker } from "../components/ColorPicker"; |  | ||||||
| import { IconPicker } from "../components/IconPicker"; | import { IconPicker } from "../components/IconPicker"; | ||||||
| import { | import { | ||||||
|  |   isTextElement, | ||||||
|  |   redrawTextBoundingBox, | ||||||
|  |   getNonDeletedElements, | ||||||
|  | } from "../element"; | ||||||
|  | import { isLinearElement, isLinearElementType } from "../element/typeChecks"; | ||||||
|  | import { ColorPicker } from "../components/ColorPicker"; | ||||||
|  | import { AppState } from "../../src/types"; | ||||||
|  | import { t } from "../i18n"; | ||||||
|  | import { register } from "./register"; | ||||||
|  | import { newElementWith } from "../element/mutateElement"; | ||||||
|  | import { DEFAULT_FONT_SIZE, DEFAULT_FONT_FAMILY } from "../constants"; | ||||||
|  | import { randomInteger } from "../random"; | ||||||
|  | import { | ||||||
|  |   FillHachureIcon, | ||||||
|  |   FillCrossHatchIcon, | ||||||
|  |   FillSolidIcon, | ||||||
|  |   StrokeWidthIcon, | ||||||
|  |   StrokeStyleSolidIcon, | ||||||
|  |   StrokeStyleDashedIcon, | ||||||
|  |   StrokeStyleDottedIcon, | ||||||
|  |   EdgeSharpIcon, | ||||||
|  |   EdgeRoundIcon, | ||||||
|  |   SloppinessArchitectIcon, | ||||||
|  |   SloppinessArtistIcon, | ||||||
|  |   SloppinessCartoonistIcon, | ||||||
|   ArrowheadArrowIcon, |   ArrowheadArrowIcon, | ||||||
|   ArrowheadBarIcon, |   ArrowheadBarIcon, | ||||||
|   ArrowheadDotIcon, |   ArrowheadDotIcon, | ||||||
|   ArrowheadNoneIcon, |   ArrowheadNoneIcon, | ||||||
|   EdgeRoundIcon, |  | ||||||
|   EdgeSharpIcon, |  | ||||||
|   FillCrossHatchIcon, |  | ||||||
|   FillHachureIcon, |  | ||||||
|   FillSolidIcon, |  | ||||||
|   FontFamilyCodeIcon, |  | ||||||
|   FontFamilyHandDrawnIcon, |  | ||||||
|   FontFamilyNormalIcon, |  | ||||||
|   FontSizeExtraLargeIcon, |  | ||||||
|   FontSizeLargeIcon, |  | ||||||
|   FontSizeMediumIcon, |  | ||||||
|   FontSizeSmallIcon, |  | ||||||
|   SloppinessArchitectIcon, |  | ||||||
|   SloppinessArtistIcon, |  | ||||||
|   SloppinessCartoonistIcon, |  | ||||||
|   StrokeStyleDashedIcon, |  | ||||||
|   StrokeStyleDottedIcon, |  | ||||||
|   StrokeStyleSolidIcon, |  | ||||||
|   StrokeWidthIcon, |  | ||||||
|   TextAlignCenterIcon, |  | ||||||
|   TextAlignLeftIcon, |  | ||||||
|   TextAlignRightIcon, |  | ||||||
| } from "../components/icons"; | } from "../components/icons"; | ||||||
| import { | import { EVENT_CHANGE, trackEvent } from "../analytics"; | ||||||
|   DEFAULT_FONT_FAMILY, | import colors from "../colors"; | ||||||
|   DEFAULT_FONT_SIZE, |  | ||||||
|   FONT_FAMILY, |  | ||||||
| } from "../constants"; |  | ||||||
| import { |  | ||||||
|   getNonDeletedElements, |  | ||||||
|   isTextElement, |  | ||||||
|   redrawTextBoundingBox, |  | ||||||
| } from "../element"; |  | ||||||
| import { newElementWith } from "../element/mutateElement"; |  | ||||||
| import { isLinearElement, isLinearElementType } from "../element/typeChecks"; |  | ||||||
| import { |  | ||||||
|   Arrowhead, |  | ||||||
|   ExcalidrawElement, |  | ||||||
|   ExcalidrawLinearElement, |  | ||||||
|   ExcalidrawTextElement, |  | ||||||
|   FontFamilyValues, |  | ||||||
|   TextAlign, |  | ||||||
| } from "../element/types"; |  | ||||||
| import { getLanguage, t } from "../i18n"; |  | ||||||
| import { randomInteger } from "../random"; |  | ||||||
| import { |  | ||||||
|   canChangeSharpness, |  | ||||||
|   canHaveArrowheads, |  | ||||||
|   getCommonAttributeOfSelectedElements, |  | ||||||
|   getTargetElements, |  | ||||||
|   isSomeElementSelected, |  | ||||||
| } from "../scene"; |  | ||||||
| import { hasStrokeColor } from "../scene/comparisons"; |  | ||||||
| import { register } from "./register"; |  | ||||||
|  |  | ||||||
| const changeProperty = ( | const changeProperty = ( | ||||||
|   elements: readonly ExcalidrawElement[], |   elements: readonly ExcalidrawElement[], | ||||||
| @@ -102,21 +92,23 @@ const getFormValue = function <T>( | |||||||
| export const actionChangeStrokeColor = register({ | export const actionChangeStrokeColor = register({ | ||||||
|   name: "changeStrokeColor", |   name: "changeStrokeColor", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     if (value !== appState.currentItemStrokeColor) { | ||||||
|  |       trackEvent( | ||||||
|  |         EVENT_CHANGE, | ||||||
|  |         "stroke color", | ||||||
|  |         colors.elementStroke.includes(value) | ||||||
|  |           ? `${value} (picker ${colors.elementStroke.indexOf(value)})` | ||||||
|  |           : value, | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|     return { |     return { | ||||||
|       ...(value.currentItemStrokeColor && { |       elements: changeProperty(elements, appState, (el) => | ||||||
|         elements: changeProperty(elements, appState, (el) => { |         newElementWith(el, { | ||||||
|           return hasStrokeColor(el.type) |           strokeColor: value, | ||||||
|             ? newElementWith(el, { |  | ||||||
|                 strokeColor: value.currentItemStrokeColor, |  | ||||||
|               }) |  | ||||||
|             : el; |  | ||||||
|         }), |         }), | ||||||
|       }), |       ), | ||||||
|       appState: { |       appState: { ...appState, currentItemStrokeColor: value }, | ||||||
|         ...appState, |       commitToHistory: true, | ||||||
|         ...value, |  | ||||||
|       }, |  | ||||||
|       commitToHistory: !!value.currentItemStrokeColor, |  | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
| @@ -131,11 +123,7 @@ export const actionChangeStrokeColor = register({ | |||||||
|           (element) => element.strokeColor, |           (element) => element.strokeColor, | ||||||
|           appState.currentItemStrokeColor, |           appState.currentItemStrokeColor, | ||||||
|         )} |         )} | ||||||
|         onChange={(color) => updateData({ currentItemStrokeColor: color })} |         onChange={updateData} | ||||||
|         isActive={appState.openPopup === "strokeColorPicker"} |  | ||||||
|         setActive={(active) => |  | ||||||
|           updateData({ openPopup: active ? "strokeColorPicker" : null }) |  | ||||||
|         } |  | ||||||
|       /> |       /> | ||||||
|     </> |     </> | ||||||
|   ), |   ), | ||||||
| @@ -144,19 +132,24 @@ export const actionChangeStrokeColor = register({ | |||||||
| export const actionChangeBackgroundColor = register({ | export const actionChangeBackgroundColor = register({ | ||||||
|   name: "changeBackgroundColor", |   name: "changeBackgroundColor", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     if (value !== appState.currentItemBackgroundColor) { | ||||||
|  |       trackEvent( | ||||||
|  |         EVENT_CHANGE, | ||||||
|  |         "background color", | ||||||
|  |         colors.elementBackground.includes(value) | ||||||
|  |           ? `${value} (picker ${colors.elementBackground.indexOf(value)})` | ||||||
|  |           : value, | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|       ...(value.currentItemBackgroundColor && { |       elements: changeProperty(elements, appState, (el) => | ||||||
|         elements: changeProperty(elements, appState, (el) => |         newElementWith(el, { | ||||||
|           newElementWith(el, { |           backgroundColor: value, | ||||||
|             backgroundColor: value.currentItemBackgroundColor, |         }), | ||||||
|           }), |       ), | ||||||
|         ), |       appState: { ...appState, currentItemBackgroundColor: value }, | ||||||
|       }), |       commitToHistory: true, | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         ...value, |  | ||||||
|       }, |  | ||||||
|       commitToHistory: !!value.currentItemBackgroundColor, |  | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
| @@ -171,11 +164,7 @@ export const actionChangeBackgroundColor = register({ | |||||||
|           (element) => element.backgroundColor, |           (element) => element.backgroundColor, | ||||||
|           appState.currentItemBackgroundColor, |           appState.currentItemBackgroundColor, | ||||||
|         )} |         )} | ||||||
|         onChange={(color) => updateData({ currentItemBackgroundColor: color })} |         onChange={updateData} | ||||||
|         isActive={appState.openPopup === "backgroundColorPicker"} |  | ||||||
|         setActive={(active) => |  | ||||||
|           updateData({ openPopup: active ? "backgroundColorPicker" : null }) |  | ||||||
|         } |  | ||||||
|       /> |       /> | ||||||
|     </> |     </> | ||||||
|   ), |   ), | ||||||
| @@ -184,6 +173,7 @@ export const actionChangeBackgroundColor = register({ | |||||||
| export const actionChangeFillStyle = register({ | export const actionChangeFillStyle = register({ | ||||||
|   name: "changeFillStyle", |   name: "changeFillStyle", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     trackEvent(EVENT_CHANGE, "fill", value); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -202,17 +192,17 @@ export const actionChangeFillStyle = register({ | |||||||
|           { |           { | ||||||
|             value: "hachure", |             value: "hachure", | ||||||
|             text: t("labels.hachure"), |             text: t("labels.hachure"), | ||||||
|             icon: <FillHachureIcon theme={appState.theme} />, |             icon: <FillHachureIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: "cross-hatch", |             value: "cross-hatch", | ||||||
|             text: t("labels.crossHatch"), |             text: t("labels.crossHatch"), | ||||||
|             icon: <FillCrossHatchIcon theme={appState.theme} />, |             icon: <FillCrossHatchIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: "solid", |             value: "solid", | ||||||
|             text: t("labels.solid"), |             text: t("labels.solid"), | ||||||
|             icon: <FillSolidIcon theme={appState.theme} />, |             icon: <FillSolidIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|         ]} |         ]} | ||||||
|         group="fill" |         group="fill" | ||||||
| @@ -233,6 +223,7 @@ export const actionChangeFillStyle = register({ | |||||||
| export const actionChangeStrokeWidth = register({ | export const actionChangeStrokeWidth = register({ | ||||||
|   name: "changeStrokeWidth", |   name: "changeStrokeWidth", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     trackEvent(EVENT_CHANGE, "stroke", "width", value); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -252,17 +243,32 @@ export const actionChangeStrokeWidth = register({ | |||||||
|           { |           { | ||||||
|             value: 1, |             value: 1, | ||||||
|             text: t("labels.thin"), |             text: t("labels.thin"), | ||||||
|             icon: <StrokeWidthIcon theme={appState.theme} strokeWidth={2} />, |             icon: ( | ||||||
|  |               <StrokeWidthIcon | ||||||
|  |                 appearance={appState.appearance} | ||||||
|  |                 strokeWidth={2} | ||||||
|  |               /> | ||||||
|  |             ), | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: 2, |             value: 2, | ||||||
|             text: t("labels.bold"), |             text: t("labels.bold"), | ||||||
|             icon: <StrokeWidthIcon theme={appState.theme} strokeWidth={6} />, |             icon: ( | ||||||
|  |               <StrokeWidthIcon | ||||||
|  |                 appearance={appState.appearance} | ||||||
|  |                 strokeWidth={6} | ||||||
|  |               /> | ||||||
|  |             ), | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: 4, |             value: 4, | ||||||
|             text: t("labels.extraBold"), |             text: t("labels.extraBold"), | ||||||
|             icon: <StrokeWidthIcon theme={appState.theme} strokeWidth={10} />, |             icon: ( | ||||||
|  |               <StrokeWidthIcon | ||||||
|  |                 appearance={appState.appearance} | ||||||
|  |                 strokeWidth={10} | ||||||
|  |               /> | ||||||
|  |             ), | ||||||
|           }, |           }, | ||||||
|         ]} |         ]} | ||||||
|         value={getFormValue( |         value={getFormValue( | ||||||
| @@ -280,6 +286,7 @@ export const actionChangeStrokeWidth = register({ | |||||||
| export const actionChangeSloppiness = register({ | export const actionChangeSloppiness = register({ | ||||||
|   name: "changeSloppiness", |   name: "changeSloppiness", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     trackEvent(EVENT_CHANGE, "stroke", "sloppiness", value); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -300,17 +307,17 @@ export const actionChangeSloppiness = register({ | |||||||
|           { |           { | ||||||
|             value: 0, |             value: 0, | ||||||
|             text: t("labels.architect"), |             text: t("labels.architect"), | ||||||
|             icon: <SloppinessArchitectIcon theme={appState.theme} />, |             icon: <SloppinessArchitectIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: 1, |             value: 1, | ||||||
|             text: t("labels.artist"), |             text: t("labels.artist"), | ||||||
|             icon: <SloppinessArtistIcon theme={appState.theme} />, |             icon: <SloppinessArtistIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: 2, |             value: 2, | ||||||
|             text: t("labels.cartoonist"), |             text: t("labels.cartoonist"), | ||||||
|             icon: <SloppinessCartoonistIcon theme={appState.theme} />, |             icon: <SloppinessCartoonistIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|         ]} |         ]} | ||||||
|         value={getFormValue( |         value={getFormValue( | ||||||
| @@ -328,6 +335,7 @@ export const actionChangeSloppiness = register({ | |||||||
| export const actionChangeStrokeStyle = register({ | export const actionChangeStrokeStyle = register({ | ||||||
|   name: "changeStrokeStyle", |   name: "changeStrokeStyle", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     trackEvent(EVENT_CHANGE, "style", value); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -347,17 +355,17 @@ export const actionChangeStrokeStyle = register({ | |||||||
|           { |           { | ||||||
|             value: "solid", |             value: "solid", | ||||||
|             text: t("labels.strokeStyle_solid"), |             text: t("labels.strokeStyle_solid"), | ||||||
|             icon: <StrokeStyleSolidIcon theme={appState.theme} />, |             icon: <StrokeStyleSolidIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: "dashed", |             value: "dashed", | ||||||
|             text: t("labels.strokeStyle_dashed"), |             text: t("labels.strokeStyle_dashed"), | ||||||
|             icon: <StrokeStyleDashedIcon theme={appState.theme} />, |             icon: <StrokeStyleDashedIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: "dotted", |             value: "dotted", | ||||||
|             text: t("labels.strokeStyle_dotted"), |             text: t("labels.strokeStyle_dotted"), | ||||||
|             icon: <StrokeStyleDottedIcon theme={appState.theme} />, |             icon: <StrokeStyleDottedIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|         ]} |         ]} | ||||||
|         value={getFormValue( |         value={getFormValue( | ||||||
| @@ -375,6 +383,7 @@ export const actionChangeStrokeStyle = register({ | |||||||
| export const actionChangeOpacity = register({ | export const actionChangeOpacity = register({ | ||||||
|   name: "changeOpacity", |   name: "changeOpacity", | ||||||
|   perform: (elements, appState, value) => { |   perform: (elements, appState, value) => { | ||||||
|  |     trackEvent(EVENT_CHANGE, "opacity", "value", value); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -446,29 +455,13 @@ export const actionChangeFontSize = register({ | |||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
|     <fieldset> |     <fieldset> | ||||||
|       <legend>{t("labels.fontSize")}</legend> |       <legend>{t("labels.fontSize")}</legend> | ||||||
|       <ButtonIconSelect |       <ButtonSelect | ||||||
|         group="font-size" |         group="font-size" | ||||||
|         options={[ |         options={[ | ||||||
|           { |           { value: 16, text: t("labels.small") }, | ||||||
|             value: 16, |           { value: 20, text: t("labels.medium") }, | ||||||
|             text: t("labels.small"), |           { value: 28, text: t("labels.large") }, | ||||||
|             icon: <FontSizeSmallIcon theme={appState.theme} />, |           { value: 36, text: t("labels.veryLarge") }, | ||||||
|           }, |  | ||||||
|           { |  | ||||||
|             value: 20, |  | ||||||
|             text: t("labels.medium"), |  | ||||||
|             icon: <FontSizeMediumIcon theme={appState.theme} />, |  | ||||||
|           }, |  | ||||||
|           { |  | ||||||
|             value: 28, |  | ||||||
|             text: t("labels.large"), |  | ||||||
|             icon: <FontSizeLargeIcon theme={appState.theme} />, |  | ||||||
|           }, |  | ||||||
|           { |  | ||||||
|             value: 36, |  | ||||||
|             text: t("labels.veryLarge"), |  | ||||||
|             icon: <FontSizeExtraLargeIcon theme={appState.theme} />, |  | ||||||
|           }, |  | ||||||
|         ]} |         ]} | ||||||
|         value={getFormValue( |         value={getFormValue( | ||||||
|           elements, |           elements, | ||||||
| @@ -505,32 +498,16 @@ export const actionChangeFontFamily = register({ | |||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   PanelComponent: ({ elements, appState, updateData }) => { |   PanelComponent: ({ elements, appState, updateData }) => { | ||||||
|     const options: { |     const options: { value: FontFamily; text: string }[] = [ | ||||||
|       value: FontFamilyValues; |       { value: 1, text: t("labels.handDrawn") }, | ||||||
|       text: string; |       { value: 2, text: t("labels.normal") }, | ||||||
|       icon: JSX.Element; |       { value: 3, text: t("labels.code") }, | ||||||
|     }[] = [ |  | ||||||
|       { |  | ||||||
|         value: FONT_FAMILY.Virgil, |  | ||||||
|         text: t("labels.handDrawn"), |  | ||||||
|         icon: <FontFamilyHandDrawnIcon theme={appState.theme} />, |  | ||||||
|       }, |  | ||||||
|       { |  | ||||||
|         value: FONT_FAMILY.Helvetica, |  | ||||||
|         text: t("labels.normal"), |  | ||||||
|         icon: <FontFamilyNormalIcon theme={appState.theme} />, |  | ||||||
|       }, |  | ||||||
|       { |  | ||||||
|         value: FONT_FAMILY.Cascadia, |  | ||||||
|         text: t("labels.code"), |  | ||||||
|         icon: <FontFamilyCodeIcon theme={appState.theme} />, |  | ||||||
|       }, |  | ||||||
|     ]; |     ]; | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|       <fieldset> |       <fieldset> | ||||||
|         <legend>{t("labels.fontFamily")}</legend> |         <legend>{t("labels.fontFamily")}</legend> | ||||||
|         <ButtonIconSelect<FontFamilyValues | false> |         <ButtonSelect<FontFamily | false> | ||||||
|           group="font-family" |           group="font-family" | ||||||
|           options={options} |           options={options} | ||||||
|           value={getFormValue( |           value={getFormValue( | ||||||
| @@ -571,24 +548,12 @@ export const actionChangeTextAlign = register({ | |||||||
|   PanelComponent: ({ elements, appState, updateData }) => ( |   PanelComponent: ({ elements, appState, updateData }) => ( | ||||||
|     <fieldset> |     <fieldset> | ||||||
|       <legend>{t("labels.textAlign")}</legend> |       <legend>{t("labels.textAlign")}</legend> | ||||||
|       <ButtonIconSelect<TextAlign | false> |       <ButtonSelect<TextAlign | false> | ||||||
|         group="text-align" |         group="text-align" | ||||||
|         options={[ |         options={[ | ||||||
|           { |           { value: "left", text: t("labels.left") }, | ||||||
|             value: "left", |           { value: "center", text: t("labels.center") }, | ||||||
|             text: t("labels.left"), |           { value: "right", text: t("labels.right") }, | ||||||
|             icon: <TextAlignLeftIcon theme={appState.theme} />, |  | ||||||
|           }, |  | ||||||
|           { |  | ||||||
|             value: "center", |  | ||||||
|             text: t("labels.center"), |  | ||||||
|             icon: <TextAlignCenterIcon theme={appState.theme} />, |  | ||||||
|           }, |  | ||||||
|           { |  | ||||||
|             value: "right", |  | ||||||
|             text: t("labels.right"), |  | ||||||
|             icon: <TextAlignRightIcon theme={appState.theme} />, |  | ||||||
|           }, |  | ||||||
|         ]} |         ]} | ||||||
|         value={getFormValue( |         value={getFormValue( | ||||||
|           elements, |           elements, | ||||||
| @@ -615,6 +580,7 @@ export const actionChangeSharpness = register({ | |||||||
|     const shouldUpdateForLinearElements = targetElements.length |     const shouldUpdateForLinearElements = targetElements.length | ||||||
|       ? targetElements.every(isLinearElement) |       ? targetElements.every(isLinearElement) | ||||||
|       : isLinearElementType(appState.elementType); |       : isLinearElementType(appState.elementType); | ||||||
|  |     trackEvent(EVENT_CHANGE, "edge", value); | ||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => |       elements: changeProperty(elements, appState, (el) => | ||||||
|         newElementWith(el, { |         newElementWith(el, { | ||||||
| @@ -642,12 +608,12 @@ export const actionChangeSharpness = register({ | |||||||
|           { |           { | ||||||
|             value: "sharp", |             value: "sharp", | ||||||
|             text: t("labels.sharp"), |             text: t("labels.sharp"), | ||||||
|             icon: <EdgeSharpIcon theme={appState.theme} />, |             icon: <EdgeSharpIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|             value: "round", |             value: "round", | ||||||
|             text: t("labels.round"), |             text: t("labels.round"), | ||||||
|             icon: <EdgeRoundIcon theme={appState.theme} />, |             icon: <EdgeRoundIcon appearance={appState.appearance} />, | ||||||
|           }, |           }, | ||||||
|         ]} |         ]} | ||||||
|         value={getFormValue( |         value={getFormValue( | ||||||
| @@ -676,6 +642,12 @@ export const actionChangeArrowhead = register({ | |||||||
|     return { |     return { | ||||||
|       elements: changeProperty(elements, appState, (el) => { |       elements: changeProperty(elements, appState, (el) => { | ||||||
|         if (isLinearElement(el)) { |         if (isLinearElement(el)) { | ||||||
|  |           trackEvent( | ||||||
|  |             EVENT_CHANGE, | ||||||
|  |             `arrowhead ${value.position}`, | ||||||
|  |             value.type || "none", | ||||||
|  |           ); | ||||||
|  |  | ||||||
|           const { position, type } = value; |           const { position, type } = value; | ||||||
|  |  | ||||||
|           if (position === "start") { |           if (position === "start") { | ||||||
| @@ -715,27 +687,40 @@ export const actionChangeArrowhead = register({ | |||||||
|               { |               { | ||||||
|                 value: null, |                 value: null, | ||||||
|                 text: t("labels.arrowhead_none"), |                 text: t("labels.arrowhead_none"), | ||||||
|                 icon: <ArrowheadNoneIcon theme={appState.theme} />, |                 icon: <ArrowheadNoneIcon appearance={appState.appearance} />, | ||||||
|                 keyBinding: "q", |                 keyBinding: "q", | ||||||
|               }, |               }, | ||||||
|               { |               { | ||||||
|                 value: "arrow", |                 value: "arrow", | ||||||
|                 text: t("labels.arrowhead_arrow"), |                 text: t("labels.arrowhead_arrow"), | ||||||
|                 icon: ( |                 icon: ( | ||||||
|                   <ArrowheadArrowIcon theme={appState.theme} flip={!isRTL} /> |                   <ArrowheadArrowIcon | ||||||
|  |                     appearance={appState.appearance} | ||||||
|  |                     flip={!isRTL} | ||||||
|  |                   /> | ||||||
|                 ), |                 ), | ||||||
|                 keyBinding: "w", |                 keyBinding: "w", | ||||||
|               }, |               }, | ||||||
|               { |               { | ||||||
|                 value: "bar", |                 value: "bar", | ||||||
|                 text: t("labels.arrowhead_bar"), |                 text: t("labels.arrowhead_bar"), | ||||||
|                 icon: <ArrowheadBarIcon theme={appState.theme} flip={!isRTL} />, |                 icon: ( | ||||||
|  |                   <ArrowheadBarIcon | ||||||
|  |                     appearance={appState.appearance} | ||||||
|  |                     flip={!isRTL} | ||||||
|  |                   /> | ||||||
|  |                 ), | ||||||
|                 keyBinding: "e", |                 keyBinding: "e", | ||||||
|               }, |               }, | ||||||
|               { |               { | ||||||
|                 value: "dot", |                 value: "dot", | ||||||
|                 text: t("labels.arrowhead_dot"), |                 text: t("labels.arrowhead_dot"), | ||||||
|                 icon: <ArrowheadDotIcon theme={appState.theme} flip={!isRTL} />, |                 icon: ( | ||||||
|  |                   <ArrowheadDotIcon | ||||||
|  |                     appearance={appState.appearance} | ||||||
|  |                     flip={!isRTL} | ||||||
|  |                   /> | ||||||
|  |                 ), | ||||||
|                 keyBinding: "r", |                 keyBinding: "r", | ||||||
|               }, |               }, | ||||||
|             ]} |             ]} | ||||||
| @@ -758,27 +743,40 @@ export const actionChangeArrowhead = register({ | |||||||
|                 value: null, |                 value: null, | ||||||
|                 text: t("labels.arrowhead_none"), |                 text: t("labels.arrowhead_none"), | ||||||
|                 keyBinding: "q", |                 keyBinding: "q", | ||||||
|                 icon: <ArrowheadNoneIcon theme={appState.theme} />, |                 icon: <ArrowheadNoneIcon appearance={appState.appearance} />, | ||||||
|               }, |               }, | ||||||
|               { |               { | ||||||
|                 value: "arrow", |                 value: "arrow", | ||||||
|                 text: t("labels.arrowhead_arrow"), |                 text: t("labels.arrowhead_arrow"), | ||||||
|                 keyBinding: "w", |                 keyBinding: "w", | ||||||
|                 icon: ( |                 icon: ( | ||||||
|                   <ArrowheadArrowIcon theme={appState.theme} flip={isRTL} /> |                   <ArrowheadArrowIcon | ||||||
|  |                     appearance={appState.appearance} | ||||||
|  |                     flip={isRTL} | ||||||
|  |                   /> | ||||||
|                 ), |                 ), | ||||||
|               }, |               }, | ||||||
|               { |               { | ||||||
|                 value: "bar", |                 value: "bar", | ||||||
|                 text: t("labels.arrowhead_bar"), |                 text: t("labels.arrowhead_bar"), | ||||||
|                 keyBinding: "e", |                 keyBinding: "e", | ||||||
|                 icon: <ArrowheadBarIcon theme={appState.theme} flip={isRTL} />, |                 icon: ( | ||||||
|  |                   <ArrowheadBarIcon | ||||||
|  |                     appearance={appState.appearance} | ||||||
|  |                     flip={isRTL} | ||||||
|  |                   /> | ||||||
|  |                 ), | ||||||
|               }, |               }, | ||||||
|               { |               { | ||||||
|                 value: "dot", |                 value: "dot", | ||||||
|                 text: t("labels.arrowhead_dot"), |                 text: t("labels.arrowhead_dot"), | ||||||
|                 keyBinding: "r", |                 keyBinding: "r", | ||||||
|                 icon: <ArrowheadDotIcon theme={appState.theme} flip={isRTL} />, |                 icon: ( | ||||||
|  |                   <ArrowheadDotIcon | ||||||
|  |                     appearance={appState.appearance} | ||||||
|  |                     flip={isRTL} | ||||||
|  |                   /> | ||||||
|  |                 ), | ||||||
|               }, |               }, | ||||||
|             ]} |             ]} | ||||||
|             value={getFormValue<Arrowhead | null>( |             value={getFormValue<Arrowhead | null>( | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import { | |||||||
|   redrawTextBoundingBox, |   redrawTextBoundingBox, | ||||||
| } from "../element"; | } from "../element"; | ||||||
| import { CODES, KEYS } from "../keys"; | import { CODES, KEYS } from "../keys"; | ||||||
| import { t } from "../i18n"; |  | ||||||
| import { register } from "./register"; | import { register } from "./register"; | ||||||
| import { mutateElement, newElementWith } from "../element/mutateElement"; | import { mutateElement, newElementWith } from "../element/mutateElement"; | ||||||
| import { | import { | ||||||
| @@ -24,16 +23,13 @@ export const actionCopyStyles = register({ | |||||||
|       copiedStyles = JSON.stringify(element); |       copiedStyles = JSON.stringify(element); | ||||||
|     } |     } | ||||||
|     return { |     return { | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         toastMessage: t("toast.copyStyles"), |  | ||||||
|       }, |  | ||||||
|       commitToHistory: false, |       commitToHistory: false, | ||||||
|     }; |     }; | ||||||
|   }, |   }, | ||||||
|   contextItemLabel: "labels.copyStyles", |   contextItemLabel: "labels.copyStyles", | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.C, |     event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.C, | ||||||
|  |   contextMenuOrder: 0, | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const actionPasteStyles = register({ | export const actionPasteStyles = register({ | ||||||
| @@ -73,4 +69,5 @@ export const actionPasteStyles = register({ | |||||||
|   contextItemLabel: "labels.pasteStyles", |   contextItemLabel: "labels.pasteStyles", | ||||||
|   keyTest: (event) => |   keyTest: (event) => | ||||||
|     event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.V, |     event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.V, | ||||||
|  |   contextMenuOrder: 1, | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,22 +0,0 @@ | |||||||
| import { CODES, KEYS } from "../keys"; |  | ||||||
| import { register } from "./register"; |  | ||||||
| import { GRID_SIZE } from "../constants"; |  | ||||||
| import { AppState } from "../types"; |  | ||||||
| import { trackEvent } from "../analytics"; |  | ||||||
|  |  | ||||||
| export const actionToggleGridMode = register({ |  | ||||||
|   name: "gridMode", |  | ||||||
|   perform(elements, appState) { |  | ||||||
|     trackEvent("view", "mode", "grid"); |  | ||||||
|     return { |  | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         gridSize: this.checked!(appState) ? null : GRID_SIZE, |  | ||||||
|       }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   checked: (appState: AppState) => appState.gridSize !== null, |  | ||||||
|   contextItemLabel: "labels.showGrid", |  | ||||||
|   keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.code === CODES.QUOTE, |  | ||||||
| }); |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| import { register } from "./register"; |  | ||||||
| import { CODES, KEYS } from "../keys"; |  | ||||||
|  |  | ||||||
| export const actionToggleStats = register({ |  | ||||||
|   name: "stats", |  | ||||||
|   perform(elements, appState) { |  | ||||||
|     return { |  | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         showStats: !this.checked!(appState), |  | ||||||
|       }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   checked: (appState) => appState.showStats, |  | ||||||
|   contextItemLabel: "stats.title", |  | ||||||
|   keyTest: (event) => |  | ||||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.SLASH, |  | ||||||
| }); |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| import { CODES, KEYS } from "../keys"; |  | ||||||
| import { register } from "./register"; |  | ||||||
| import { trackEvent } from "../analytics"; |  | ||||||
|  |  | ||||||
| export const actionToggleViewMode = register({ |  | ||||||
|   name: "viewMode", |  | ||||||
|   perform(elements, appState) { |  | ||||||
|     trackEvent("view", "mode", "view"); |  | ||||||
|     return { |  | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         viewModeEnabled: !this.checked!(appState), |  | ||||||
|       }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   checked: (appState) => appState.viewModeEnabled, |  | ||||||
|   contextItemLabel: "labels.viewMode", |  | ||||||
|   keyTest: (event) => |  | ||||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.R, |  | ||||||
| }); |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| import { CODES, KEYS } from "../keys"; |  | ||||||
| import { register } from "./register"; |  | ||||||
| import { trackEvent } from "../analytics"; |  | ||||||
|  |  | ||||||
| export const actionToggleZenMode = register({ |  | ||||||
|   name: "zenMode", |  | ||||||
|   perform(elements, appState) { |  | ||||||
|     trackEvent("view", "mode", "zen"); |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|       appState: { |  | ||||||
|         ...appState, |  | ||||||
|         zenModeEnabled: !this.checked!(appState), |  | ||||||
|       }, |  | ||||||
|       commitToHistory: false, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|   checked: (appState) => appState.zenModeEnabled, |  | ||||||
|   contextItemLabel: "buttons.zenMode", |  | ||||||
|   keyTest: (event) => |  | ||||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.Z, |  | ||||||
| }); |  | ||||||
| @@ -38,7 +38,7 @@ export const actionSendBackward = register({ | |||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.sendBackward")} — ${getShortcutKey("CtrlOrCmd+[")}`} |       title={`${t("labels.sendBackward")} — ${getShortcutKey("CtrlOrCmd+[")}`} | ||||||
|     > |     > | ||||||
|       <SendBackwardIcon theme={appState.theme} /> |       <SendBackwardIcon appearance={appState.appearance} /> | ||||||
|     </button> |     </button> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
| @@ -65,7 +65,7 @@ export const actionBringForward = register({ | |||||||
|       onClick={() => updateData(null)} |       onClick={() => updateData(null)} | ||||||
|       title={`${t("labels.bringForward")} — ${getShortcutKey("CtrlOrCmd+]")}`} |       title={`${t("labels.bringForward")} — ${getShortcutKey("CtrlOrCmd+]")}`} | ||||||
|     > |     > | ||||||
|       <BringForwardIcon theme={appState.theme} /> |       <BringForwardIcon appearance={appState.appearance} /> | ||||||
|     </button> |     </button> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
| @@ -99,7 +99,7 @@ export const actionSendToBack = register({ | |||||||
|           : getShortcutKey("CtrlOrCmd+Shift+[") |           : getShortcutKey("CtrlOrCmd+Shift+[") | ||||||
|       }`} |       }`} | ||||||
|     > |     > | ||||||
|       <SendToBackIcon theme={appState.theme} /> |       <SendToBackIcon appearance={appState.appearance} /> | ||||||
|     </button> |     </button> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
| @@ -133,7 +133,7 @@ export const actionBringToFront = register({ | |||||||
|           : getShortcutKey("CtrlOrCmd+Shift+]") |           : getShortcutKey("CtrlOrCmd+Shift+]") | ||||||
|       }`} |       }`} | ||||||
|     > |     > | ||||||
|       <BringToFrontIcon theme={appState.theme} /> |       <BringToFrontIcon appearance={appState.appearance} /> | ||||||
|     </button> |     </button> | ||||||
|   ), |   ), | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -26,7 +26,6 @@ export { | |||||||
|   actionZoomOut, |   actionZoomOut, | ||||||
|   actionResetZoom, |   actionResetZoom, | ||||||
|   actionZoomToFit, |   actionZoomToFit, | ||||||
|   actionToggleTheme, |  | ||||||
| } from "./actionCanvas"; | } from "./actionCanvas"; | ||||||
|  |  | ||||||
| export { actionFinalize } from "./actionFinalize"; | export { actionFinalize } from "./actionFinalize"; | ||||||
| @@ -34,8 +33,8 @@ export { actionFinalize } from "./actionFinalize"; | |||||||
| export { | export { | ||||||
|   actionChangeProjectName, |   actionChangeProjectName, | ||||||
|   actionChangeExportBackground, |   actionChangeExportBackground, | ||||||
|   actionSaveToActiveFile, |   actionSaveScene, | ||||||
|   actionSaveFileToDisk, |   actionSaveAsScene, | ||||||
|   actionLoadScene, |   actionLoadScene, | ||||||
| } from "./actionExport"; | } from "./actionExport"; | ||||||
|  |  | ||||||
| @@ -66,17 +65,3 @@ export { | |||||||
|   distributeHorizontally, |   distributeHorizontally, | ||||||
|   distributeVertically, |   distributeVertically, | ||||||
| } from "./actionDistribute"; | } from "./actionDistribute"; | ||||||
|  |  | ||||||
| export { actionFlipHorizontal, actionFlipVertical } from "./actionFlip"; |  | ||||||
|  |  | ||||||
| export { |  | ||||||
|   actionCopy, |  | ||||||
|   actionCut, |  | ||||||
|   actionCopyAsPng, |  | ||||||
|   actionCopyAsSvg, |  | ||||||
| } from "./actionClipboard"; |  | ||||||
|  |  | ||||||
| export { actionToggleGridMode } from "./actionToggleGridMode"; |  | ||||||
| export { actionToggleZenMode } from "./actionToggleZenMode"; |  | ||||||
|  |  | ||||||
| export { actionToggleStats } from "./actionToggleStats"; |  | ||||||
|   | |||||||
| @@ -3,13 +3,14 @@ import { | |||||||
|   Action, |   Action, | ||||||
|   ActionsManagerInterface, |   ActionsManagerInterface, | ||||||
|   UpdaterFn, |   UpdaterFn, | ||||||
|  |   ActionFilterFn, | ||||||
|   ActionName, |   ActionName, | ||||||
|   ActionResult, |   ActionResult, | ||||||
|   PanelComponentProps, |  | ||||||
| } from "./types"; | } from "./types"; | ||||||
| import { ExcalidrawElement } from "../element/types"; | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { AppClassProperties, AppState } from "../types"; | import { AppState } from "../types"; | ||||||
| import { MODES } from "../constants"; | import { t } from "../i18n"; | ||||||
|  | import { ShortcutName } from "./shortcuts"; | ||||||
|  |  | ||||||
| export class ActionManager implements ActionsManagerInterface { | export class ActionManager implements ActionsManagerInterface { | ||||||
|   actions = {} as ActionsManagerInterface["actions"]; |   actions = {} as ActionsManagerInterface["actions"]; | ||||||
| @@ -17,14 +18,13 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|   updater: (actionResult: ActionResult | Promise<ActionResult>) => void; |   updater: (actionResult: ActionResult | Promise<ActionResult>) => void; | ||||||
|  |  | ||||||
|   getAppState: () => Readonly<AppState>; |   getAppState: () => Readonly<AppState>; | ||||||
|  |  | ||||||
|   getElementsIncludingDeleted: () => readonly ExcalidrawElement[]; |   getElementsIncludingDeleted: () => readonly ExcalidrawElement[]; | ||||||
|   app: AppClassProperties; |  | ||||||
|  |  | ||||||
|   constructor( |   constructor( | ||||||
|     updater: UpdaterFn, |     updater: UpdaterFn, | ||||||
|     getAppState: () => AppState, |     getAppState: () => AppState, | ||||||
|     getElementsIncludingDeleted: () => readonly ExcalidrawElement[], |     getElementsIncludingDeleted: () => readonly ExcalidrawElement[], | ||||||
|     app: AppClassProperties, |  | ||||||
|   ) { |   ) { | ||||||
|     this.updater = (actionResult) => { |     this.updater = (actionResult) => { | ||||||
|       if (actionResult && "then" in actionResult) { |       if (actionResult && "then" in actionResult) { | ||||||
| @@ -37,7 +37,6 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|     }; |     }; | ||||||
|     this.getAppState = getAppState; |     this.getAppState = getAppState; | ||||||
|     this.getElementsIncludingDeleted = getElementsIncludingDeleted; |     this.getElementsIncludingDeleted = getElementsIncludingDeleted; | ||||||
|     this.app = app; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   registerAction(action: Action) { |   registerAction(action: Action) { | ||||||
| @@ -48,15 +47,11 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|     actions.forEach((action) => this.registerAction(action)); |     actions.forEach((action) => this.registerAction(action)); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleKeyDown(event: React.KeyboardEvent | KeyboardEvent) { |   handleKeyDown(event: KeyboardEvent) { | ||||||
|     const canvasActions = this.app.props.UIOptions.canvasActions; |  | ||||||
|     const data = Object.values(this.actions) |     const data = Object.values(this.actions) | ||||||
|       .sort((a, b) => (b.keyPriority || 0) - (a.keyPriority || 0)) |       .sort((a, b) => (b.keyPriority || 0) - (a.keyPriority || 0)) | ||||||
|       .filter( |       .filter( | ||||||
|         (action) => |         (action) => | ||||||
|           (action.name in canvasActions |  | ||||||
|             ? canvasActions[action.name as keyof typeof canvasActions] |  | ||||||
|             : true) && |  | ||||||
|           action.keyTest && |           action.keyTest && | ||||||
|           action.keyTest( |           action.keyTest( | ||||||
|             event, |             event, | ||||||
| @@ -68,12 +63,6 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|     if (data.length === 0) { |     if (data.length === 0) { | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|     const { viewModeEnabled } = this.getAppState(); |  | ||||||
|     if (viewModeEnabled) { |  | ||||||
|       if (!Object.values(MODES).includes(data[0].name)) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     event.preventDefault(); |     event.preventDefault(); | ||||||
|     this.updater( |     this.updater( | ||||||
| @@ -81,7 +70,6 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|         this.getElementsIncludingDeleted(), |         this.getElementsIncludingDeleted(), | ||||||
|         this.getAppState(), |         this.getAppState(), | ||||||
|         null, |         null, | ||||||
|         this.app, |  | ||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|     return true; |     return true; | ||||||
| @@ -93,24 +81,49 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|         this.getElementsIncludingDeleted(), |         this.getElementsIncludingDeleted(), | ||||||
|         this.getAppState(), |         this.getAppState(), | ||||||
|         null, |         null, | ||||||
|         this.app, |  | ||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   getContextMenuItems(actionFilter: ActionFilterFn = (action) => action) { | ||||||
|    * @param data additional data sent to the PanelComponent |     return Object.values(this.actions) | ||||||
|    */ |       .filter(actionFilter) | ||||||
|   renderAction = (name: ActionName, data?: PanelComponentProps["data"]) => { |       .filter((action) => "contextItemLabel" in action) | ||||||
|     const canvasActions = this.app.props.UIOptions.canvasActions; |       .filter((action) => | ||||||
|  |         action.contextItemPredicate | ||||||
|  |           ? action.contextItemPredicate( | ||||||
|  |               this.getElementsIncludingDeleted(), | ||||||
|  |               this.getAppState(), | ||||||
|  |             ) | ||||||
|  |           : true, | ||||||
|  |       ) | ||||||
|  |       .sort( | ||||||
|  |         (a, b) => | ||||||
|  |           (a.contextMenuOrder !== undefined ? a.contextMenuOrder : 999) - | ||||||
|  |           (b.contextMenuOrder !== undefined ? b.contextMenuOrder : 999), | ||||||
|  |       ) | ||||||
|  |       .map((action) => ({ | ||||||
|  |         // take last bit of the label  "labels.<shortcutName>" | ||||||
|  |         shortcutName: action.contextItemLabel?.split(".").pop() as ShortcutName, | ||||||
|  |         label: action.contextItemLabel ? t(action.contextItemLabel) : "", | ||||||
|  |         action: () => { | ||||||
|  |           this.updater( | ||||||
|  |             action.perform( | ||||||
|  |               this.getElementsIncludingDeleted(), | ||||||
|  |               this.getAppState(), | ||||||
|  |               null, | ||||||
|  |             ), | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |       })); | ||||||
|  |   } | ||||||
|  |  | ||||||
|     if ( |   // Id is an attribute that we can use to pass in data like keys. | ||||||
|       this.actions[name] && |   // This is needed for dynamically generated action components | ||||||
|       "PanelComponent" in this.actions[name] && |   // like the user list. We can use this key to extract more | ||||||
|       (name in canvasActions |   // data from app state. This is an alternative to generic prop hell! | ||||||
|         ? canvasActions[name as keyof typeof canvasActions] |   renderAction = (name: ActionName, id?: string) => { | ||||||
|         : true) |     if (this.actions[name] && "PanelComponent" in this.actions[name]) { | ||||||
|     ) { |  | ||||||
|       const action = this.actions[name]; |       const action = this.actions[name]; | ||||||
|       const PanelComponent = action.PanelComponent!; |       const PanelComponent = action.PanelComponent!; | ||||||
|       const updateData = (formState?: any) => { |       const updateData = (formState?: any) => { | ||||||
| @@ -119,7 +132,6 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|             this.getElementsIncludingDeleted(), |             this.getElementsIncludingDeleted(), | ||||||
|             this.getAppState(), |             this.getAppState(), | ||||||
|             formState, |             formState, | ||||||
|             this.app, |  | ||||||
|           ), |           ), | ||||||
|         ); |         ); | ||||||
|       }; |       }; | ||||||
| @@ -129,8 +141,7 @@ export class ActionManager implements ActionsManagerInterface { | |||||||
|           elements={this.getElementsIncludingDeleted()} |           elements={this.getElementsIncludingDeleted()} | ||||||
|           appState={this.getAppState()} |           appState={this.getAppState()} | ||||||
|           updateData={updateData} |           updateData={updateData} | ||||||
|           appProps={this.app.props} |           id={id} | ||||||
|           data={data} |  | ||||||
|         /> |         /> | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ export type ShortcutName = | |||||||
|   | "copyStyles" |   | "copyStyles" | ||||||
|   | "pasteStyles" |   | "pasteStyles" | ||||||
|   | "selectAll" |   | "selectAll" | ||||||
|   | "deleteSelectedElements" |   | "delete" | ||||||
|   | "duplicateSelection" |   | "duplicateSelection" | ||||||
|   | "sendBackward" |   | "sendBackward" | ||||||
|   | "bringForward" |   | "bringForward" | ||||||
| @@ -19,13 +19,9 @@ export type ShortcutName = | |||||||
|   | "copyAsSvg" |   | "copyAsSvg" | ||||||
|   | "group" |   | "group" | ||||||
|   | "ungroup" |   | "ungroup" | ||||||
|   | "gridMode" |   | "toggleGridMode" | ||||||
|   | "zenMode" |   | "toggleStats" | ||||||
|   | "stats" |   | "addToLibrary"; | ||||||
|   | "addToLibrary" |  | ||||||
|   | "viewMode" |  | ||||||
|   | "flipHorizontal" |  | ||||||
|   | "flipVertical"; |  | ||||||
|  |  | ||||||
| const shortcutMap: Record<ShortcutName, string[]> = { | const shortcutMap: Record<ShortcutName, string[]> = { | ||||||
|   cut: [getShortcutKey("CtrlOrCmd+X")], |   cut: [getShortcutKey("CtrlOrCmd+X")], | ||||||
| @@ -34,10 +30,10 @@ const shortcutMap: Record<ShortcutName, string[]> = { | |||||||
|   copyStyles: [getShortcutKey("CtrlOrCmd+Alt+C")], |   copyStyles: [getShortcutKey("CtrlOrCmd+Alt+C")], | ||||||
|   pasteStyles: [getShortcutKey("CtrlOrCmd+Alt+V")], |   pasteStyles: [getShortcutKey("CtrlOrCmd+Alt+V")], | ||||||
|   selectAll: [getShortcutKey("CtrlOrCmd+A")], |   selectAll: [getShortcutKey("CtrlOrCmd+A")], | ||||||
|   deleteSelectedElements: [getShortcutKey("Del")], |   delete: [getShortcutKey("Del")], | ||||||
|   duplicateSelection: [ |   duplicateSelection: [ | ||||||
|     getShortcutKey("CtrlOrCmd+D"), |     getShortcutKey("CtrlOrCmd+D"), | ||||||
|     getShortcutKey(`Alt+${t("helpDialog.drag")}`), |     getShortcutKey(`Alt+${t("shortcutsDialog.drag")}`), | ||||||
|   ], |   ], | ||||||
|   sendBackward: [getShortcutKey("CtrlOrCmd+[")], |   sendBackward: [getShortcutKey("CtrlOrCmd+[")], | ||||||
|   bringForward: [getShortcutKey("CtrlOrCmd+]")], |   bringForward: [getShortcutKey("CtrlOrCmd+]")], | ||||||
| @@ -55,13 +51,9 @@ const shortcutMap: Record<ShortcutName, string[]> = { | |||||||
|   copyAsSvg: [], |   copyAsSvg: [], | ||||||
|   group: [getShortcutKey("CtrlOrCmd+G")], |   group: [getShortcutKey("CtrlOrCmd+G")], | ||||||
|   ungroup: [getShortcutKey("CtrlOrCmd+Shift+G")], |   ungroup: [getShortcutKey("CtrlOrCmd+Shift+G")], | ||||||
|   gridMode: [getShortcutKey("CtrlOrCmd+'")], |   toggleGridMode: [getShortcutKey("CtrlOrCmd+'")], | ||||||
|   zenMode: [getShortcutKey("Alt+Z")], |   toggleStats: [], | ||||||
|   stats: [getShortcutKey("Alt+/")], |  | ||||||
|   addToLibrary: [], |   addToLibrary: [], | ||||||
|   flipHorizontal: [getShortcutKey("Shift+H")], |  | ||||||
|   flipVertical: [getShortcutKey("Shift+V")], |  | ||||||
|   viewMode: [getShortcutKey("Alt+R")], |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const getShortcutFromShortcutName = (name: ShortcutName) => { | export const getShortcutFromShortcutName = (name: ShortcutName) => { | ||||||
|   | |||||||
| @@ -1,25 +1,14 @@ | |||||||
| import React from "react"; | import React from "react"; | ||||||
| import { ExcalidrawElement } from "../element/types"; | import { ExcalidrawElement } from "../element/types"; | ||||||
| import { | import { AppState } from "../types"; | ||||||
|   AppClassProperties, |  | ||||||
|   AppState, |  | ||||||
|   ExcalidrawProps, |  | ||||||
|   BinaryFiles, |  | ||||||
| } from "../types"; |  | ||||||
| import { ToolButtonSize } from "../components/ToolButton"; |  | ||||||
|  |  | ||||||
| /** if false, the action should be prevented */ | /** if false, the action should be prevented */ | ||||||
| export type ActionResult = | export type ActionResult = | ||||||
|   | { |   | { | ||||||
|       elements?: readonly ExcalidrawElement[] | null; |       elements?: readonly ExcalidrawElement[] | null; | ||||||
|       appState?: MarkOptional< |       appState?: MarkOptional<AppState, "offsetTop" | "offsetLeft"> | null; | ||||||
|         AppState, |  | ||||||
|         "offsetTop" | "offsetLeft" | "width" | "height" |  | ||||||
|       > | null; |  | ||||||
|       files?: BinaryFiles | null; |  | ||||||
|       commitToHistory: boolean; |       commitToHistory: boolean; | ||||||
|       syncHistory?: boolean; |       syncHistory?: boolean; | ||||||
|       replaceFiles?: boolean; |  | ||||||
|     } |     } | ||||||
|   | false; |   | false; | ||||||
|  |  | ||||||
| @@ -27,18 +16,12 @@ type ActionFn = ( | |||||||
|   elements: readonly ExcalidrawElement[], |   elements: readonly ExcalidrawElement[], | ||||||
|   appState: Readonly<AppState>, |   appState: Readonly<AppState>, | ||||||
|   formData: any, |   formData: any, | ||||||
|   app: AppClassProperties, |  | ||||||
| ) => ActionResult | Promise<ActionResult>; | ) => ActionResult | Promise<ActionResult>; | ||||||
|  |  | ||||||
| export type UpdaterFn = (res: ActionResult) => void; | export type UpdaterFn = (res: ActionResult) => void; | ||||||
| export type ActionFilterFn = (action: Action) => void; | export type ActionFilterFn = (action: Action) => void; | ||||||
|  |  | ||||||
| export type ActionName = | export type ActionName = | ||||||
|   | "copy" |  | ||||||
|   | "cut" |  | ||||||
|   | "paste" |  | ||||||
|   | "copyAsPng" |  | ||||||
|   | "copyAsSvg" |  | ||||||
|   | "sendBackward" |   | "sendBackward" | ||||||
|   | "bringForward" |   | "bringForward" | ||||||
|   | "sendToBack" |   | "sendToBack" | ||||||
| @@ -46,14 +29,10 @@ export type ActionName = | |||||||
|   | "copyStyles" |   | "copyStyles" | ||||||
|   | "selectAll" |   | "selectAll" | ||||||
|   | "pasteStyles" |   | "pasteStyles" | ||||||
|   | "gridMode" |  | ||||||
|   | "zenMode" |  | ||||||
|   | "stats" |  | ||||||
|   | "changeStrokeColor" |   | "changeStrokeColor" | ||||||
|   | "changeBackgroundColor" |   | "changeBackgroundColor" | ||||||
|   | "changeFillStyle" |   | "changeFillStyle" | ||||||
|   | "changeStrokeWidth" |   | "changeStrokeWidth" | ||||||
|   | "changeStrokeShape" |  | ||||||
|   | "changeSloppiness" |   | "changeSloppiness" | ||||||
|   | "changeStrokeStyle" |   | "changeStrokeStyle" | ||||||
|   | "changeArrowhead" |   | "changeArrowhead" | ||||||
| @@ -67,9 +46,9 @@ export type ActionName = | |||||||
|   | "changeProjectName" |   | "changeProjectName" | ||||||
|   | "changeExportBackground" |   | "changeExportBackground" | ||||||
|   | "changeExportEmbedScene" |   | "changeExportEmbedScene" | ||||||
|   | "changeExportScale" |   | "changeShouldAddWatermark" | ||||||
|   | "saveToActiveFile" |   | "saveScene" | ||||||
|   | "saveFileToDisk" |   | "saveAsScene" | ||||||
|   | "loadScene" |   | "loadScene" | ||||||
|   | "duplicateSelection" |   | "duplicateSelection" | ||||||
|   | "deleteSelectedElements" |   | "deleteSelectedElements" | ||||||
| @@ -79,7 +58,6 @@ export type ActionName = | |||||||
|   | "zoomOut" |   | "zoomOut" | ||||||
|   | "resetZoom" |   | "resetZoom" | ||||||
|   | "zoomToFit" |   | "zoomToFit" | ||||||
|   | "zoomToSelection" |  | ||||||
|   | "changeFontFamily" |   | "changeFontFamily" | ||||||
|   | "changeTextAlign" |   | "changeTextAlign" | ||||||
|   | "toggleFullScreen" |   | "toggleFullScreen" | ||||||
| @@ -96,43 +74,37 @@ export type ActionName = | |||||||
|   | "alignVerticallyCentered" |   | "alignVerticallyCentered" | ||||||
|   | "alignHorizontallyCentered" |   | "alignHorizontallyCentered" | ||||||
|   | "distributeHorizontally" |   | "distributeHorizontally" | ||||||
|   | "distributeVertically" |   | "distributeVertically"; | ||||||
|   | "flipHorizontal" |  | ||||||
|   | "flipVertical" |  | ||||||
|   | "viewMode" |  | ||||||
|   | "exportWithDarkMode" |  | ||||||
|   | "toggleTheme"; |  | ||||||
|  |  | ||||||
| export type PanelComponentProps = { |  | ||||||
|   elements: readonly ExcalidrawElement[]; |  | ||||||
|   appState: AppState; |  | ||||||
|   updateData: (formData?: any) => void; |  | ||||||
|   appProps: ExcalidrawProps; |  | ||||||
|   data?: Partial<{ id: string; size: ToolButtonSize }>; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export interface Action { | export interface Action { | ||||||
|   name: ActionName; |   name: ActionName; | ||||||
|   PanelComponent?: React.FC<PanelComponentProps>; |   PanelComponent?: React.FC<{ | ||||||
|  |     elements: readonly ExcalidrawElement[]; | ||||||
|  |     appState: AppState; | ||||||
|  |     updateData: (formData?: any) => void; | ||||||
|  |     id?: string; | ||||||
|  |   }>; | ||||||
|   perform: ActionFn; |   perform: ActionFn; | ||||||
|   keyPriority?: number; |   keyPriority?: number; | ||||||
|   keyTest?: ( |   keyTest?: ( | ||||||
|     event: React.KeyboardEvent | KeyboardEvent, |     event: KeyboardEvent, | ||||||
|     appState: AppState, |     appState: AppState, | ||||||
|     elements: readonly ExcalidrawElement[], |     elements: readonly ExcalidrawElement[], | ||||||
|   ) => boolean; |   ) => boolean; | ||||||
|   contextItemLabel?: string; |   contextItemLabel?: string; | ||||||
|  |   contextMenuOrder?: number; | ||||||
|   contextItemPredicate?: ( |   contextItemPredicate?: ( | ||||||
|     elements: readonly ExcalidrawElement[], |     elements: readonly ExcalidrawElement[], | ||||||
|     appState: AppState, |     appState: AppState, | ||||||
|   ) => boolean; |   ) => boolean; | ||||||
|   checked?: (appState: Readonly<AppState>) => boolean; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| export interface ActionsManagerInterface { | export interface ActionsManagerInterface { | ||||||
|   actions: Record<ActionName, Action>; |   actions: Record<ActionName, Action>; | ||||||
|   registerAction: (action: Action) => void; |   registerAction: (action: Action) => void; | ||||||
|   handleKeyDown: (event: React.KeyboardEvent | KeyboardEvent) => boolean; |   handleKeyDown: (event: KeyboardEvent) => boolean; | ||||||
|  |   getContextMenuItems: ( | ||||||
|  |     actionFilter: ActionFilterFn, | ||||||
|  |   ) => { label: string; action: () => void }[]; | ||||||
|   renderAction: (name: ActionName) => React.ReactElement | null; |   renderAction: (name: ActionName) => React.ReactElement | null; | ||||||
|   executeAction: (action: Action) => void; |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,18 +1,24 @@ | |||||||
| export const trackEvent = | export const EVENT_ACTION = "action"; | ||||||
|   typeof process !== "undefined" && | export const EVENT_ALIGN = "align"; | ||||||
|   process.env?.REACT_APP_GOOGLE_ANALYTICS_ID && | export const EVENT_CHANGE = "change"; | ||||||
|   typeof window !== "undefined" && | export const EVENT_DIALOG = "dialog"; | ||||||
|   window.gtag | export const EVENT_EXIT = "exit"; | ||||||
|     ? (category: string, name: string, label?: string, value?: number) => { | export const EVENT_IO = "io"; | ||||||
|         window.gtag("event", name, { | export const EVENT_LAYER = "layer"; | ||||||
|           event_category: category, | export const EVENT_LIBRARY = "library"; | ||||||
|           event_label: label, | export const EVENT_LOAD = "load"; | ||||||
|           value, | export const EVENT_SHAPE = "shape"; | ||||||
|         }); | export const EVENT_SHARE = "share"; | ||||||
|       } | export const EVENT_MAGIC = "magic"; | ||||||
|     : typeof process !== "undefined" && process.env?.JEST_WORKER_ID |  | ||||||
|     ? (category: string, name: string, label?: string, value?: number) => {} | export const trackEvent = window.gtag | ||||||
|     : (category: string, name: string, label?: string, value?: number) => { |   ? (category: string, name: string, label?: string, value?: number) => { | ||||||
|         // Uncomment the next line to track locally |       window.gtag("event", name, { | ||||||
|         // console.info("Track Event", category, name, label, value); |         event_category: category, | ||||||
|       }; |         event_label: label, | ||||||
|  |         value, | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   : (category: string, name: string, label?: string, value?: number) => { | ||||||
|  |       console.info("Track Event", category, name, label, value); | ||||||
|  |     }; | ||||||
|   | |||||||
							
								
								
									
										280
									
								
								src/appState.ts
									
									
									
									
									
								
							
							
						
						| @@ -1,85 +1,79 @@ | |||||||
| import oc from "open-color"; | import oc from "open-color"; | ||||||
| import { | import { AppState, FlooredNumber, NormalizedZoomValue } from "./types"; | ||||||
|   DEFAULT_FONT_FAMILY, |  | ||||||
|   DEFAULT_FONT_SIZE, |  | ||||||
|   DEFAULT_TEXT_ALIGN, |  | ||||||
|   EXPORT_SCALES, |  | ||||||
|   THEME, |  | ||||||
| } from "./constants"; |  | ||||||
| import { t } from "./i18n"; |  | ||||||
| import { AppState, NormalizedZoomValue } from "./types"; |  | ||||||
| import { getDateTime } from "./utils"; | import { getDateTime } from "./utils"; | ||||||
|  | import { t } from "./i18n"; | ||||||
| const defaultExportScale = EXPORT_SCALES.includes(devicePixelRatio) | import { | ||||||
|   ? devicePixelRatio |   DEFAULT_FONT_SIZE, | ||||||
|   : 1; |   DEFAULT_FONT_FAMILY, | ||||||
|  |   DEFAULT_TEXT_ALIGN, | ||||||
|  | } from "./constants"; | ||||||
|  |  | ||||||
| export const getDefaultAppState = (): Omit< | export const getDefaultAppState = (): Omit< | ||||||
|   AppState, |   AppState, | ||||||
|   "offsetTop" | "offsetLeft" | "width" | "height" |   "offsetTop" | "offsetLeft" | ||||||
| > => { | > => { | ||||||
|   return { |   return { | ||||||
|     theme: THEME.LIGHT, |     appearance: "light", | ||||||
|     collaborators: new Map(), |  | ||||||
|     currentChartType: "bar", |  | ||||||
|     currentItemBackgroundColor: "transparent", |  | ||||||
|     currentItemEndArrowhead: "arrow", |  | ||||||
|     currentItemFillStyle: "hachure", |  | ||||||
|     currentItemFontFamily: DEFAULT_FONT_FAMILY, |  | ||||||
|     currentItemFontSize: DEFAULT_FONT_SIZE, |  | ||||||
|     currentItemLinearStrokeSharpness: "round", |  | ||||||
|     currentItemOpacity: 100, |  | ||||||
|     currentItemRoughness: 1, |  | ||||||
|     currentItemStartArrowhead: null, |  | ||||||
|     currentItemStrokeColor: oc.black, |  | ||||||
|     currentItemStrokeSharpness: "sharp", |  | ||||||
|     currentItemStrokeStyle: "solid", |  | ||||||
|     currentItemStrokeWidth: 1, |  | ||||||
|     currentItemTextAlign: DEFAULT_TEXT_ALIGN, |  | ||||||
|     cursorButton: "up", |  | ||||||
|     draggingElement: null, |  | ||||||
|     editingElement: null, |  | ||||||
|     editingGroupId: null, |  | ||||||
|     editingLinearElement: null, |  | ||||||
|     elementLocked: false, |  | ||||||
|     elementType: "selection", |  | ||||||
|     errorMessage: null, |  | ||||||
|     exportBackground: true, |  | ||||||
|     exportScale: defaultExportScale, |  | ||||||
|     exportEmbedScene: false, |  | ||||||
|     exportWithDarkMode: false, |  | ||||||
|     fileHandle: null, |  | ||||||
|     gridSize: null, |  | ||||||
|     isBindingEnabled: true, |  | ||||||
|     isLibraryOpen: false, |  | ||||||
|     isLoading: false, |     isLoading: false, | ||||||
|  |     errorMessage: null, | ||||||
|  |     draggingElement: null, | ||||||
|  |     resizingElement: null, | ||||||
|  |     multiElement: null, | ||||||
|  |     editingElement: null, | ||||||
|  |     startBoundElement: null, | ||||||
|  |     editingLinearElement: null, | ||||||
|  |     elementType: "selection", | ||||||
|  |     elementLocked: false, | ||||||
|  |     exportBackground: true, | ||||||
|  |     exportEmbedScene: false, | ||||||
|  |     shouldAddWatermark: false, | ||||||
|  |     currentItemStrokeColor: oc.black, | ||||||
|  |     currentItemBackgroundColor: "transparent", | ||||||
|  |     currentItemFillStyle: "hachure", | ||||||
|  |     currentItemStrokeWidth: 1, | ||||||
|  |     currentItemStrokeStyle: "solid", | ||||||
|  |     currentItemRoughness: 1, | ||||||
|  |     currentItemOpacity: 100, | ||||||
|  |     currentItemFontSize: DEFAULT_FONT_SIZE, | ||||||
|  |     currentItemFontFamily: DEFAULT_FONT_FAMILY, | ||||||
|  |     currentItemTextAlign: DEFAULT_TEXT_ALIGN, | ||||||
|  |     currentItemStrokeSharpness: "sharp", | ||||||
|  |     currentItemLinearStrokeSharpness: "round", | ||||||
|  |     currentItemStartArrowhead: null, | ||||||
|  |     currentItemEndArrowhead: "arrow", | ||||||
|  |     viewBackgroundColor: oc.white, | ||||||
|  |     scrollX: 0 as FlooredNumber, | ||||||
|  |     scrollY: 0 as FlooredNumber, | ||||||
|  |     cursorX: 0, | ||||||
|  |     cursorY: 0, | ||||||
|  |     cursorButton: "up", | ||||||
|  |     scrolledOutside: false, | ||||||
|  |     name: `${t("labels.untitled")}-${getDateTime()}`, | ||||||
|  |     isBindingEnabled: true, | ||||||
|     isResizing: false, |     isResizing: false, | ||||||
|     isRotating: false, |     isRotating: false, | ||||||
|     lastPointerDownWith: "mouse", |  | ||||||
|     multiElement: null, |  | ||||||
|     name: `${t("labels.untitled")}-${getDateTime()}`, |  | ||||||
|     openMenu: null, |  | ||||||
|     openPopup: null, |  | ||||||
|     pasteDialog: { shown: false, data: null }, |  | ||||||
|     previousSelectedElementIds: {}, |  | ||||||
|     resizingElement: null, |  | ||||||
|     scrolledOutside: false, |  | ||||||
|     scrollX: 0, |  | ||||||
|     scrollY: 0, |  | ||||||
|     selectedElementIds: {}, |  | ||||||
|     selectedGroupIds: {}, |  | ||||||
|     selectionElement: null, |     selectionElement: null, | ||||||
|  |     zoom: { | ||||||
|  |       value: 1 as NormalizedZoomValue, | ||||||
|  |       translation: { x: 0, y: 0 }, | ||||||
|  |     }, | ||||||
|  |     openMenu: null, | ||||||
|  |     lastPointerDownWith: "mouse", | ||||||
|  |     selectedElementIds: {}, | ||||||
|  |     previousSelectedElementIds: {}, | ||||||
|     shouldCacheIgnoreZoom: false, |     shouldCacheIgnoreZoom: false, | ||||||
|     showHelpDialog: false, |     showShortcutsDialog: false, | ||||||
|     showStats: false, |  | ||||||
|     startBoundElement: null, |  | ||||||
|     suggestedBindings: [], |     suggestedBindings: [], | ||||||
|     toastMessage: null, |  | ||||||
|     viewBackgroundColor: oc.white, |  | ||||||
|     zenModeEnabled: false, |     zenModeEnabled: false, | ||||||
|     zoom: { value: 1 as NormalizedZoomValue, translation: { x: 0, y: 0 } }, |     gridSize: null, | ||||||
|     viewModeEnabled: false, |     editingGroupId: null, | ||||||
|     pendingImageElement: null, |     selectedGroupIds: {}, | ||||||
|  |     width: window.innerWidth, | ||||||
|  |     height: window.innerHeight, | ||||||
|  |     isLibraryOpen: false, | ||||||
|  |     fileHandle: null, | ||||||
|  |     collaborators: new Map(), | ||||||
|  |     showStats: false, | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -93,87 +87,74 @@ const APP_STATE_STORAGE_CONF = (< | |||||||
|     browser: boolean; |     browser: boolean; | ||||||
|     /** whether to keep when exporting to file/database */ |     /** whether to keep when exporting to file/database */ | ||||||
|     export: boolean; |     export: boolean; | ||||||
|     /** server (shareLink/collab/...) */ |  | ||||||
|     server: boolean; |  | ||||||
|   }, |   }, | ||||||
|   T extends Record<keyof AppState, Values> |   T extends Record<keyof AppState, Values> | ||||||
| >( | >( | ||||||
|   config: { [K in keyof T]: K extends keyof AppState ? T[K] : never }, |   config: { [K in keyof T]: K extends keyof AppState ? T[K] : never }, | ||||||
| ) => config)({ | ) => config)({ | ||||||
|   theme: { browser: true, export: false, server: false }, |   appearance: { browser: true, export: false }, | ||||||
|   collaborators: { browser: false, export: false, server: false }, |   currentItemBackgroundColor: { browser: true, export: false }, | ||||||
|   currentChartType: { browser: true, export: false, server: false }, |   currentItemFillStyle: { browser: true, export: false }, | ||||||
|   currentItemBackgroundColor: { browser: true, export: false, server: false }, |   currentItemFontFamily: { browser: true, export: false }, | ||||||
|   currentItemEndArrowhead: { browser: true, export: false, server: false }, |   currentItemFontSize: { browser: true, export: false }, | ||||||
|   currentItemFillStyle: { browser: true, export: false, server: false }, |   currentItemOpacity: { browser: true, export: false }, | ||||||
|   currentItemFontFamily: { browser: true, export: false, server: false }, |   currentItemRoughness: { browser: true, export: false }, | ||||||
|   currentItemFontSize: { browser: true, export: false, server: false }, |   currentItemStrokeColor: { browser: true, export: false }, | ||||||
|   currentItemLinearStrokeSharpness: { |   currentItemStrokeStyle: { browser: true, export: false }, | ||||||
|     browser: true, |   currentItemStrokeWidth: { browser: true, export: false }, | ||||||
|     export: false, |   currentItemTextAlign: { browser: true, export: false }, | ||||||
|     server: false, |   currentItemStrokeSharpness: { browser: true, export: false }, | ||||||
|   }, |   currentItemLinearStrokeSharpness: { browser: true, export: false }, | ||||||
|   currentItemOpacity: { browser: true, export: false, server: false }, |   currentItemStartArrowhead: { browser: true, export: false }, | ||||||
|   currentItemRoughness: { browser: true, export: false, server: false }, |   currentItemEndArrowhead: { browser: true, export: false }, | ||||||
|   currentItemStartArrowhead: { browser: true, export: false, server: false }, |   cursorButton: { browser: true, export: false }, | ||||||
|   currentItemStrokeColor: { browser: true, export: false, server: false }, |   cursorX: { browser: true, export: false }, | ||||||
|   currentItemStrokeSharpness: { browser: true, export: false, server: false }, |   cursorY: { browser: true, export: false }, | ||||||
|   currentItemStrokeStyle: { browser: true, export: false, server: false }, |   draggingElement: { browser: false, export: false }, | ||||||
|   currentItemStrokeWidth: { browser: true, export: false, server: false }, |   editingElement: { browser: false, export: false }, | ||||||
|   currentItemTextAlign: { browser: true, export: false, server: false }, |   startBoundElement: { browser: false, export: false }, | ||||||
|   cursorButton: { browser: true, export: false, server: false }, |   editingGroupId: { browser: true, export: false }, | ||||||
|   draggingElement: { browser: false, export: false, server: false }, |   editingLinearElement: { browser: false, export: false }, | ||||||
|   editingElement: { browser: false, export: false, server: false }, |   elementLocked: { browser: true, export: false }, | ||||||
|   editingGroupId: { browser: true, export: false, server: false }, |   elementType: { browser: true, export: false }, | ||||||
|   editingLinearElement: { browser: false, export: false, server: false }, |   errorMessage: { browser: false, export: false }, | ||||||
|   elementLocked: { browser: true, export: false, server: false }, |   exportBackground: { browser: true, export: false }, | ||||||
|   elementType: { browser: true, export: false, server: false }, |   exportEmbedScene: { browser: true, export: false }, | ||||||
|   errorMessage: { browser: false, export: false, server: false }, |   gridSize: { browser: true, export: true }, | ||||||
|   exportBackground: { browser: true, export: false, server: false }, |   height: { browser: false, export: false }, | ||||||
|   exportEmbedScene: { browser: true, export: false, server: false }, |   isBindingEnabled: { browser: false, export: false }, | ||||||
|   exportScale: { browser: true, export: false, server: false }, |   isLibraryOpen: { browser: false, export: false }, | ||||||
|   exportWithDarkMode: { browser: true, export: false, server: false }, |   isLoading: { browser: false, export: false }, | ||||||
|   fileHandle: { browser: false, export: false, server: false }, |   isResizing: { browser: false, export: false }, | ||||||
|   gridSize: { browser: true, export: true, server: true }, |   isRotating: { browser: false, export: false }, | ||||||
|   height: { browser: false, export: false, server: false }, |   lastPointerDownWith: { browser: true, export: false }, | ||||||
|   isBindingEnabled: { browser: false, export: false, server: false }, |   multiElement: { browser: false, export: false }, | ||||||
|   isLibraryOpen: { browser: false, export: false, server: false }, |   name: { browser: true, export: false }, | ||||||
|   isLoading: { browser: false, export: false, server: false }, |   openMenu: { browser: true, export: false }, | ||||||
|   isResizing: { browser: false, export: false, server: false }, |   previousSelectedElementIds: { browser: true, export: false }, | ||||||
|   isRotating: { browser: false, export: false, server: false }, |   resizingElement: { browser: false, export: false }, | ||||||
|   lastPointerDownWith: { browser: true, export: false, server: false }, |   scrolledOutside: { browser: true, export: false }, | ||||||
|   multiElement: { browser: false, export: false, server: false }, |   scrollX: { browser: true, export: false }, | ||||||
|   name: { browser: true, export: false, server: false }, |   scrollY: { browser: true, export: false }, | ||||||
|   offsetLeft: { browser: false, export: false, server: false }, |   selectedElementIds: { browser: true, export: false }, | ||||||
|   offsetTop: { browser: false, export: false, server: false }, |   selectedGroupIds: { browser: true, export: false }, | ||||||
|   openMenu: { browser: true, export: false, server: false }, |   selectionElement: { browser: false, export: false }, | ||||||
|   openPopup: { browser: false, export: false, server: false }, |   shouldAddWatermark: { browser: true, export: false }, | ||||||
|   pasteDialog: { browser: false, export: false, server: false }, |   shouldCacheIgnoreZoom: { browser: true, export: false }, | ||||||
|   previousSelectedElementIds: { browser: true, export: false, server: false }, |   showShortcutsDialog: { browser: false, export: false }, | ||||||
|   resizingElement: { browser: false, export: false, server: false }, |   suggestedBindings: { browser: false, export: false }, | ||||||
|   scrolledOutside: { browser: true, export: false, server: false }, |   viewBackgroundColor: { browser: true, export: true }, | ||||||
|   scrollX: { browser: true, export: false, server: false }, |   width: { browser: false, export: false }, | ||||||
|   scrollY: { browser: true, export: false, server: false }, |   zenModeEnabled: { browser: true, export: false }, | ||||||
|   selectedElementIds: { browser: true, export: false, server: false }, |   zoom: { browser: true, export: false }, | ||||||
|   selectedGroupIds: { browser: true, export: false, server: false }, |   offsetTop: { browser: false, export: false }, | ||||||
|   selectionElement: { browser: false, export: false, server: false }, |   offsetLeft: { browser: false, export: false }, | ||||||
|   shouldCacheIgnoreZoom: { browser: true, export: false, server: false }, |   fileHandle: { browser: false, export: false }, | ||||||
|   showHelpDialog: { browser: false, export: false, server: false }, |   collaborators: { browser: false, export: false }, | ||||||
|   showStats: { browser: true, export: false, server: false }, |   showStats: { browser: true, export: false }, | ||||||
|   startBoundElement: { browser: false, export: false, server: false }, |  | ||||||
|   suggestedBindings: { browser: false, export: false, server: false }, |  | ||||||
|   toastMessage: { browser: false, export: false, server: false }, |  | ||||||
|   viewBackgroundColor: { browser: true, export: true, server: true }, |  | ||||||
|   width: { browser: false, export: false, server: false }, |  | ||||||
|   zenModeEnabled: { browser: true, export: false, server: false }, |  | ||||||
|   zoom: { browser: true, export: false, server: false }, |  | ||||||
|   viewModeEnabled: { browser: false, export: false, server: false }, |  | ||||||
|   pendingImageElement: { browser: false, export: false, server: false }, |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| const _clearAppStateForStorage = < | const _clearAppStateForStorage = <ExportType extends "export" | "browser">( | ||||||
|   ExportType extends "export" | "browser" | "server" |  | ||||||
| >( |  | ||||||
|   appState: Partial<AppState>, |   appState: Partial<AppState>, | ||||||
|   exportType: ExportType, |   exportType: ExportType, | ||||||
| ) => { | ) => { | ||||||
| @@ -185,11 +166,14 @@ const _clearAppStateForStorage = < | |||||||
|   const stateForExport = {} as { [K in ExportableKeys]?: typeof appState[K] }; |   const stateForExport = {} as { [K in ExportableKeys]?: typeof appState[K] }; | ||||||
|   for (const key of Object.keys(appState) as (keyof typeof appState)[]) { |   for (const key of Object.keys(appState) as (keyof typeof appState)[]) { | ||||||
|     const propConfig = APP_STATE_STORAGE_CONF[key]; |     const propConfig = APP_STATE_STORAGE_CONF[key]; | ||||||
|  |     if (!propConfig) { | ||||||
|  |       console.error( | ||||||
|  |         `_clearAppStateForStorage: appState key "${key}" config doesn't exist for "${exportType}" export type`, | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|     if (propConfig?.[exportType]) { |     if (propConfig?.[exportType]) { | ||||||
|       const nextValue = appState[key]; |       // @ts-ignore see https://github.com/microsoft/TypeScript/issues/31445 | ||||||
|  |       stateForExport[key] = appState[key]; | ||||||
|       // https://github.com/microsoft/TypeScript/issues/31445 |  | ||||||
|       (stateForExport as any)[key] = nextValue; |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   return stateForExport; |   return stateForExport; | ||||||
| @@ -202,7 +186,3 @@ export const clearAppStateForLocalStorage = (appState: Partial<AppState>) => { | |||||||
| export const cleanAppStateForExport = (appState: Partial<AppState>) => { | export const cleanAppStateForExport = (appState: Partial<AppState>) => { | ||||||
|   return _clearAppStateForStorage(appState, "export"); |   return _clearAppStateForStorage(appState, "export"); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const clearAppStateForDatabase = (appState: Partial<AppState>) => { |  | ||||||
|   return _clearAppStateForStorage(appState, "server"); |  | ||||||
| }; |  | ||||||
|   | |||||||