mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-26 08:24:20 +01:00 
			
		
		
		
	Compare commits
	
		
			245 Commits
		
	
	
		
			cleanipp
			...
			updatescen
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | eb206cc932 | ||
|   | 16c287c848 | ||
|   | 78024873e5 | ||
|   | 4e41bd9dbb | ||
|   | edc23b854f | ||
|   | de99484a1f | ||
|   | 4843c49556 | ||
|   | d565413082 | ||
|   | dcda7184d0 | ||
|   | add1785ace | ||
|   | 8d413670c8 | ||
|   | f774452124 | ||
|   | 0e3eb3cc63 | ||
|   | db4ed1ecb1 | ||
|   | 489f45b910 | ||
|   | a17be085b0 | ||
|   | 4e07a608d3 | ||
|   | c1379c3c10 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 70c36e196c | ||
|   | 9757f8e2e5 | ||
|   | 052b73d95b | ||
|   | e90e56452f | ||
|   | edc62c550a | ||
|   | 84a1863233 | ||
|   | 1f295955d0 | ||
|   | 8f45aa2924 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b4c0bc6ace | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8fdc25ce10 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | efc2db3af1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fa186c69cb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f2adda46e5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0786b472cc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 15ae64eb7a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f3b9bd7c1d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eae25329f0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 99de8e79e2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | adcfaf34de | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 04ae836874 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f6ffa7ed45 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2964d36d9a | ||
|   | b9e70ec666 | ||
|   | f1daff2437 | ||
|   | 91c8b6ecbf | ||
|   | 47c26cd4cf | ||
|   | 3780a155f4 | ||
|   | 6252b22b42 | ||
|   | bb612fd768 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1bb3d71d22 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3912e87b1d | ||
|   | 72f1134878 | ||
|   | 6e629383ea | ||
|   | 612e71e38b | ||
|   | 00ac804dd8 | ||
|   | 1f7f07fbfb | ||
|   | beffc290fd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9b58efd363 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 23b672484b | ||
|   | 3a0a638a0d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 862c065e33 | ||
|   | 420703ba50 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4d6ef81c83 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 88b980a8e0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d85c1955b4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a27e599798 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d25ed7bf88 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 28601b88bd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1edfcc08d8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 030814c254 | ||
|   | eb24e8ffe4 | ||
|   | 07e71a8071 | ||
|   | bf97414530 | ||
|   | f8c3c431da | ||
|   | ef792ad906 | ||
|   | 06a945aded | ||
|   | 0d2f72f87e | ||
|   | 3d1cbf444d | ||
|   | c77c9ce65a | ||
|   | f295ba98c5 | ||
|   | 91eb8834e8 | ||
|   | 418589e7ad | ||
|   | bd63dcbcc5 | ||
|   | 785a944ac2 | ||
|   | 32acde500e | ||
|   | 053353841a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 286642ffcf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7090938ec1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6233bc52e3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7f8ee06710 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3e200634e0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 701b02d6df | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 535b2b682e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a4a736d94 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bb207ef861 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2fcbf8658f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6a8c6e7f47 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f962503425 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 02fdc506ee | ||
|   | 2ba6088e97 | ||
|   | 4e421e6e9e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d213dbb42d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a5779dd5d8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 628b4c1eec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 31333e597b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 872b340f1b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 83e8167adf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | baea88942c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae35037a3b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4f31ae1e4b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 704ee30ae6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 176baef9c2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3558f07fe0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fd24c74ab1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae6892501d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1e6adaf0b5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7ccd38a37f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e6ce9e0ea7 | ||
|   | c0e05445b1 | ||
|   | b44531d94a | ||
|   | 464c2cc05e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd13c2ed48 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2e58aaae66 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d4c14d484c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 06d1871640 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e7a59335e4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0dbef18044 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e16c2d592f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f4cf4610f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1bee959660 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f609a4ac3a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eee9d1bc16 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ecdc2582e9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 245e13a884 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8a322b22ed | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3f784d76fc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ba5afe9139 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9e6f351672 | ||
|   | fcd10a6a43 | ||
|   | 3bc18f6aed | ||
|   | 7c5481b877 | ||
|   | d17464fbaa | ||
|   | baf9da2b83 | ||
|   | 4bfcf105a5 | ||
|   | 74e82d0d7c | ||
|   | 9dd2257932 | ||
|   | 9c0f832a41 | ||
|   | 6cafb6bb90 | ||
|   | e6cd97c4f2 | ||
|   | ba9b65b051 | ||
|   | 830fb64a25 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5b343a9d46 | ||
|   | f8beb305de | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4b4eecbd27 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e8bd910b9b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9a3e1bca5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2a922dd477 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 07dab85ebf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d2ce4a7523 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dc73f3a9eb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d100f38750 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 503500cc74 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a828a43d9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d88884466b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d3d470ac3d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 54df521a78 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c5557b5cc1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 51875fd627 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b2ba61bbcf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3b7f62c9a0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aeafb81479 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dfe81bf6b2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1d332d597a | ||
|   | b5fc8757a4 | ||
|   | ecbd5ba55d | ||
|   | 6967d8c985 | ||
|   | 4b253c7362 | ||
|   | 4fdddb518a | ||
|   | 0b2e4dd60b | ||
|   | f162512988 | ||
|   | 2f7154cdf2 | ||
|   | 5ab0ce5a33 | ||
|   | 073f4032f3 | ||
|   | 73cba59d2d | ||
|   | 4085071347 | ||
|   | 8a63187d4f | ||
|   | 9c51ba6067 | ||
|   | bf50c9cae7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1801048763 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 414deea084 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 979d28d5c6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f8b7e417f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8c968cd13e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f798000006 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7ff3a71179 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6c0804d4c3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae8e7aca16 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 842b185aa6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f59387471e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 60eb709eb3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 02539bbb89 | ||
|   | 77ae5d4605 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bdd4f69bf6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 87ca829490 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a31a7fd766 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e70f02063f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 769f727bd4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bc994fcbe2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ac5e058222 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d61970cdac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 489d4b7469 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d88de08872 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 42882e2a93 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f6374e5bde | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aac9d4e837 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d9103b8b24 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8b56346011 | ||
|   | e63a0ec5be | ||
|   | 86222662f2 | ||
|   | 066560311b | ||
|   | d6ca981f7a | ||
|   | f7f98d9dda | ||
|   | 1a67642fd1 | ||
|   | 6aa22bada8 | ||
|   | 00209ef9c3 | ||
|   | b79ef0d428 | ||
|   | dc25fe06d0 | ||
|   | 1e17c1967b | ||
|   | 23a8891e0e | ||
|   | 6c81a32d62 | ||
|   | f0f5430313 | ||
|   | e18e945cd3 | ||
|   | bd0c6e63ff | ||
|   | dbae33e4f8 | ||
|   | 0f4a053759 | ||
|   | 1837147c55 | ||
|   | 15f698dc21 | ||
|   | ce507b0a0b | ||
|   | 02598c6163 | ||
|   | 3e2890bd21 | ||
|   | 210649f383 | ||
|   | f8087e01c8 | ||
|   | e2522645f7 | ||
|   | d46a9166be | ||
|   | 675da16ca4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2b1b62d8f2 | ||
|   | 627c56ef1c | 
| @@ -1,10 +1,10 @@ | ||||
| * | ||||
| !.env | ||||
| !.eslintrc.json | ||||
| !.npmrc | ||||
| !.prettierrc | ||||
| !package.json | ||||
| !public/ | ||||
| !src/ | ||||
| !.npmrc | ||||
| !.eslintrc.json | ||||
| !.prettierrc | ||||
| !package-lock.json | ||||
| !package.json | ||||
| !tsconfig.json | ||||
| !.env | ||||
| !yarn.lock | ||||
|   | ||||
							
								
								
									
										12
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| # http://EditorConfig.org | ||||
|  | ||||
| # top-level EditorConfig file | ||||
| root = true | ||||
|  | ||||
| [*] | ||||
| charset = utf-8 | ||||
| end_of_line = lf | ||||
| indent_size = 2 | ||||
| indent_style = space | ||||
| insert_final_newline = true | ||||
| trim_trailing_whitespace = true | ||||
| @@ -4,3 +4,4 @@ package-lock.json | ||||
| .vscode/ | ||||
| firebase/ | ||||
| dist/ | ||||
| public/workbox | ||||
|   | ||||
| @@ -1,40 +1,6 @@ | ||||
| { | ||||
|   "extends": ["prettier", "react-app"], | ||||
|   "plugins": ["prettier"], | ||||
|   "extends": ["@excalidraw/eslint-config", "react-app"], | ||||
|   "rules": { | ||||
|     "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" | ||||
|     "import/no-anonymous-default-export": "off" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/assets/crowdin.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.github/assets/crowdin.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| <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> | ||||
| After Width: | Height: | Size: 4.1 KiB | 
							
								
								
									
										9
									
								
								.github/assets/sentry.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.github/assets/sentry.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| <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> | ||||
| After Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										3
									
								
								.github/assets/vercel.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.github/assets/vercel.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| <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> | ||||
| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										9
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,36 +1,33 @@ | ||||
| version: 2 | ||||
| updates: | ||||
|   - package-ecosystem: npm | ||||
|     directory: "/" | ||||
|     directory: / | ||||
|     schedule: | ||||
|       interval: weekly | ||||
|       day: sunday | ||||
|       time: "01:00" | ||||
|     open-pull-requests-limit: 99 | ||||
|     reviewers: | ||||
|       - lipis | ||||
|     assignees: | ||||
|       - lipis | ||||
|  | ||||
|   - package-ecosystem: npm | ||||
|     directory: "/src/packages/excalidraw/" | ||||
|     directory: /src/packages/excalidraw/ | ||||
|     schedule: | ||||
|       interval: weekly | ||||
|       day: sunday | ||||
|       time: "01:00" | ||||
|     open-pull-requests-limit: 99 | ||||
|     reviewers: | ||||
|       - ad1992 | ||||
|     assignees: | ||||
|       - ad1992 | ||||
|  | ||||
|   - package-ecosystem: npm | ||||
|     directory: "/src/packages/utils/" | ||||
|     directory: /src/packages/utils/ | ||||
|     schedule: | ||||
|       interval: weekly | ||||
|       day: sunday | ||||
|       time: "01:00" | ||||
|     open-pull-requests-limit: 99 | ||||
|     reviewers: | ||||
|       - ad1992 | ||||
|     assignees: | ||||
|   | ||||
							
								
								
									
										5
									
								
								.github/workflows/build-docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/build-docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,9 +6,8 @@ on: | ||||
|       - master | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|   build-docker: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v1 | ||||
|       - uses: actions/checkout@v2 | ||||
|       - run: docker build -t excalidraw . | ||||
|   | ||||
							
								
								
									
										20
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,27 +7,23 @@ on: | ||||
|   pull_request: | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|   packages: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v1 | ||||
|  | ||||
|       - uses: actions/checkout@v2 | ||||
|       - name: Setup Node.js 14.x | ||||
|         uses: actions/setup-node@v1 | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 14.x | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: | | ||||
|           npm ci | ||||
|           npm ci --prefix src/packages/excalidraw | ||||
|           npm ci --prefix src/packages/utils | ||||
|  | ||||
|           yarn --frozen-lockfile | ||||
|           yarn --cwd src/packages/excalidraw | ||||
|           yarn --cwd src/packages/utils | ||||
|       - name: Build @excalidraw/excalidraw | ||||
|         run: | | ||||
|           npm run pack --prefix src/packages/excalidraw | ||||
|  | ||||
|           yarn --cwd src/packages/excalidraw run pack | ||||
|       - name: Build @excalidraw/utils | ||||
|         run: | | ||||
|           npm run pack --prefix src/packages/utils | ||||
|           yarn --cwd src/packages/utils run pack | ||||
|   | ||||
							
								
								
									
										7
									
								
								.github/workflows/cancel.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/cancel.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,14 @@ | ||||
| name: Cancel previous runs | ||||
|  | ||||
| on: push | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|   pull_request: | ||||
|  | ||||
| jobs: | ||||
|   cancel: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     timeout-minutes: 3 | ||||
|     steps: | ||||
|       - uses: styfle/cancel-workflow-action@0.6.0 | ||||
|   | ||||
							
								
								
									
										14
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,22 +1,22 @@ | ||||
| name: Lint | ||||
|  | ||||
| on: push | ||||
| on: pull_request | ||||
|  | ||||
| jobs: | ||||
|   lint: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v1 | ||||
|       - uses: actions/checkout@v2 | ||||
|  | ||||
|       - name: Setup Node.js 14.x | ||||
|         uses: actions/setup-node@v1 | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 14.x | ||||
|  | ||||
|       - name: Install and lint | ||||
|         run: | | ||||
|           npm ci | ||||
|           npm run test:other | ||||
|           npm run test:code | ||||
|           npm run test:typecheck | ||||
|           yarn --frozen-lockfile | ||||
|           yarn test:other | ||||
|           yarn test:code | ||||
|           yarn test:typecheck | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/locales-coverage.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/locales-coverage.yml
									
									
									
									
										vendored
									
									
								
							| @@ -3,7 +3,7 @@ name: Build locales coverage | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - "l10n_master" | ||||
|       - l10n_master | ||||
|  | ||||
| jobs: | ||||
|   locales: | ||||
| @@ -15,13 +15,13 @@ jobs: | ||||
|           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} | ||||
|  | ||||
|       - name: Setup Node.js 14.x | ||||
|         uses: actions/setup-node@v1 | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 14.x | ||||
|  | ||||
|       - name: Create report file | ||||
|         run: | | ||||
|           npm run locales-coverage | ||||
|           yarn locales-coverage | ||||
|           FILE_CHANGED=$(git diff src/locales/percentages.json) | ||||
|           if [ ! -z "${FILE_CHANGED}" ]; then | ||||
|             git config --global user.name 'Excalidraw Bot' | ||||
|   | ||||
							
								
								
									
										5
									
								
								.github/workflows/semantic-pr-title.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/semantic-pr-title.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| name: "Semantic PR title" | ||||
| name: Semantic PR title | ||||
|  | ||||
| on: | ||||
|   pull_request_target: | ||||
| @@ -8,9 +8,8 @@ on: | ||||
|       - synchronize | ||||
|  | ||||
| jobs: | ||||
|   main: | ||||
|   semantic: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: amannn/action-semantic-pull-request@v3.0.0 | ||||
|         env: | ||||
|   | ||||
							
								
								
									
										17
									
								
								.github/workflows/sentry-production.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.github/workflows/sentry-production.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| name: New Sentry Production Release | ||||
| name: New Sentry production release | ||||
|  | ||||
| on: | ||||
|   push: | ||||
| @@ -6,28 +6,23 @@ on: | ||||
|       - master | ||||
|  | ||||
| jobs: | ||||
|   release: | ||||
|   sentry: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v1.0.0 | ||||
|  | ||||
|       - uses: actions/checkout@v2 | ||||
|       - name: Setup Node.js 14.x | ||||
|         uses: actions/setup-node@v1 | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 14.x | ||||
|  | ||||
|       - name: Install and build | ||||
|         run: | | ||||
|           npm ci | ||||
|           npm run build:app | ||||
|           yarn --frozen-lockfile | ||||
|           yarn build:app | ||||
|         env: | ||||
|           CI: true | ||||
|  | ||||
|       - name: Install Sentry | ||||
|         run: | | ||||
|           curl -sL https://sentry.io/get-cli/ | bash | ||||
|  | ||||
|       - name: Create new Sentry release | ||||
|         run: | | ||||
|           export SENTRY_RELEASE=$(sentry-cli releases propose-version) | ||||
|   | ||||
							
								
								
									
										13
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,20 +1,17 @@ | ||||
| name: Tests | ||||
|  | ||||
| on: push | ||||
| on: pull_request | ||||
|  | ||||
| jobs: | ||||
|   test: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v1 | ||||
|  | ||||
|       - uses: actions/checkout@v2 | ||||
|       - name: Setup Node.js 14.x | ||||
|         uses: actions/setup-node@v1 | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 14.x | ||||
|  | ||||
|       - name: Install and test | ||||
|         run: | | ||||
|           npm ci | ||||
|           npm run test:app | ||||
|           yarn --frozen-lockfile | ||||
|           yarn test:app | ||||
|   | ||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,7 @@ firebase | ||||
| logs | ||||
| node_modules | ||||
| npm-debug.log* | ||||
| package-lock.json | ||||
| static | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| yarn.lock | ||||
|   | ||||
| @@ -1,4 +0,0 @@ | ||||
| { | ||||
|   "proseWrap": "never", | ||||
|   "trailingComma": "all" | ||||
| } | ||||
| @@ -5,7 +5,7 @@ | ||||
| ### Option 1 - Manual | ||||
| 
 | ||||
| 1. Fork and clone the repo | ||||
| 1. Run `npm install` to install dependencies | ||||
| 1. Run `yarn` to install dependencies | ||||
| 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: | ||||
| @@ -19,7 +19,7 @@ | ||||
| ### Option 2 - CodeSandbox | ||||
| 
 | ||||
| 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. Tap on `Fork Sandbox` | ||||
| 1. Write your code | ||||
| @@ -35,7 +35,6 @@ Make sure the title starts with a semantic prefix: | ||||
| 
 | ||||
| - **feat**: A new feature | ||||
| - **fix**: A bug fix | ||||
| - **improvement**: An improvement to a current feature | ||||
| - **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 | ||||
| @@ -2,13 +2,13 @@ FROM node:14-alpine AS build | ||||
|  | ||||
| WORKDIR /opt/node_app | ||||
|  | ||||
| COPY package.json package-lock.json ./ | ||||
| RUN npm i --no-optional | ||||
| COPY package.json yarn.lock ./ | ||||
| RUN yarn --ignore-optional | ||||
|  | ||||
| ARG NODE_ENV=production | ||||
|  | ||||
| COPY . . | ||||
| RUN npm run build:app:docker | ||||
| RUN yarn build:app:docker | ||||
|  | ||||
| FROM nginx:1.17-alpine | ||||
|  | ||||
|   | ||||
							
								
								
									
										55
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ | ||||
|   <a href="https://excalidraw.com"> | ||||
|     <img width="540" src="./public/og-image-sm.png" alt="Excalidraw logo: Sketch handrawn like diagrams." /> | ||||
|   </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.<br>Collaborative and end-to-end encrypted.</h3> | ||||
|   <p> | ||||
|     <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"> | ||||
| @@ -11,6 +11,7 @@ | ||||
|       <img src="https://badges.crowdin.net/excalidraw/localized.svg"> | ||||
|     </a> | ||||
|   </p> | ||||
|   <p>Ask questions or hang out on our <a target="_blank" href="https://discord.gg/UexuTaE">discord.gg/UexuTaE</a>.</p> | ||||
| </div> | ||||
|  | ||||
| ## Try it now | ||||
| @@ -19,6 +20,18 @@ 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/). | ||||
|  | ||||
| ## 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 | ||||
| @@ -41,7 +54,7 @@ Translations will be available on the app if they exceed a certain threshold of | ||||
|  | ||||
| ### Create a collaboration session manually | ||||
|  | ||||
| In order to create a session manually you just need to generate a link of this form: | ||||
| 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} | ||||
| @@ -61,18 +74,28 @@ The second set of digits is the encryption key. The Excalidraw server doesn’t | ||||
|  | ||||
| Find a growing list of libraries containing assets for your drawings at [libraries.excalidraw.com](https://libraries.excalidraw.com). | ||||
|  | ||||
| ## Developement | ||||
| ## Embedding Excalidraw in your App? | ||||
|  | ||||
| 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 | ||||
|  | ||||
| - 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! | ||||
|  | ||||
| ### Local Installation | ||||
|  | ||||
| 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) | ||||
| - [Git](https://git-scm.com/downloads) | ||||
|  | ||||
| #### Clone the repo | ||||
|  | ||||
| ```bash | ||||
| @@ -81,26 +104,26 @@ git clone https://github.com/excalidraw/excalidraw.git | ||||
|  | ||||
| #### Commands | ||||
|  | ||||
| | Command               | Description                       | | ||||
| | --------------------- | --------------------------------- | | ||||
| | `npm install`         | Install the dependencies          | | ||||
| | `npm start`           | Run the project                   | | ||||
| | `npm run fix`         | Reformat all files with Prettier  | | ||||
| | `npm test`            | Run tests                         | | ||||
| | `npm run test:update` | Update test snapshots             | | ||||
| | `npm run test:code`   | Test for formatting with Prettier | | ||||
| | Command            | Description                       | | ||||
| | ------------------ | --------------------------------- | | ||||
| | `yarn`             | Install the dependencies          | | ||||
| | `yarn start`       | Run the project                   | | ||||
| | `yarn fix`         | Reformat all files with Prettier  | | ||||
| | `yarn test`        | Run tests                         | | ||||
| | `yarn test:update` | Update test snapshots             | | ||||
| | `yarn test:code`   | Test for formatting with Prettier | | ||||
|  | ||||
| #### Docker Compose | ||||
|  | ||||
| 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 | ||||
| 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 | ||||
| docker build -t excalidraw/excalidraw . | ||||
| @@ -111,7 +134,7 @@ 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.** | ||||
|  | ||||
| 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 | ||||
|  | ||||
|   | ||||
| @@ -18,7 +18,7 @@ services: | ||||
|     volumes: | ||||
|       - ./:/opt/node_app/app:delegated | ||||
|       - ./package.json:/opt/node_app/package.json | ||||
|       - ./package-lock.json:/opt/node_app/package-lock.json | ||||
|       - ./yarn.lock:/opt/node_app/yarn.lock | ||||
|       - notused:/opt/node_app/app/node_modules | ||||
|  | ||||
| volumes: | ||||
|   | ||||
							
								
								
									
										23391
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										23391
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										72
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										72
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,10 +1,5 @@ | ||||
| { | ||||
|   "browserslist": { | ||||
|     "development": [ | ||||
|       "last 1 chrome version", | ||||
|       "last 1 firefox version", | ||||
|       "last 1 safari version" | ||||
|     ], | ||||
|     "production": [ | ||||
|       ">0.2%", | ||||
|       "not dead", | ||||
| @@ -16,24 +11,28 @@ | ||||
|       "not chrome < 70", | ||||
|       "not and_uc < 13", | ||||
|       "not samsung < 10" | ||||
|     ], | ||||
|     "development": [ | ||||
|       "last 1 chrome version", | ||||
|       "last 1 firefox version", | ||||
|       "last 1 safari version" | ||||
|     ] | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@sentry/browser": "6.0.1", | ||||
|     "@sentry/integrations": "6.0.1", | ||||
|     "@sentry/browser": "6.2.2", | ||||
|     "@sentry/integrations": "6.2.1", | ||||
|     "@testing-library/jest-dom": "5.11.9", | ||||
|     "@testing-library/react": "11.2.3", | ||||
|     "@testing-library/react": "11.2.5", | ||||
|     "@types/jest": "26.0.20", | ||||
|     "@types/react": "17.0.0", | ||||
|     "@types/react-dom": "17.0.0", | ||||
|     "@types/socket.io-client": "1.4.35", | ||||
|     "browser-fs-access": "0.13.0", | ||||
|     "@types/react": "17.0.2", | ||||
|     "@types/react-dom": "17.0.1", | ||||
|     "@types/socket.io-client": "1.4.36", | ||||
|     "browser-fs-access": "0.14.2", | ||||
|     "clsx": "1.1.1", | ||||
|     "firebase": "8.2.5", | ||||
|     "firebase": "8.2.10", | ||||
|     "i18next-browser-languagedetector": "6.0.1", | ||||
|     "lodash.throttle": "4.1.1", | ||||
|     "nanoid": "3.1.20", | ||||
|     "node-sass": "4.14.1", | ||||
|     "nanoid": "3.1.21", | ||||
|     "open-color": "1.8.0", | ||||
|     "pako": "1.0.11", | ||||
|     "png-chunk-text": "1.0.0", | ||||
| @@ -43,26 +42,30 @@ | ||||
|     "pwacompat": "2.0.17", | ||||
|     "react": "17.0.1", | ||||
|     "react-dom": "17.0.1", | ||||
|     "react-scripts": "4.0.1", | ||||
|     "react-scripts": "4.0.3", | ||||
|     "roughjs": "4.3.1", | ||||
|     "sass": "1.32.8", | ||||
|     "socket.io-client": "2.3.1", | ||||
|     "typescript": "4.1.3" | ||||
|     "typescript": "4.2.3" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@excalidraw/eslint-config": "1.0.0", | ||||
|     "@excalidraw/prettier-config": "1.0.2", | ||||
|     "@types/lodash.throttle": "4.1.6", | ||||
|     "@types/pako": "1.0.1", | ||||
|     "eslint-config-prettier": "7.2.0", | ||||
|     "@types/resize-observer-browser": "0.1.5", | ||||
|     "eslint-config-prettier": "8.1.0", | ||||
|     "eslint-plugin-prettier": "3.3.1", | ||||
|     "firebase-tools": "9.2.2", | ||||
|     "firebase-tools": "9.6.1", | ||||
|     "husky": "4.3.8", | ||||
|     "jest-canvas-mock": "2.3.0", | ||||
|     "lint-staged": "10.5.3", | ||||
|     "jest-canvas-mock": "2.3.1", | ||||
|     "lint-staged": "10.5.4", | ||||
|     "pepjs": "0.5.3", | ||||
|     "prettier": "2.2.1", | ||||
|     "rewire": "5.0.0" | ||||
|   }, | ||||
|   "engines": { | ||||
|     "node": ">=12.0.0" | ||||
|     "node": ">=14.0.0" | ||||
|   }, | ||||
|   "homepage": ".", | ||||
|   "husky": { | ||||
| @@ -71,34 +74,35 @@ | ||||
|     } | ||||
|   }, | ||||
|   "jest": { | ||||
|     "resetMocks": false, | ||||
|     "transformIgnorePatterns": [ | ||||
|       "node_modules/(?!(roughjs|points-on-curve|path-data-parser|points-on-path|browser-fs-access)/)" | ||||
|     ] | ||||
|     ], | ||||
|     "resetMocks": false | ||||
|   }, | ||||
|   "name": "excalidraw", | ||||
|   "prettier": "@excalidraw/prettier-config", | ||||
|   "private": true, | ||||
|   "scripts": { | ||||
|     "build": "npm run build:app && npm run build:version", | ||||
|     "build-node": "node ./scripts/build-node.js", | ||||
|     "build:app": "REACT_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA react-scripts build", | ||||
|     "build:app:docker": "REACT_APP_DISABLE_SENTRY=true react-scripts build", | ||||
|     "build:app": "REACT_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA react-scripts build", | ||||
|     "build:version": "node ./scripts/build-version.js", | ||||
|     "build": "yarn build:app && yarn build:version", | ||||
|     "eject": "react-scripts eject", | ||||
|     "fix": "npm run fix:other && npm run fix:code", | ||||
|     "fix:code": "npm run test:code -- --fix", | ||||
|     "fix:other": "npm run prettier -- --write", | ||||
|     "fix:code": "yarn test:code --fix", | ||||
|     "fix:other": "yarn prettier --write", | ||||
|     "fix": "yarn fix:other && yarn fix:code", | ||||
|     "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", | ||||
|     "start": "react-scripts start", | ||||
|     "test": "npm run test:app", | ||||
|     "test:all": "npm run test:typecheck && npm run test:code && npm run test:other && npm run test:app -- --watchAll=false", | ||||
|     "test:all": "yarn test:typecheck && yarn test:code && yarn test:other && yarn test:app --watchAll=false", | ||||
|     "test:app": "react-scripts test --passWithNoTests", | ||||
|     "test:code": "eslint --max-warnings=0 --ignore-path .gitignore --ext .js,.ts,.tsx .", | ||||
|     "test:code": "eslint --max-warnings=0 --ext .js,.ts,.tsx .", | ||||
|     "test:debug": "react-scripts --inspect-brk test --runInBand --no-cache", | ||||
|     "test:other": "npm run prettier -- --list-different", | ||||
|     "test:other": "yarn prettier --list-different", | ||||
|     "test:typecheck": "tsc", | ||||
|     "test:update": "npm run test:app -- --updateSnapshot --watchAll=false" | ||||
|     "test:update": "yarn test:app --updateSnapshot --watchAll=false", | ||||
|     "test": "yarn test:app" | ||||
|   } | ||||
| } | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								public/Virgil.woff2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/Virgil.woff2
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,7 +1,7 @@ | ||||
| /* http://www.eaglefonts.com/fg-virgil-ttf-131249.htm */ | ||||
| @font-face { | ||||
|   font-family: "Virgil"; | ||||
|   src: url("FG_Virgil.woff2"); | ||||
|   src: url("Virgil.woff2"); | ||||
|   font-display: swap; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -57,9 +57,10 @@ | ||||
|  | ||||
|     <!-- Excalidraw version --> | ||||
|     <meta name="version" content="{version}" /> | ||||
|  | ||||
|     <link | ||||
|       rel="preload" | ||||
|       href="FG_Virgil.woff2" | ||||
|       href="Virgil.woff2" | ||||
|       as="font" | ||||
|       type="font/woff2" | ||||
|       crossorigin="anonymous" | ||||
| @@ -85,7 +86,9 @@ | ||||
|     /> | ||||
|  | ||||
|     <link rel="stylesheet" href="fonts.css" type="text/css" /> | ||||
|  | ||||
|     <script> | ||||
|       window.EXCALIDRAW_ASSET_PATH = "/"; | ||||
|     </script> | ||||
|     <% if (process.env.REACT_APP_GOOGLE_ANALYTICS_ID) { %> | ||||
|     <script | ||||
|       async | ||||
| @@ -109,8 +112,7 @@ | ||||
|           Roboto, Helvetica, Arial, sans-serif; | ||||
|         font-family: var(--ui-font); | ||||
|         -webkit-text-size-adjust: 100%; | ||||
|         -webkit-user-select: none; | ||||
|         user-select: none; | ||||
|  | ||||
|         width: 100vw; | ||||
|         height: 100vh; | ||||
|       } | ||||
|   | ||||
| @@ -26,5 +26,18 @@ | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "capture_links": "new_client" | ||||
|   "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"] | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-background-sync.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-background-sync.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-broadcast-update.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-broadcast-update.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-cacheable-response.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-cacheable-response.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-core.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-core.prod.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										2
									
								
								public/workbox/workbox-expiration.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-expiration.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-navigation-preload.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-navigation-preload.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-offline-ga.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-offline-ga.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-precaching.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-precaching.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-range-requests.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-range-requests.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-routing.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-routing.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-strategies.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-strategies.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-streams.prod.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-streams.prod.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-sw.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-sw.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| !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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-window.prod.es5.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-window.prod.es5.mjs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-window.prod.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-window.prod.mjs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| 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 | ||||
							
								
								
									
										2
									
								
								public/workbox/workbox-window.prod.umd.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								public/workbox/workbox-window.prod.umd.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| !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 | ||||
| @@ -5,7 +5,7 @@ | ||||
|  | ||||
| // In order to run: | ||||
| //   npm install canvas # please do not check it in | ||||
| //   npm run build-node | ||||
| //   yarn build-node | ||||
| //   node build/static/js/build-node.js | ||||
| //   open test.png | ||||
|  | ||||
|   | ||||
| @@ -18,11 +18,13 @@ const crowdinMap = { | ||||
|   "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", | ||||
| @@ -40,7 +42,7 @@ const crowdinMap = { | ||||
| const flags = { | ||||
|   "ar-SA": "🇸🇦", | ||||
|   "bg-BG": "🇧🇬", | ||||
|   "ca-ES": "🇪🇸", | ||||
|   "ca-ES": "🏳", | ||||
|   "de-DE": "🇩🇪", | ||||
|   "el-GR": "🇬🇷", | ||||
|   "es-ES": "🇪🇸", | ||||
| @@ -53,11 +55,13 @@ const flags = { | ||||
|   "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": "🇧🇷", | ||||
| @@ -88,11 +92,13 @@ const languages = { | ||||
|   "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", | ||||
|   | ||||
| @@ -58,7 +58,7 @@ export const actionAlignTop = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<AlignTopIcon appearance={appState.appearance} />} | ||||
|       icon={<AlignTopIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.alignTop")} — ${getShortcutKey( | ||||
|         "CtrlOrCmd+Shift+Up", | ||||
| @@ -87,7 +87,7 @@ export const actionAlignBottom = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<AlignBottomIcon appearance={appState.appearance} />} | ||||
|       icon={<AlignBottomIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.alignBottom")} — ${getShortcutKey( | ||||
|         "CtrlOrCmd+Shift+Down", | ||||
| @@ -116,7 +116,7 @@ export const actionAlignLeft = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<AlignLeftIcon appearance={appState.appearance} />} | ||||
|       icon={<AlignLeftIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.alignLeft")} — ${getShortcutKey( | ||||
|         "CtrlOrCmd+Shift+Left", | ||||
| @@ -145,7 +145,7 @@ export const actionAlignRight = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<AlignRightIcon appearance={appState.appearance} />} | ||||
|       icon={<AlignRightIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.alignRight")} — ${getShortcutKey( | ||||
|         "CtrlOrCmd+Shift+Right", | ||||
| @@ -172,7 +172,7 @@ export const actionAlignVerticallyCentered = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<CenterVerticallyIcon appearance={appState.appearance} />} | ||||
|       icon={<CenterVerticallyIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={t("labels.centerVertically")} | ||||
|       aria-label={t("labels.centerVertically")} | ||||
| @@ -197,7 +197,7 @@ export const actionAlignHorizontallyCentered = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<CenterHorizontallyIcon appearance={appState.appearance} />} | ||||
|       icon={<CenterHorizontallyIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={t("labels.centerHorizontally")} | ||||
|       aria-label={t("labels.centerHorizontally")} | ||||
|   | ||||
| @@ -48,7 +48,7 @@ export const actionClearCanvas = register({ | ||||
|       ), | ||||
|       appState: { | ||||
|         ...getDefaultAppState(), | ||||
|         appearance: appState.appearance, | ||||
|         theme: appState.theme, | ||||
|         elementLocked: appState.elementLocked, | ||||
|         exportBackground: appState.exportBackground, | ||||
|         exportEmbedScene: appState.exportEmbedScene, | ||||
|   | ||||
| @@ -17,7 +17,8 @@ export const actionCopy = register({ | ||||
|     }; | ||||
|   }, | ||||
|   contextItemLabel: "labels.copy", | ||||
|   keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.code === CODES.C, | ||||
|   // don't supply a shortcut since we handle this conditionally via onCopy event | ||||
|   keyTest: undefined, | ||||
| }); | ||||
|  | ||||
| export const actionCut = register({ | ||||
| @@ -94,7 +95,14 @@ export const actionCopyAsPng = register({ | ||||
|       return { | ||||
|         appState: { | ||||
|           ...appState, | ||||
|           toastMessage: t("toast.copyToClipboardAsPng"), | ||||
|           toastMessage: t("toast.copyToClipboardAsPng", { | ||||
|             exportSelection: selectedElements.length | ||||
|               ? t("toast.selection") | ||||
|               : t("toast.canvas"), | ||||
|             exportColorScheme: appState.exportWithDarkMode | ||||
|               ? t("buttons.darkMode") | ||||
|               : t("buttons.lightMode"), | ||||
|           }), | ||||
|         }, | ||||
|         commitToHistory: false, | ||||
|       }; | ||||
|   | ||||
| @@ -53,7 +53,7 @@ export const distributeHorizontally = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<DistributeHorizontallyIcon appearance={appState.appearance} />} | ||||
|       icon={<DistributeHorizontallyIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.distributeHorizontally")} — ${getShortcutKey( | ||||
|         "Alt+H", | ||||
| @@ -81,7 +81,7 @@ export const distributeVertically = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<DistributeVerticallyIcon appearance={appState.appearance} />} | ||||
|       icon={<DistributeVerticallyIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.distributeVertically")} — ${getShortcutKey("Alt+V")}`} | ||||
|       aria-label={t("labels.distributeVertically")} | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import { ProjectName } from "../components/ProjectName"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import "../components/ToolIcon.scss"; | ||||
| import { Tooltip } from "../components/Tooltip"; | ||||
| import { DarkModeToggle, Appearence } from "../components/DarkModeToggle"; | ||||
| import { loadFromJSON, saveAsJSON } from "../data"; | ||||
| import { t } from "../i18n"; | ||||
| import useIsMobile from "../is-mobile"; | ||||
| @@ -17,11 +18,12 @@ export const actionChangeProjectName = register({ | ||||
|     trackEvent("change", "title"); | ||||
|     return { appState: { ...appState, name: value }, commitToHistory: false }; | ||||
|   }, | ||||
|   PanelComponent: ({ appState, updateData }) => ( | ||||
|   PanelComponent: ({ appState, updateData, appProps }) => ( | ||||
|     <ProjectName | ||||
|       label={t("labels.fileTitle")} | ||||
|       value={appState.name || "Unnamed"} | ||||
|       onChange={(name: string) => updateData(name)} | ||||
|       isNameEditable={typeof appProps.name === "undefined"} | ||||
|     /> | ||||
|   ), | ||||
| }); | ||||
| @@ -96,9 +98,24 @@ export const actionChangeShouldAddWatermark = register({ | ||||
| export const actionSaveScene = register({ | ||||
|   name: "saveScene", | ||||
|   perform: async (elements, appState, value) => { | ||||
|     const fileHandleExists = !!appState.fileHandle; | ||||
|     try { | ||||
|       const { fileHandle } = await saveAsJSON(elements, appState); | ||||
|       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) { | ||||
|       if (error?.name !== "AbortError") { | ||||
|         console.error(error); | ||||
| @@ -189,3 +206,31 @@ export const actionLoadScene = register({ | ||||
|     /> | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| 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 ? "dark" : "light"} | ||||
|         onChange={(theme: Appearence) => { | ||||
|           updateData(theme === "dark"); | ||||
|         }} | ||||
|         title={t("labels.toggleExportColorScheme")} | ||||
|       /> | ||||
|     </div> | ||||
|   ), | ||||
| }); | ||||
|   | ||||
| @@ -18,7 +18,7 @@ import { isBindingElement } from "../element/typeChecks"; | ||||
|  | ||||
| export const actionFinalize = register({ | ||||
|   name: "finalize", | ||||
|   perform: (elements, appState) => { | ||||
|   perform: (elements, appState, _, { canvas }) => { | ||||
|     if (appState.editingLinearElement) { | ||||
|       const { | ||||
|         elementId, | ||||
| @@ -83,7 +83,7 @@ export const actionFinalize = register({ | ||||
|       // If the multi point line closes the loop, | ||||
|       // set the last point to first point. | ||||
|       // This ensures that loop remains closed at different scales. | ||||
|       const isLoop = isPathALoop(multiPointElement.points); | ||||
|       const isLoop = isPathALoop(multiPointElement.points, appState.zoom.value); | ||||
|       if ( | ||||
|         multiPointElement.type === "line" || | ||||
|         multiPointElement.type === "draw" | ||||
| @@ -126,7 +126,7 @@ export const actionFinalize = register({ | ||||
|       (!appState.elementLocked && appState.elementType !== "draw") || | ||||
|       !multiPointElement | ||||
|     ) { | ||||
|       resetCursor(); | ||||
|       resetCursor(canvas); | ||||
|     } | ||||
|     return { | ||||
|       elements: newElements, | ||||
|   | ||||
| @@ -134,7 +134,7 @@ export const actionGroup = register({ | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|       type="button" | ||||
|       icon={<GroupIcon appearance={appState.appearance} />} | ||||
|       icon={<GroupIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.group")} — ${getShortcutKey("CtrlOrCmd+G")}`} | ||||
|       aria-label={t("labels.group")} | ||||
| @@ -181,7 +181,7 @@ export const actionUngroup = register({ | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       hidden={getSelectedGroupIds(appState).length === 0} | ||||
|       icon={<UngroupIcon appearance={appState.appearance} />} | ||||
|       icon={<UngroupIcon theme={appState.theme} />} | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.ungroup")} — ${getShortcutKey("CtrlOrCmd+Shift+G")}`} | ||||
|       aria-label={t("labels.ungroup")} | ||||
|   | ||||
| @@ -42,7 +42,7 @@ export const actionGoToCollaborator = register({ | ||||
|       return null; | ||||
|     } | ||||
|  | ||||
|     const { background, stroke } = getClientColors(clientId); | ||||
|     const { background, stroke } = getClientColors(clientId, appState); | ||||
|     const shortName = getClientInitials(collaborator.username); | ||||
|  | ||||
|     return ( | ||||
|   | ||||
| @@ -169,17 +169,17 @@ export const actionChangeFillStyle = register({ | ||||
|           { | ||||
|             value: "hachure", | ||||
|             text: t("labels.hachure"), | ||||
|             icon: <FillHachureIcon appearance={appState.appearance} />, | ||||
|             icon: <FillHachureIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: "cross-hatch", | ||||
|             text: t("labels.crossHatch"), | ||||
|             icon: <FillCrossHatchIcon appearance={appState.appearance} />, | ||||
|             icon: <FillCrossHatchIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: "solid", | ||||
|             text: t("labels.solid"), | ||||
|             icon: <FillSolidIcon appearance={appState.appearance} />, | ||||
|             icon: <FillSolidIcon theme={appState.theme} />, | ||||
|           }, | ||||
|         ]} | ||||
|         group="fill" | ||||
| @@ -219,32 +219,17 @@ export const actionChangeStrokeWidth = register({ | ||||
|           { | ||||
|             value: 1, | ||||
|             text: t("labels.thin"), | ||||
|             icon: ( | ||||
|               <StrokeWidthIcon | ||||
|                 appearance={appState.appearance} | ||||
|                 strokeWidth={2} | ||||
|               /> | ||||
|             ), | ||||
|             icon: <StrokeWidthIcon theme={appState.theme} strokeWidth={2} />, | ||||
|           }, | ||||
|           { | ||||
|             value: 2, | ||||
|             text: t("labels.bold"), | ||||
|             icon: ( | ||||
|               <StrokeWidthIcon | ||||
|                 appearance={appState.appearance} | ||||
|                 strokeWidth={6} | ||||
|               /> | ||||
|             ), | ||||
|             icon: <StrokeWidthIcon theme={appState.theme} strokeWidth={6} />, | ||||
|           }, | ||||
|           { | ||||
|             value: 4, | ||||
|             text: t("labels.extraBold"), | ||||
|             icon: ( | ||||
|               <StrokeWidthIcon | ||||
|                 appearance={appState.appearance} | ||||
|                 strokeWidth={10} | ||||
|               /> | ||||
|             ), | ||||
|             icon: <StrokeWidthIcon theme={appState.theme} strokeWidth={10} />, | ||||
|           }, | ||||
|         ]} | ||||
|         value={getFormValue( | ||||
| @@ -282,17 +267,17 @@ export const actionChangeSloppiness = register({ | ||||
|           { | ||||
|             value: 0, | ||||
|             text: t("labels.architect"), | ||||
|             icon: <SloppinessArchitectIcon appearance={appState.appearance} />, | ||||
|             icon: <SloppinessArchitectIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: 1, | ||||
|             text: t("labels.artist"), | ||||
|             icon: <SloppinessArtistIcon appearance={appState.appearance} />, | ||||
|             icon: <SloppinessArtistIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: 2, | ||||
|             text: t("labels.cartoonist"), | ||||
|             icon: <SloppinessCartoonistIcon appearance={appState.appearance} />, | ||||
|             icon: <SloppinessCartoonistIcon theme={appState.theme} />, | ||||
|           }, | ||||
|         ]} | ||||
|         value={getFormValue( | ||||
| @@ -329,17 +314,17 @@ export const actionChangeStrokeStyle = register({ | ||||
|           { | ||||
|             value: "solid", | ||||
|             text: t("labels.strokeStyle_solid"), | ||||
|             icon: <StrokeStyleSolidIcon appearance={appState.appearance} />, | ||||
|             icon: <StrokeStyleSolidIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: "dashed", | ||||
|             text: t("labels.strokeStyle_dashed"), | ||||
|             icon: <StrokeStyleDashedIcon appearance={appState.appearance} />, | ||||
|             icon: <StrokeStyleDashedIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: "dotted", | ||||
|             text: t("labels.strokeStyle_dotted"), | ||||
|             icon: <StrokeStyleDottedIcon appearance={appState.appearance} />, | ||||
|             icon: <StrokeStyleDottedIcon theme={appState.theme} />, | ||||
|           }, | ||||
|         ]} | ||||
|         value={getFormValue( | ||||
| @@ -580,12 +565,12 @@ export const actionChangeSharpness = register({ | ||||
|           { | ||||
|             value: "sharp", | ||||
|             text: t("labels.sharp"), | ||||
|             icon: <EdgeSharpIcon appearance={appState.appearance} />, | ||||
|             icon: <EdgeSharpIcon theme={appState.theme} />, | ||||
|           }, | ||||
|           { | ||||
|             value: "round", | ||||
|             text: t("labels.round"), | ||||
|             icon: <EdgeRoundIcon appearance={appState.appearance} />, | ||||
|             icon: <EdgeRoundIcon theme={appState.theme} />, | ||||
|           }, | ||||
|         ]} | ||||
|         value={getFormValue( | ||||
| @@ -653,40 +638,27 @@ export const actionChangeArrowhead = register({ | ||||
|               { | ||||
|                 value: null, | ||||
|                 text: t("labels.arrowhead_none"), | ||||
|                 icon: <ArrowheadNoneIcon appearance={appState.appearance} />, | ||||
|                 icon: <ArrowheadNoneIcon theme={appState.theme} />, | ||||
|                 keyBinding: "q", | ||||
|               }, | ||||
|               { | ||||
|                 value: "arrow", | ||||
|                 text: t("labels.arrowhead_arrow"), | ||||
|                 icon: ( | ||||
|                   <ArrowheadArrowIcon | ||||
|                     appearance={appState.appearance} | ||||
|                     flip={!isRTL} | ||||
|                   /> | ||||
|                   <ArrowheadArrowIcon theme={appState.theme} flip={!isRTL} /> | ||||
|                 ), | ||||
|                 keyBinding: "w", | ||||
|               }, | ||||
|               { | ||||
|                 value: "bar", | ||||
|                 text: t("labels.arrowhead_bar"), | ||||
|                 icon: ( | ||||
|                   <ArrowheadBarIcon | ||||
|                     appearance={appState.appearance} | ||||
|                     flip={!isRTL} | ||||
|                   /> | ||||
|                 ), | ||||
|                 icon: <ArrowheadBarIcon theme={appState.theme} flip={!isRTL} />, | ||||
|                 keyBinding: "e", | ||||
|               }, | ||||
|               { | ||||
|                 value: "dot", | ||||
|                 text: t("labels.arrowhead_dot"), | ||||
|                 icon: ( | ||||
|                   <ArrowheadDotIcon | ||||
|                     appearance={appState.appearance} | ||||
|                     flip={!isRTL} | ||||
|                   /> | ||||
|                 ), | ||||
|                 icon: <ArrowheadDotIcon theme={appState.theme} flip={!isRTL} />, | ||||
|                 keyBinding: "r", | ||||
|               }, | ||||
|             ]} | ||||
| @@ -709,40 +681,27 @@ export const actionChangeArrowhead = register({ | ||||
|                 value: null, | ||||
|                 text: t("labels.arrowhead_none"), | ||||
|                 keyBinding: "q", | ||||
|                 icon: <ArrowheadNoneIcon appearance={appState.appearance} />, | ||||
|                 icon: <ArrowheadNoneIcon theme={appState.theme} />, | ||||
|               }, | ||||
|               { | ||||
|                 value: "arrow", | ||||
|                 text: t("labels.arrowhead_arrow"), | ||||
|                 keyBinding: "w", | ||||
|                 icon: ( | ||||
|                   <ArrowheadArrowIcon | ||||
|                     appearance={appState.appearance} | ||||
|                     flip={isRTL} | ||||
|                   /> | ||||
|                   <ArrowheadArrowIcon theme={appState.theme} flip={isRTL} /> | ||||
|                 ), | ||||
|               }, | ||||
|               { | ||||
|                 value: "bar", | ||||
|                 text: t("labels.arrowhead_bar"), | ||||
|                 keyBinding: "e", | ||||
|                 icon: ( | ||||
|                   <ArrowheadBarIcon | ||||
|                     appearance={appState.appearance} | ||||
|                     flip={isRTL} | ||||
|                   /> | ||||
|                 ), | ||||
|                 icon: <ArrowheadBarIcon theme={appState.theme} flip={isRTL} />, | ||||
|               }, | ||||
|               { | ||||
|                 value: "dot", | ||||
|                 text: t("labels.arrowhead_dot"), | ||||
|                 keyBinding: "r", | ||||
|                 icon: ( | ||||
|                   <ArrowheadDotIcon | ||||
|                     appearance={appState.appearance} | ||||
|                     flip={isRTL} | ||||
|                   /> | ||||
|                 ), | ||||
|                 icon: <ArrowheadDotIcon theme={appState.theme} flip={isRTL} />, | ||||
|               }, | ||||
|             ]} | ||||
|             value={getFormValue<Arrowhead | null>( | ||||
|   | ||||
| @@ -2,10 +2,12 @@ import { CODES, KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { GRID_SIZE } from "../constants"; | ||||
| import { AppState } from "../types"; | ||||
| import { trackEvent } from "../analytics"; | ||||
|  | ||||
| export const actionToggleGridMode = register({ | ||||
|   name: "gridMode", | ||||
|   perform(elements, appState) { | ||||
|     trackEvent("view", "mode", "grid"); | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
| @@ -15,6 +17,6 @@ export const actionToggleGridMode = register({ | ||||
|     }; | ||||
|   }, | ||||
|   checked: (appState: AppState) => appState.gridSize !== null, | ||||
|   contextItemLabel: "labels.gridMode", | ||||
|   contextItemLabel: "labels.showGrid", | ||||
|   keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.code === CODES.QUOTE, | ||||
| }); | ||||
|   | ||||
							
								
								
									
										22
									
								
								src/actions/actionToggleViewMode.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/actions/actionToggleViewMode.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| 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), | ||||
|         selectedElementIds: {}, | ||||
|       }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   }, | ||||
|   checked: (appState) => appState.viewModeEnabled, | ||||
|   contextItemLabel: "labels.viewMode", | ||||
|   keyTest: (event) => | ||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.R, | ||||
| }); | ||||
| @@ -1,9 +1,12 @@ | ||||
| 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, | ||||
|   | ||||
| @@ -38,7 +38,7 @@ export const actionSendBackward = register({ | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.sendBackward")} — ${getShortcutKey("CtrlOrCmd+[")}`} | ||||
|     > | ||||
|       <SendBackwardIcon appearance={appState.appearance} /> | ||||
|       <SendBackwardIcon theme={appState.theme} /> | ||||
|     </button> | ||||
|   ), | ||||
| }); | ||||
| @@ -65,7 +65,7 @@ export const actionBringForward = register({ | ||||
|       onClick={() => updateData(null)} | ||||
|       title={`${t("labels.bringForward")} — ${getShortcutKey("CtrlOrCmd+]")}`} | ||||
|     > | ||||
|       <BringForwardIcon appearance={appState.appearance} /> | ||||
|       <BringForwardIcon theme={appState.theme} /> | ||||
|     </button> | ||||
|   ), | ||||
| }); | ||||
| @@ -99,7 +99,7 @@ export const actionSendToBack = register({ | ||||
|           : getShortcutKey("CtrlOrCmd+Shift+[") | ||||
|       }`} | ||||
|     > | ||||
|       <SendToBackIcon appearance={appState.appearance} /> | ||||
|       <SendToBackIcon theme={appState.theme} /> | ||||
|     </button> | ||||
|   ), | ||||
| }); | ||||
| @@ -133,7 +133,7 @@ export const actionBringToFront = register({ | ||||
|           : getShortcutKey("CtrlOrCmd+Shift+]") | ||||
|       }`} | ||||
|     > | ||||
|       <BringToFrontIcon appearance={appState.appearance} /> | ||||
|       <BringToFrontIcon theme={appState.theme} /> | ||||
|     </button> | ||||
|   ), | ||||
| }); | ||||
|   | ||||
| @@ -7,11 +7,12 @@ import { | ||||
|   ActionResult, | ||||
| } from "./types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppState } from "../types"; | ||||
| import { AppState, ExcalidrawProps } from "../types"; | ||||
| import { MODES } from "../constants"; | ||||
|  | ||||
| // This is the <App> component, but for now we don't care about anything but its | ||||
| // `canvas` state. | ||||
| type App = { canvas: HTMLCanvasElement | null }; | ||||
| type App = { canvas: HTMLCanvasElement | null; props: ExcalidrawProps }; | ||||
|  | ||||
| export class ActionManager implements ActionsManagerInterface { | ||||
|   actions = {} as ActionsManagerInterface["actions"]; | ||||
| @@ -66,6 +67,12 @@ export class ActionManager implements ActionsManagerInterface { | ||||
|     if (data.length === 0) { | ||||
|       return false; | ||||
|     } | ||||
|     const { viewModeEnabled } = this.getAppState(); | ||||
|     if (viewModeEnabled) { | ||||
|       if (!Object.values(MODES).includes(data[0].name)) { | ||||
|         return false; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     event.preventDefault(); | ||||
|     this.updater( | ||||
| @@ -115,6 +122,7 @@ export class ActionManager implements ActionsManagerInterface { | ||||
|           appState={this.getAppState()} | ||||
|           updateData={updateData} | ||||
|           id={id} | ||||
|           appProps={this.app.props} | ||||
|         /> | ||||
|       ); | ||||
|     } | ||||
|   | ||||
| @@ -22,7 +22,8 @@ export type ShortcutName = | ||||
|   | "gridMode" | ||||
|   | "zenMode" | ||||
|   | "stats" | ||||
|   | "addToLibrary"; | ||||
|   | "addToLibrary" | ||||
|   | "viewMode"; | ||||
|  | ||||
| const shortcutMap: Record<ShortcutName, string[]> = { | ||||
|   cut: [getShortcutKey("CtrlOrCmd+X")], | ||||
| @@ -56,6 +57,7 @@ const shortcutMap: Record<ShortcutName, string[]> = { | ||||
|   zenMode: [getShortcutKey("Alt+Z")], | ||||
|   stats: [], | ||||
|   addToLibrary: [], | ||||
|   viewMode: [getShortcutKey("Alt+R")], | ||||
| }; | ||||
|  | ||||
| export const getShortcutFromShortcutName = (name: ShortcutName) => { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppState } from "../types"; | ||||
| import { AppState, ExcalidrawProps } from "../types"; | ||||
|  | ||||
| /** if false, the action should be prevented */ | ||||
| export type ActionResult = | ||||
| @@ -84,7 +84,9 @@ export type ActionName = | ||||
|   | "alignVerticallyCentered" | ||||
|   | "alignHorizontallyCentered" | ||||
|   | "distributeHorizontally" | ||||
|   | "distributeVertically"; | ||||
|   | "distributeVertically" | ||||
|   | "viewMode" | ||||
|   | "exportWithDarkMode"; | ||||
|  | ||||
| export interface Action { | ||||
|   name: ActionName; | ||||
| @@ -92,6 +94,7 @@ export interface Action { | ||||
|     elements: readonly ExcalidrawElement[]; | ||||
|     appState: AppState; | ||||
|     updateData: (formData?: any) => void; | ||||
|     appProps: ExcalidrawProps; | ||||
|     id?: string; | ||||
|   }>; | ||||
|   perform: ActionFn; | ||||
|   | ||||
| @@ -13,7 +13,7 @@ export const getDefaultAppState = (): Omit< | ||||
|   "offsetTop" | "offsetLeft" | ||||
| > => { | ||||
|   return { | ||||
|     appearance: "light", | ||||
|     theme: "light", | ||||
|     collaborators: new Map(), | ||||
|     currentChartType: "bar", | ||||
|     currentItemBackgroundColor: "transparent", | ||||
| @@ -40,6 +40,7 @@ export const getDefaultAppState = (): Omit< | ||||
|     errorMessage: null, | ||||
|     exportBackground: true, | ||||
|     exportEmbedScene: false, | ||||
|     exportWithDarkMode: false, | ||||
|     fileHandle: null, | ||||
|     gridSize: null, | ||||
|     height: window.innerHeight, | ||||
| @@ -72,6 +73,7 @@ export const getDefaultAppState = (): Omit< | ||||
|     width: window.innerWidth, | ||||
|     zenModeEnabled: false, | ||||
|     zoom: { value: 1 as NormalizedZoomValue, translation: { x: 0, y: 0 } }, | ||||
|     viewModeEnabled: false, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| @@ -90,7 +92,7 @@ const APP_STATE_STORAGE_CONF = (< | ||||
| >( | ||||
|   config: { [K in keyof T]: K extends keyof AppState ? T[K] : never }, | ||||
| ) => config)({ | ||||
|   appearance: { browser: true, export: false }, | ||||
|   theme: { browser: true, export: false }, | ||||
|   collaborators: { browser: false, export: false }, | ||||
|   currentChartType: { browser: true, export: false }, | ||||
|   currentItemBackgroundColor: { browser: true, export: false }, | ||||
| @@ -117,6 +119,7 @@ const APP_STATE_STORAGE_CONF = (< | ||||
|   errorMessage: { browser: false, export: false }, | ||||
|   exportBackground: { browser: true, export: false }, | ||||
|   exportEmbedScene: { browser: true, export: false }, | ||||
|   exportWithDarkMode: { browser: true, export: false }, | ||||
|   fileHandle: { browser: false, export: false }, | ||||
|   gridSize: { browser: true, export: true }, | ||||
|   height: { browser: false, export: false }, | ||||
| @@ -151,6 +154,7 @@ const APP_STATE_STORAGE_CONF = (< | ||||
|   width: { browser: false, export: false }, | ||||
|   zenModeEnabled: { browser: true, export: false }, | ||||
|   zoom: { browser: true, export: false }, | ||||
|   viewModeEnabled: { browser: false, export: false }, | ||||
| }); | ||||
|  | ||||
| const _clearAppStateForStorage = <ExportType extends "export" | "browser">( | ||||
|   | ||||
| @@ -1,6 +1,13 @@ | ||||
| import colors from "./colors"; | ||||
| import { AppState } from "./types"; | ||||
|  | ||||
| export const getClientColors = (clientId: string) => { | ||||
| export const getClientColors = (clientId: string, appState: AppState) => { | ||||
|   if (appState?.collaborators) { | ||||
|     const currentUser = appState.collaborators.get(clientId); | ||||
|     if (currentUser?.color) { | ||||
|       return currentUser.color; | ||||
|     } | ||||
|   } | ||||
|   // Naive way of getting an integer out of the clientId | ||||
|   const sum = clientId.split("").reduce((a, str) => a + str.charCodeAt(0), 0); | ||||
|  | ||||
|   | ||||
| @@ -151,10 +151,12 @@ const LIBRARY_ICON = ( | ||||
| ); | ||||
|  | ||||
| export const ShapesSwitcher = ({ | ||||
|   canvas, | ||||
|   elementType, | ||||
|   setAppState, | ||||
|   isLibraryOpen, | ||||
| }: { | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   elementType: ExcalidrawElement["type"]; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   isLibraryOpen: boolean; | ||||
| @@ -185,7 +187,7 @@ export const ShapesSwitcher = ({ | ||||
|               multiElement: null, | ||||
|               selectedElementIds: {}, | ||||
|             }); | ||||
|             setCursorForShape(value); | ||||
|             setCursorForShape(canvas, value); | ||||
|             setAppState({}); | ||||
|           }} | ||||
|         /> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -7,20 +7,24 @@ export const BackgroundPickerAndDarkModeToggle = ({ | ||||
|   appState, | ||||
|   setAppState, | ||||
|   actionManager, | ||||
|   showThemeBtn, | ||||
| }: { | ||||
|   actionManager: ActionManager; | ||||
|   appState: AppState; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   showThemeBtn: boolean; | ||||
| }) => ( | ||||
|   <div style={{ display: "flex" }}> | ||||
|     {actionManager.renderAction("changeViewBackgroundColor")} | ||||
|     <div style={{ marginInlineStart: "0.25rem" }}> | ||||
|       <DarkModeToggle | ||||
|         value={appState.appearance} | ||||
|         onChange={(appearance) => { | ||||
|           setAppState({ appearance }); | ||||
|         }} | ||||
|       /> | ||||
|     </div> | ||||
|     {showThemeBtn && ( | ||||
|       <div style={{ marginInlineStart: "0.25rem" }}> | ||||
|         <DarkModeToggle | ||||
|           value={appState.theme} | ||||
|           onChange={(theme) => { | ||||
|             setAppState({ theme }); | ||||
|           }} | ||||
|         /> | ||||
|       </div> | ||||
|     )} | ||||
|   </div> | ||||
| ); | ||||
|   | ||||
| @@ -14,11 +14,11 @@ export const ButtonIconCycle = <T extends any>({ | ||||
| }) => { | ||||
|   const current = options.find((op) => op.value === value); | ||||
|  | ||||
|   const cycle = () => { | ||||
|   function cycle() { | ||||
|     const index = options.indexOf(current!); | ||||
|     const next = (index + 1) % options.length; | ||||
|     onChange(options[next].value); | ||||
|   }; | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <label key={group} className={clsx({ active: current!.value !== null })}> | ||||
|   | ||||
| @@ -4,7 +4,8 @@ | ||||
|   .CollabButton.is-collaborating { | ||||
|     background-color: var(--button-special-active-bg-color); | ||||
|  | ||||
|     .ToolIcon__icon svg { | ||||
|     .ToolIcon__icon svg, | ||||
|     .ToolIcon__label { | ||||
|       color: var(--icon-green-fill-color); | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -25,8 +25,8 @@ const CollabButton = ({ | ||||
|         onClick={onClick} | ||||
|         icon={users} | ||||
|         type="button" | ||||
|         title={t("buttons.roomDialog")} | ||||
|         aria-label={t("buttons.roomDialog")} | ||||
|         title={t("labels.liveCollaboration")} | ||||
|         aria-label={t("labels.liveCollaboration")} | ||||
|         showAriaLabel={useIsMobile()} | ||||
|       > | ||||
|         {collaboratorCount > 0 && ( | ||||
|   | ||||
| @@ -73,7 +73,7 @@ | ||||
|     box-sizing: border-box; | ||||
|     border: 1px solid #ddd; | ||||
|     background-color: currentColor !important; | ||||
|     filter: var(--appearance-filter); | ||||
|     filter: var(--theme-filter); | ||||
|  | ||||
|     &:focus { | ||||
|       /* TODO: only show the border when the color is too light to see as a shadow */ | ||||
| @@ -164,7 +164,7 @@ | ||||
|     margin: 0; | ||||
|     font-size: 1rem; | ||||
|     background-color: var(--input-bg-color); | ||||
|     color: var(--text-color-primary); | ||||
|     color: var(--text-primary-color); | ||||
|     border: 0; | ||||
|     outline: none; | ||||
|     height: 1.75em; | ||||
| @@ -192,7 +192,7 @@ | ||||
|     position: relative; | ||||
|     overflow: hidden; | ||||
|     background-color: transparent !important; | ||||
|     filter: var(--appearance-filter); | ||||
|     filter: var(--theme-filter); | ||||
|  | ||||
|     &:after { | ||||
|       content: ""; | ||||
| @@ -228,7 +228,7 @@ | ||||
|   } | ||||
|  | ||||
|   .color-picker-type-elementBackground .color-picker-keybinding { | ||||
|     color: #fff; | ||||
|     color: $oc-white; | ||||
|   } | ||||
|  | ||||
|   .color-picker-swatch[aria-label="transparent"] .color-picker-keybinding { | ||||
| @@ -239,12 +239,12 @@ | ||||
|     color: #d4d4d4; | ||||
|   } | ||||
|  | ||||
|   &.Appearance_dark { | ||||
|   &.theme--dark { | ||||
|     .color-picker-type-elementBackground .color-picker-keybinding { | ||||
|       color: #000; | ||||
|       color: $oc-black; | ||||
|     } | ||||
|     .color-picker-swatch[aria-label="transparent"] .color-picker-keybinding { | ||||
|       color: #000; | ||||
|       color: $oc-black; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -13,7 +13,7 @@ import { Action } from "../actions/types"; | ||||
| import { ActionManager } from "../actions/manager"; | ||||
| import { AppState } from "../types"; | ||||
|  | ||||
| type ContextMenuOption = "separator" | Action; | ||||
| export type ContextMenuOption = "separator" | Action; | ||||
|  | ||||
| type ContextMenuProps = { | ||||
|   options: ContextMenuOption[]; | ||||
| @@ -34,11 +34,11 @@ const ContextMenu = ({ | ||||
| }: ContextMenuProps) => { | ||||
|   const isDarkTheme = !!document | ||||
|     .querySelector(".excalidraw") | ||||
|     ?.classList.contains("Appearance_dark"); | ||||
|     ?.classList.contains("theme--dark"); | ||||
|   return ( | ||||
|     <div | ||||
|       className={clsx("excalidraw", { | ||||
|         "Appearance_dark Appearance_dark-background-none": isDarkTheme, | ||||
|         "theme--dark theme--dark-background-none": isDarkTheme, | ||||
|       })} | ||||
|     > | ||||
|       <Popover | ||||
|   | ||||
| @@ -10,13 +10,19 @@ export type Appearence = "light" | "dark"; | ||||
| export const DarkModeToggle = (props: { | ||||
|   value: Appearence; | ||||
|   onChange: (value: Appearence) => void; | ||||
|   title?: string; | ||||
| }) => { | ||||
|   const title = props.title | ||||
|     ? props.title | ||||
|     : props.value === "dark" | ||||
|     ? t("buttons.lightMode") | ||||
|     : t("buttons.darkMode"); | ||||
|  | ||||
|   return ( | ||||
|     <label | ||||
|       className={`ToolIcon ToolIcon_type_floating ToolIcon_size_M`} | ||||
|       title={ | ||||
|         props.value === "dark" ? t("buttons.lightMode") : t("buttons.darkMode") | ||||
|       } | ||||
|       className="ToolIcon ToolIcon_type_floating ToolIcon_size_M" | ||||
|       data-testid="toggle-dark-mode" | ||||
|       title={title} | ||||
|     > | ||||
|       <input | ||||
|         className="ToolIcon_type_checkbox ToolIcon_toggle_opaque" | ||||
| @@ -25,11 +31,7 @@ export const DarkModeToggle = (props: { | ||||
|           props.onChange(event.target.checked ? "dark" : "light") | ||||
|         } | ||||
|         checked={props.value === "dark"} | ||||
|         aria-label={ | ||||
|           props.value === "dark" | ||||
|             ? t("buttons.lightMode") | ||||
|             : t("buttons.darkMode") | ||||
|         } | ||||
|         aria-label={title} | ||||
|       /> | ||||
|       <div className="ToolIcon__icon"> | ||||
|         {props.value === "light" ? ICONS.MOON : ICONS.SUN} | ||||
|   | ||||
| @@ -16,7 +16,7 @@ | ||||
|     max-height: 25rem; | ||||
|   } | ||||
|  | ||||
|   &.Appearance_dark .ExportDialog__preview canvas { | ||||
|   &.theme--dark .ExportDialog__preview canvas { | ||||
|     filter: none; | ||||
|   } | ||||
|  | ||||
| @@ -34,6 +34,14 @@ | ||||
|  | ||||
|     .TextInput { | ||||
|       height: calc(1rem - 3px); | ||||
|  | ||||
|       &--readonly { | ||||
|         background: none; | ||||
|         border: none; | ||||
|         &:hover { | ||||
|           background: none; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -19,6 +19,9 @@ import { ToolButton } from "./ToolButton"; | ||||
| const scales = [1, 2, 3]; | ||||
| const defaultScale = scales.includes(devicePixelRatio) ? devicePixelRatio : 1; | ||||
|  | ||||
| const supportsContextFilters = | ||||
|   "filter" in document.createElement("canvas").getContext("2d")!; | ||||
|  | ||||
| export const ErrorCanvasPreview = () => { | ||||
|   return ( | ||||
|     <div> | ||||
| @@ -128,6 +131,8 @@ const ExportModal = ({ | ||||
|   return ( | ||||
|     <div className="ExportDialog"> | ||||
|       <div className="ExportDialog__preview" ref={previewRef} /> | ||||
|       {supportsContextFilters && | ||||
|         actionManager.renderAction("exportWithDarkMode")} | ||||
|       <Stack.Col gap={2} align="center"> | ||||
|         <div className="ExportDialog__actions"> | ||||
|           <Stack.Row gap={2}> | ||||
| @@ -252,6 +257,7 @@ export const ExportDialog = ({ | ||||
|         onClick={() => { | ||||
|           setModalIsShown(true); | ||||
|         }} | ||||
|         data-testid="export-button" | ||||
|         icon={exportFile} | ||||
|         type="button" | ||||
|         aria-label={t("buttons.export")} | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import React from "react"; | ||||
|  | ||||
| // https://github.com/tholman/github-corners | ||||
| export const GitHubCorner = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => ( | ||||
|   ({ theme }: { theme: "light" | "dark" }) => ( | ||||
|     <svg | ||||
|       xmlns="http://www.w3.org/2000/svg" | ||||
|       width="40" | ||||
| @@ -19,18 +19,18 @@ export const GitHubCorner = React.memo( | ||||
|       > | ||||
|         <path | ||||
|           d="M0 0l115 115h15l12 27 108 108V0z" | ||||
|           fill={appearance === "light" ? oc.gray[6] : oc.gray[8]} | ||||
|           fill={theme === "light" ? oc.gray[6] : oc.gray[8]} | ||||
|         /> | ||||
|         <path | ||||
|           className="octo-arm" | ||||
|           d="M128 109c-15-9-9-19-9-19 3-7 2-11 2-11-1-7 3-2 3-2 4 5 2 11 2 11-3 10 5 15 9 16" | ||||
|           style={{ transformOrigin: "130px 106px" }} | ||||
|           fill={appearance === "light" ? oc.white : oc.black} | ||||
|           fill={theme === "light" ? oc.white : oc.black} | ||||
|         /> | ||||
|         <path | ||||
|           className="octo-body" | ||||
|           d="M115 115s4 2 5 0l14-14c3-2 6-3 8-3-8-11-15-24 2-41 5-5 10-7 16-7 1-2 3-7 12-11 0 0 5 3 7 16 4 2 8 5 12 9s7 8 9 12c14 3 17 7 17 7-4 8-9 11-11 11 0 6-2 11-7 16-16 16-30 10-41 2 0 3-1 7-5 11l-12 11c-1 1 1 5 1 5z" | ||||
|           fill={appearance === "light" ? oc.white : oc.black} | ||||
|           fill={theme === "light" ? oc.white : oc.black} | ||||
|         /> | ||||
|       </a> | ||||
|     </svg> | ||||
|   | ||||
| @@ -224,9 +224,13 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|                   shortcuts={[getShortcutKey("Alt+Z")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.gridMode")} | ||||
|                   label={t("labels.showGrid")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+'")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.viewMode")} | ||||
|                   shortcuts={[getShortcutKey("Alt+R")]} | ||||
|                 /> | ||||
|               </ShortcutIsland> | ||||
|             </Column> | ||||
|             <Column> | ||||
|   | ||||
| @@ -121,7 +121,7 @@ | ||||
|   } | ||||
|  | ||||
|   .picker-type-elementBackground .picker-keybinding { | ||||
|     color: #fff; | ||||
|     color: $oc-white; | ||||
|   } | ||||
|  | ||||
|   .picker-swatch[aria-label="transparent"] .picker-keybinding { | ||||
| @@ -132,12 +132,12 @@ | ||||
|     color: #d4d4d4; | ||||
|   } | ||||
|  | ||||
|   &.Appearance_dark { | ||||
|   &.theme--dark { | ||||
|     .picker-type-elementBackground .picker-keybinding { | ||||
|       color: #000; | ||||
|       color: $oc-black; | ||||
|     } | ||||
|     .picker-swatch[aria-label="transparent"] .picker-keybinding { | ||||
|       color: #000; | ||||
|       color: $oc-black; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,12 +1,7 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { LoadingMessage } from "./LoadingMessage"; | ||||
| import { | ||||
|   defaultLang, | ||||
|   Language, | ||||
|   languages, | ||||
|   setLanguageFirstTime, | ||||
| } from "../i18n"; | ||||
| import { defaultLang, Language, languages, setLanguage } from "../i18n"; | ||||
|  | ||||
| interface Props { | ||||
|   langCode: Language["code"]; | ||||
| @@ -23,7 +18,7 @@ export class InitializeApp extends React.Component<Props, State> { | ||||
|     const currentLang = | ||||
|       languages.find((lang) => lang.code === this.props.langCode) || | ||||
|       defaultLang; | ||||
|     await setLanguageFirstTime(currentLang); | ||||
|     await setLanguage(currentLang); | ||||
|     this.setState({ | ||||
|       isLoading: false, | ||||
|     }); | ||||
|   | ||||
| @@ -19,9 +19,9 @@ | ||||
|       } | ||||
|  | ||||
|       a { | ||||
|         margin-left: auto; | ||||
|         margin-inline-start: auto; | ||||
|         // 17px for scrollbar (needed for overlay scrollbars on Big Sur?) + 1px extra | ||||
|         padding-right: 18px; | ||||
|         padding-inline-end: 18px; | ||||
|         white-space: nowrap; | ||||
|       } | ||||
|     } | ||||
| @@ -38,6 +38,8 @@ | ||||
|   } | ||||
|  | ||||
|   .layer-ui__wrapper { | ||||
|     z-index: var(--zIndex-layerUI); | ||||
|  | ||||
|     .encrypted-icon { | ||||
|       position: relative; | ||||
|       margin-inline-start: 15px; | ||||
|   | ||||
| @@ -11,13 +11,13 @@ import { CLASSES } from "../constants"; | ||||
| import { exportCanvas } from "../data"; | ||||
| import { importLibraryFromJSON, saveLibraryAsJSON } from "../data/json"; | ||||
| import { Library } from "../data/library"; | ||||
| import { showSelectedShapeActions } from "../element"; | ||||
| import { isTextElement, showSelectedShapeActions } from "../element"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
| import { Language, t } from "../i18n"; | ||||
| import useIsMobile from "../is-mobile"; | ||||
| import { calculateScrollCenter, getSelectedElements } from "../scene"; | ||||
| import { ExportType } from "../scene/types"; | ||||
| import { AppState, LibraryItem, LibraryItems } from "../types"; | ||||
| import { AppState, ExcalidrawProps, LibraryItem, LibraryItems } from "../types"; | ||||
| import { muteFSAbortError } from "../utils"; | ||||
| import { SelectedShapeActions, ShapesSwitcher, ZoomActions } from "./Actions"; | ||||
| import { BackgroundPickerAndDarkModeToggle } from "./BackgroundPickerAndDarkModeToggle"; | ||||
| @@ -27,7 +27,7 @@ import { ExportCB, ExportDialog } from "./ExportDialog"; | ||||
| import { FixedSideContainer } from "./FixedSideContainer"; | ||||
| import { GitHubCorner } from "./GitHubCorner"; | ||||
| import { HintViewer } from "./HintViewer"; | ||||
| import { exportFile, load, shield } from "./icons"; | ||||
| import { exportFile, load, shield, trash } from "./icons"; | ||||
| import { Island } from "./Island"; | ||||
| import "./LayerUI.scss"; | ||||
| import { LibraryUnit } from "./LibraryUnit"; | ||||
| @@ -52,6 +52,8 @@ interface LayerUIProps { | ||||
|   onLockToggle: () => void; | ||||
|   onInsertElements: (elements: readonly NonDeletedExcalidrawElement[]) => void; | ||||
|   zenModeEnabled: boolean; | ||||
|   showExitZenModeBtn: boolean; | ||||
|   showThemeBtn: boolean; | ||||
|   toggleZenMode: () => void; | ||||
|   langCode: Language["code"]; | ||||
|   isCollaborating: boolean; | ||||
| @@ -61,6 +63,8 @@ interface LayerUIProps { | ||||
|     canvas: HTMLCanvasElement | null, | ||||
|   ) => void; | ||||
|   renderCustomFooter?: (isMobile: boolean) => JSX.Element; | ||||
|   viewModeEnabled: boolean; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
| } | ||||
|  | ||||
| const useOnClickOutside = ( | ||||
| @@ -98,6 +102,8 @@ const LibraryMenuItems = ({ | ||||
|   onInsertShape, | ||||
|   pendingElements, | ||||
|   setAppState, | ||||
|   setLibraryItems, | ||||
|   libraryReturnUrl, | ||||
| }: { | ||||
|   library: LibraryItems; | ||||
|   pendingElements: LibraryItem; | ||||
| @@ -105,6 +111,8 @@ const LibraryMenuItems = ({ | ||||
|   onInsertShape: (elements: LibraryItem) => void; | ||||
|   onAddToLibrary: (elements: LibraryItem) => void; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   setLibraryItems: (library: LibraryItems) => void; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
| }) => { | ||||
|   const isMobile = useIsMobile(); | ||||
|   const numCells = library.length + (pendingElements.length > 0 ? 1 : 0); | ||||
| @@ -113,6 +121,8 @@ const LibraryMenuItems = ({ | ||||
|   const rows = []; | ||||
|   let addedPendingElements = false; | ||||
|  | ||||
|   const referrer = libraryReturnUrl || window.location.origin; | ||||
|  | ||||
|   rows.push( | ||||
|     <div className="layer-ui__library-header"> | ||||
|       <ToolButton | ||||
| @@ -148,8 +158,24 @@ const LibraryMenuItems = ({ | ||||
|             }); | ||||
|         }} | ||||
|       /> | ||||
|       <ToolButton | ||||
|         key="reset" | ||||
|         type="button" | ||||
|         title={t("buttons.resetLibrary")} | ||||
|         aria-label={t("buttons.resetLibrary")} | ||||
|         icon={trash} | ||||
|         onClick={() => { | ||||
|           if (window.confirm(t("alerts.resetLibrary"))) { | ||||
|             Library.resetLibrary(); | ||||
|             setLibraryItems([]); | ||||
|           } | ||||
|         }} | ||||
|       /> | ||||
|  | ||||
|       <a href="https://libraries.excalidraw.com" target="_excalidraw_libraries"> | ||||
|       <a | ||||
|         href={`https://libraries.excalidraw.com?referrer=${referrer}`} | ||||
|         target="_excalidraw_libraries" | ||||
|       > | ||||
|         {t("labels.libraries")} | ||||
|       </a> | ||||
|     </div>, | ||||
| @@ -202,12 +228,14 @@ const LibraryMenu = ({ | ||||
|   pendingElements, | ||||
|   onAddToLibrary, | ||||
|   setAppState, | ||||
|   libraryReturnUrl, | ||||
| }: { | ||||
|   pendingElements: LibraryItem; | ||||
|   onClickOutside: (event: MouseEvent) => void; | ||||
|   onInsertShape: (elements: LibraryItem) => void; | ||||
|   onAddToLibrary: () => void; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
| }) => { | ||||
|   const ref = useRef<HTMLDivElement | null>(null); | ||||
|   useOnClickOutside(ref, (event) => { | ||||
| @@ -279,6 +307,8 @@ const LibraryMenu = ({ | ||||
|           onInsertShape={onInsertShape} | ||||
|           pendingElements={pendingElements} | ||||
|           setAppState={setAppState} | ||||
|           setLibraryItems={setLibraryItems} | ||||
|           libraryReturnUrl={libraryReturnUrl} | ||||
|         /> | ||||
|       )} | ||||
|     </Island> | ||||
| @@ -295,10 +325,14 @@ const LayerUI = ({ | ||||
|   onLockToggle, | ||||
|   onInsertElements, | ||||
|   zenModeEnabled, | ||||
|   showExitZenModeBtn, | ||||
|   showThemeBtn, | ||||
|   toggleZenMode, | ||||
|   isCollaborating, | ||||
|   onExportToBackend, | ||||
|   renderCustomFooter, | ||||
|   viewModeEnabled, | ||||
|   libraryReturnUrl, | ||||
| }: LayerUIProps) => { | ||||
|   const isMobile = useIsMobile(); | ||||
|  | ||||
| @@ -358,6 +392,28 @@ const LayerUI = ({ | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderViewModeCanvasActions = () => { | ||||
|     return ( | ||||
|       <Section | ||||
|         heading="canvasActions" | ||||
|         className={clsx("zen-mode-transition", { | ||||
|           "transition-left": zenModeEnabled, | ||||
|         })} | ||||
|       > | ||||
|         {/* the zIndex ensures this menu has higher stacking order, | ||||
|          see https://github.com/excalidraw/excalidraw/pull/1445 */} | ||||
|         <Island padding={2} style={{ zIndex: 1 }}> | ||||
|           <Stack.Col gap={4}> | ||||
|             <Stack.Row gap={1} justifyContent="space-between"> | ||||
|               {actionManager.renderAction("saveScene")} | ||||
|               {actionManager.renderAction("saveAsScene")} | ||||
|               {renderExportDialog()} | ||||
|             </Stack.Row> | ||||
|           </Stack.Col> | ||||
|         </Island> | ||||
|       </Section> | ||||
|     ); | ||||
|   }; | ||||
|   const renderCanvasActions = () => ( | ||||
|     <Section | ||||
|       heading="canvasActions" | ||||
| @@ -387,6 +443,7 @@ const LayerUI = ({ | ||||
|             actionManager={actionManager} | ||||
|             appState={appState} | ||||
|             setAppState={setAppState} | ||||
|             showThemeBtn={showThemeBtn} | ||||
|           /> | ||||
|         </Stack.Col> | ||||
|       </Island> | ||||
| @@ -400,7 +457,15 @@ const LayerUI = ({ | ||||
|         "transition-left": zenModeEnabled, | ||||
|       })} | ||||
|     > | ||||
|       <Island className={CLASSES.SHAPE_ACTIONS_MENU} padding={2}> | ||||
|       <Island | ||||
|         className={CLASSES.SHAPE_ACTIONS_MENU} | ||||
|         padding={2} | ||||
|         style={{ | ||||
|           // we want to make sure this doesn't overflow so substracting 200 | ||||
|           // which is approximately height of zoom footer and top left menu items with some buffer | ||||
|           maxHeight: `${appState.height - 200}px`, | ||||
|         }} | ||||
|       > | ||||
|         <SelectedShapeActions | ||||
|           appState={appState} | ||||
|           elements={elements} | ||||
| @@ -432,6 +497,7 @@ const LayerUI = ({ | ||||
|       onInsertShape={onInsertElements} | ||||
|       onAddToLibrary={deselectItems} | ||||
|       setAppState={setAppState} | ||||
|       libraryReturnUrl={libraryReturnUrl} | ||||
|     /> | ||||
|   ) : null; | ||||
|  | ||||
| @@ -448,54 +514,60 @@ const LayerUI = ({ | ||||
|             gap={4} | ||||
|             className={clsx({ "disable-pointerEvents": zenModeEnabled })} | ||||
|           > | ||||
|             {renderCanvasActions()} | ||||
|             {viewModeEnabled | ||||
|               ? renderViewModeCanvasActions() | ||||
|               : renderCanvasActions()} | ||||
|             {shouldRenderSelectedShapeActions && renderSelectedShapeActions()} | ||||
|           </Stack.Col> | ||||
|           <Section heading="shapes"> | ||||
|             {(heading) => ( | ||||
|               <Stack.Col gap={4} align="start"> | ||||
|                 <Stack.Row gap={1}> | ||||
|                   <Island | ||||
|                     padding={1} | ||||
|                     className={clsx({ "zen-mode": zenModeEnabled })} | ||||
|                   > | ||||
|                     <HintViewer appState={appState} elements={elements} /> | ||||
|                     {heading} | ||||
|                     <Stack.Row gap={1}> | ||||
|                       <ShapesSwitcher | ||||
|                         elementType={appState.elementType} | ||||
|                         setAppState={setAppState} | ||||
|                         isLibraryOpen={appState.isLibraryOpen} | ||||
|                       /> | ||||
|                     </Stack.Row> | ||||
|                   </Island> | ||||
|                   <LockIcon | ||||
|                     zenModeEnabled={zenModeEnabled} | ||||
|                     checked={appState.elementLocked} | ||||
|                     onChange={onLockToggle} | ||||
|                     title={t("toolBar.lock")} | ||||
|                   /> | ||||
|                 </Stack.Row> | ||||
|                 {libraryMenu} | ||||
|               </Stack.Col> | ||||
|             )} | ||||
|           </Section> | ||||
|           {!viewModeEnabled && ( | ||||
|             <Section heading="shapes"> | ||||
|               {(heading) => ( | ||||
|                 <Stack.Col gap={4} align="start"> | ||||
|                   <Stack.Row gap={1}> | ||||
|                     <Island | ||||
|                       padding={1} | ||||
|                       className={clsx({ "zen-mode": zenModeEnabled })} | ||||
|                     > | ||||
|                       <HintViewer appState={appState} elements={elements} /> | ||||
|                       {heading} | ||||
|                       <Stack.Row gap={1}> | ||||
|                         <ShapesSwitcher | ||||
|                           canvas={canvas} | ||||
|                           elementType={appState.elementType} | ||||
|                           setAppState={setAppState} | ||||
|                           isLibraryOpen={appState.isLibraryOpen} | ||||
|                         /> | ||||
|                       </Stack.Row> | ||||
|                     </Island> | ||||
|                     <LockIcon | ||||
|                       zenModeEnabled={zenModeEnabled} | ||||
|                       checked={appState.elementLocked} | ||||
|                       onChange={onLockToggle} | ||||
|                       title={t("toolBar.lock")} | ||||
|                     /> | ||||
|                   </Stack.Row> | ||||
|                   {libraryMenu} | ||||
|                 </Stack.Col> | ||||
|               )} | ||||
|             </Section> | ||||
|           )} | ||||
|           <UserList | ||||
|             className={clsx("zen-mode-transition", { | ||||
|               "transition-right": zenModeEnabled, | ||||
|             })} | ||||
|           > | ||||
|             {Array.from(appState.collaborators) | ||||
|               // Collaborator is either not initialized or is actually the current user. | ||||
|               .filter(([_, client]) => Object.keys(client).length !== 0) | ||||
|               .map(([clientId, client]) => ( | ||||
|                 <Tooltip | ||||
|                   label={client.username || "Unknown user"} | ||||
|                   key={clientId} | ||||
|                 > | ||||
|                   {actionManager.renderAction("goToCollaborator", clientId)} | ||||
|                 </Tooltip> | ||||
|               ))} | ||||
|             {appState.collaborators.size > 0 && | ||||
|               Array.from(appState.collaborators) | ||||
|                 // Collaborator is either not initialized or is actually the current user. | ||||
|                 .filter(([_, client]) => Object.keys(client).length !== 0) | ||||
|                 .map(([clientId, client]) => ( | ||||
|                   <Tooltip | ||||
|                     label={client.username || "Unknown user"} | ||||
|                     key={clientId} | ||||
|                   > | ||||
|                     {actionManager.renderAction("goToCollaborator", clientId)} | ||||
|                   </Tooltip> | ||||
|                 ))} | ||||
|           </UserList> | ||||
|         </div> | ||||
|       </FixedSideContainer> | ||||
| @@ -524,6 +596,20 @@ const LayerUI = ({ | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderGitHubCorner = () => { | ||||
|     return ( | ||||
|       <aside | ||||
|         className={clsx( | ||||
|           "layer-ui__wrapper__github-corner zen-mode-transition", | ||||
|           { | ||||
|             "transition-right": zenModeEnabled, | ||||
|           }, | ||||
|         )} | ||||
|       > | ||||
|         <GitHubCorner theme={appState.theme} /> | ||||
|       </aside> | ||||
|     ); | ||||
|   }; | ||||
|   const renderFooter = () => ( | ||||
|     <footer role="contentinfo" className="layer-ui__wrapper__footer"> | ||||
|       <div | ||||
| @@ -536,24 +622,12 @@ const LayerUI = ({ | ||||
|       </div> | ||||
|       <button | ||||
|         className={clsx("disable-zen-mode", { | ||||
|           "disable-zen-mode--visible": zenModeEnabled, | ||||
|           "disable-zen-mode--visible": showExitZenModeBtn, | ||||
|         })} | ||||
|         onClick={toggleZenMode} | ||||
|       > | ||||
|         {t("buttons.exitZenMode")} | ||||
|       </button> | ||||
|       {appState.scrolledOutside && ( | ||||
|         <button | ||||
|           className="scroll-back-to-content" | ||||
|           onClick={() => { | ||||
|             setAppState({ | ||||
|               ...calculateScrollCenter(elements, appState, canvas), | ||||
|             }); | ||||
|           }} | ||||
|         > | ||||
|           {t("buttons.scrollBackToContent")} | ||||
|         </button> | ||||
|       )} | ||||
|     </footer> | ||||
|   ); | ||||
|  | ||||
| @@ -599,26 +673,36 @@ const LayerUI = ({ | ||||
|         canvas={canvas} | ||||
|         isCollaborating={isCollaborating} | ||||
|         renderCustomFooter={renderCustomFooter} | ||||
|         viewModeEnabled={viewModeEnabled} | ||||
|         showThemeBtn={showThemeBtn} | ||||
|       /> | ||||
|     </> | ||||
|   ) : ( | ||||
|     <div className="layer-ui__wrapper"> | ||||
|     <div | ||||
|       className={clsx("layer-ui__wrapper", { | ||||
|         "disable-pointerEvents": | ||||
|           appState.draggingElement || | ||||
|           appState.resizingElement || | ||||
|           (appState.editingElement && !isTextElement(appState.editingElement)), | ||||
|       })} | ||||
|     > | ||||
|       {dialogs} | ||||
|       {renderFixedSideContainer()} | ||||
|       {renderBottomAppMenu()} | ||||
|       { | ||||
|         <aside | ||||
|           className={clsx( | ||||
|             "layer-ui__wrapper__github-corner zen-mode-transition", | ||||
|             { | ||||
|               "transition-right": zenModeEnabled, | ||||
|             }, | ||||
|           )} | ||||
|         > | ||||
|           <GitHubCorner appearance={appState.appearance} /> | ||||
|         </aside> | ||||
|       } | ||||
|       {renderGitHubCorner()} | ||||
|       {renderFooter()} | ||||
|       {appState.scrolledOutside && ( | ||||
|         <button | ||||
|           className="scroll-back-to-content" | ||||
|           onClick={() => { | ||||
|             setAppState({ | ||||
|               ...calculateScrollCenter(elements, appState, canvas), | ||||
|             }); | ||||
|           }} | ||||
|         > | ||||
|           {t("buttons.scrollBackToContent")} | ||||
|         </button> | ||||
|       )} | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
| @@ -637,6 +721,7 @@ const areEqual = (prev: LayerUIProps, next: LayerUIProps) => { | ||||
|  | ||||
|   const keys = Object.keys(prevAppState) as (keyof Partial<AppState>)[]; | ||||
|   return ( | ||||
|     prev.renderCustomFooter === next.renderCustomFooter && | ||||
|     prev.langCode === next.langCode && | ||||
|     prev.elements === next.elements && | ||||
|     keys.every((key) => prevAppState[key] === nextAppState[key]) | ||||
|   | ||||
| @@ -16,7 +16,7 @@ | ||||
|   } | ||||
|  | ||||
|   .library-unit__dragger > svg { | ||||
|     filter: var(--appearance-filter); | ||||
|     filter: var(--theme-filter); | ||||
|     flex-grow: 1; | ||||
|     max-height: 100%; | ||||
|     max-width: 100%; | ||||
|   | ||||
| @@ -29,6 +29,8 @@ type MobileMenuProps = { | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   isCollaborating: boolean; | ||||
|   renderCustomFooter?: (isMobile: boolean) => JSX.Element; | ||||
|   viewModeEnabled: boolean; | ||||
|   showThemeBtn: boolean; | ||||
| }; | ||||
|  | ||||
| export const MobileMenu = ({ | ||||
| @@ -43,121 +45,169 @@ export const MobileMenu = ({ | ||||
|   canvas, | ||||
|   isCollaborating, | ||||
|   renderCustomFooter, | ||||
| }: MobileMenuProps) => ( | ||||
|   <> | ||||
|     <FixedSideContainer side="top"> | ||||
|       <Section heading="shapes"> | ||||
|         {(heading) => ( | ||||
|           <Stack.Col gap={4} align="center"> | ||||
|             <Stack.Row gap={1}> | ||||
|               <Island padding={1}> | ||||
|                 {heading} | ||||
|                 <Stack.Row gap={1}> | ||||
|                   <ShapesSwitcher | ||||
|                     elementType={appState.elementType} | ||||
|                     setAppState={setAppState} | ||||
|                     isLibraryOpen={appState.isLibraryOpen} | ||||
|                   /> | ||||
|                 </Stack.Row> | ||||
|               </Island> | ||||
|               <LockIcon | ||||
|                 checked={appState.elementLocked} | ||||
|                 onChange={onLockToggle} | ||||
|                 title={t("toolBar.lock")} | ||||
|               /> | ||||
|             </Stack.Row> | ||||
|             {libraryMenu} | ||||
|           </Stack.Col> | ||||
|         )} | ||||
|       </Section> | ||||
|       <HintViewer appState={appState} elements={elements} /> | ||||
|     </FixedSideContainer> | ||||
|     <div | ||||
|       className="App-bottom-bar" | ||||
|       style={{ | ||||
|         marginBottom: SCROLLBAR_WIDTH + SCROLLBAR_MARGIN * 2, | ||||
|         marginLeft: SCROLLBAR_WIDTH + SCROLLBAR_MARGIN * 2, | ||||
|         marginRight: SCROLLBAR_WIDTH + SCROLLBAR_MARGIN * 2, | ||||
|       }} | ||||
|     > | ||||
|       <Island padding={0}> | ||||
|         {appState.openMenu === "canvas" ? ( | ||||
|           <Section className="App-mobile-menu" heading="canvasActions"> | ||||
|             <div className="panelColumn"> | ||||
|               <Stack.Col gap={4}> | ||||
|                 {actionManager.renderAction("loadScene")} | ||||
|                 {actionManager.renderAction("saveScene")} | ||||
|                 {actionManager.renderAction("saveAsScene")} | ||||
|                 {exportButton} | ||||
|                 {actionManager.renderAction("clearCanvas")} | ||||
|                 {onCollabButtonClick && ( | ||||
|                   <CollabButton | ||||
|                     isCollaborating={isCollaborating} | ||||
|                     collaboratorCount={appState.collaborators.size} | ||||
|                     onClick={onCollabButtonClick} | ||||
|                   /> | ||||
|                 )} | ||||
|                 <BackgroundPickerAndDarkModeToggle | ||||
|                   actionManager={actionManager} | ||||
|                   appState={appState} | ||||
|                   setAppState={setAppState} | ||||
|   viewModeEnabled, | ||||
|   showThemeBtn, | ||||
| }: MobileMenuProps) => { | ||||
|   const renderToolbar = () => { | ||||
|     return ( | ||||
|       <FixedSideContainer side="top" className="App-top-bar"> | ||||
|         <Section heading="shapes"> | ||||
|           {(heading) => ( | ||||
|             <Stack.Col gap={4} align="center"> | ||||
|               <Stack.Row gap={1}> | ||||
|                 <Island padding={1}> | ||||
|                   {heading} | ||||
|                   <Stack.Row gap={1}> | ||||
|                     <ShapesSwitcher | ||||
|                       canvas={canvas} | ||||
|                       elementType={appState.elementType} | ||||
|                       setAppState={setAppState} | ||||
|                       isLibraryOpen={appState.isLibraryOpen} | ||||
|                     /> | ||||
|                   </Stack.Row> | ||||
|                 </Island> | ||||
|                 <LockIcon | ||||
|                   checked={appState.elementLocked} | ||||
|                   onChange={onLockToggle} | ||||
|                   title={t("toolBar.lock")} | ||||
|                 /> | ||||
|                 {renderCustomFooter?.(true)} | ||||
|                 <fieldset> | ||||
|                   <legend>{t("labels.collaborators")}</legend> | ||||
|                   <UserList mobile> | ||||
|                     {Array.from(appState.collaborators) | ||||
|                       // Collaborator is either not initialized or is actually the current user. | ||||
|                       .filter(([_, client]) => Object.keys(client).length !== 0) | ||||
|                       .map(([clientId, client]) => ( | ||||
|                         <React.Fragment key={clientId}> | ||||
|                           {actionManager.renderAction( | ||||
|                             "goToCollaborator", | ||||
|                             clientId, | ||||
|                           )} | ||||
|                         </React.Fragment> | ||||
|                       ))} | ||||
|                   </UserList> | ||||
|                 </fieldset> | ||||
|               </Stack.Col> | ||||
|             </div> | ||||
|           </Section> | ||||
|         ) : appState.openMenu === "shape" && | ||||
|           showSelectedShapeActions(appState, elements) ? ( | ||||
|           <Section className="App-mobile-menu" heading="selectedShapeActions"> | ||||
|             <SelectedShapeActions | ||||
|               appState={appState} | ||||
|               elements={elements} | ||||
|               renderAction={actionManager.renderAction} | ||||
|               elementType={appState.elementType} | ||||
|             /> | ||||
|           </Section> | ||||
|         ) : null} | ||||
|         <footer className="App-toolbar"> | ||||
|           <div className="App-toolbar-content"> | ||||
|             {actionManager.renderAction("toggleCanvasMenu")} | ||||
|             {actionManager.renderAction("toggleEditMenu")} | ||||
|             {actionManager.renderAction("undo")} | ||||
|             {actionManager.renderAction("redo")} | ||||
|             {actionManager.renderAction( | ||||
|               appState.multiElement ? "finalize" : "duplicateSelection", | ||||
|             )} | ||||
|             {actionManager.renderAction("deleteSelectedElements")} | ||||
|           </div> | ||||
|           {appState.scrolledOutside && !appState.openMenu && ( | ||||
|             <button | ||||
|               className="scroll-back-to-content" | ||||
|               onClick={() => { | ||||
|                 setAppState({ | ||||
|                   ...calculateScrollCenter(elements, appState, canvas), | ||||
|                 }); | ||||
|               }} | ||||
|             > | ||||
|               {t("buttons.scrollBackToContent")} | ||||
|             </button> | ||||
|               </Stack.Row> | ||||
|               {libraryMenu} | ||||
|             </Stack.Col> | ||||
|           )} | ||||
|         </footer> | ||||
|       </Island> | ||||
|     </div> | ||||
|   </> | ||||
| ); | ||||
|         </Section> | ||||
|         <HintViewer appState={appState} elements={elements} /> | ||||
|       </FixedSideContainer> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderAppToolbar = () => { | ||||
|     if (viewModeEnabled) { | ||||
|       return ( | ||||
|         <div className="App-toolbar-content"> | ||||
|           {actionManager.renderAction("toggleCanvasMenu")} | ||||
|         </div> | ||||
|       ); | ||||
|     } | ||||
|     return ( | ||||
|       <div className="App-toolbar-content"> | ||||
|         {actionManager.renderAction("toggleCanvasMenu")} | ||||
|         {actionManager.renderAction("toggleEditMenu")} | ||||
|         {actionManager.renderAction("undo")} | ||||
|         {actionManager.renderAction("redo")} | ||||
|         {actionManager.renderAction( | ||||
|           appState.multiElement ? "finalize" : "duplicateSelection", | ||||
|         )} | ||||
|         {actionManager.renderAction("deleteSelectedElements")} | ||||
|       </div> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderCanvasActions = () => { | ||||
|     if (viewModeEnabled) { | ||||
|       return ( | ||||
|         <> | ||||
|           {actionManager.renderAction("saveScene")} | ||||
|           {actionManager.renderAction("saveAsScene")} | ||||
|           {exportButton} | ||||
|         </> | ||||
|       ); | ||||
|     } | ||||
|     return ( | ||||
|       <> | ||||
|         {actionManager.renderAction("loadScene")} | ||||
|         {actionManager.renderAction("saveScene")} | ||||
|         {actionManager.renderAction("saveAsScene")} | ||||
|         {exportButton} | ||||
|         {actionManager.renderAction("clearCanvas")} | ||||
|         {onCollabButtonClick && ( | ||||
|           <CollabButton | ||||
|             isCollaborating={isCollaborating} | ||||
|             collaboratorCount={appState.collaborators.size} | ||||
|             onClick={onCollabButtonClick} | ||||
|           /> | ||||
|         )} | ||||
|         { | ||||
|           <BackgroundPickerAndDarkModeToggle | ||||
|             actionManager={actionManager} | ||||
|             appState={appState} | ||||
|             setAppState={setAppState} | ||||
|             showThemeBtn={showThemeBtn} | ||||
|           /> | ||||
|         } | ||||
|       </> | ||||
|     ); | ||||
|   }; | ||||
|   return ( | ||||
|     <> | ||||
|       {!viewModeEnabled && renderToolbar()} | ||||
|       <div | ||||
|         className="App-bottom-bar" | ||||
|         style={{ | ||||
|           marginBottom: SCROLLBAR_WIDTH + SCROLLBAR_MARGIN * 2, | ||||
|           marginLeft: SCROLLBAR_WIDTH + SCROLLBAR_MARGIN * 2, | ||||
|           marginRight: SCROLLBAR_WIDTH + SCROLLBAR_MARGIN * 2, | ||||
|         }} | ||||
|       > | ||||
|         <Island padding={0}> | ||||
|           {appState.openMenu === "canvas" ? ( | ||||
|             <Section className="App-mobile-menu" heading="canvasActions"> | ||||
|               <div className="panelColumn"> | ||||
|                 <Stack.Col gap={4}> | ||||
|                   {renderCanvasActions()} | ||||
|                   {renderCustomFooter?.(true)} | ||||
|                   {appState.collaborators.size > 0 && ( | ||||
|                     <fieldset> | ||||
|                       <legend>{t("labels.collaborators")}</legend> | ||||
|                       <UserList mobile> | ||||
|                         {Array.from(appState.collaborators) | ||||
|                           // Collaborator is either not initialized or is actually the current user. | ||||
|                           .filter( | ||||
|                             ([_, client]) => Object.keys(client).length !== 0, | ||||
|                           ) | ||||
|                           .map(([clientId, client]) => ( | ||||
|                             <React.Fragment key={clientId}> | ||||
|                               {actionManager.renderAction( | ||||
|                                 "goToCollaborator", | ||||
|                                 clientId, | ||||
|                               )} | ||||
|                             </React.Fragment> | ||||
|                           ))} | ||||
|                       </UserList> | ||||
|                     </fieldset> | ||||
|                   )} | ||||
|                 </Stack.Col> | ||||
|               </div> | ||||
|             </Section> | ||||
|           ) : appState.openMenu === "shape" && | ||||
|             !viewModeEnabled && | ||||
|             showSelectedShapeActions(appState, elements) ? ( | ||||
|             <Section className="App-mobile-menu" heading="selectedShapeActions"> | ||||
|               <SelectedShapeActions | ||||
|                 appState={appState} | ||||
|                 elements={elements} | ||||
|                 renderAction={actionManager.renderAction} | ||||
|                 elementType={appState.elementType} | ||||
|               /> | ||||
|             </Section> | ||||
|           ) : null} | ||||
|           <footer className="App-toolbar"> | ||||
|             {renderAppToolbar()} | ||||
|             {appState.scrolledOutside && !appState.openMenu && ( | ||||
|               <button | ||||
|                 className="scroll-back-to-content" | ||||
|                 onClick={() => { | ||||
|                   setAppState({ | ||||
|                     ...calculateScrollCenter(elements, appState, canvas), | ||||
|                   }); | ||||
|                 }} | ||||
|               > | ||||
|                 {t("buttons.scrollBackToContent")} | ||||
|               </button> | ||||
|             )} | ||||
|           </footer> | ||||
|         </Island> | ||||
|       </div> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -1,8 +1,13 @@ | ||||
| @import "../css/variables.module"; | ||||
|  | ||||
| .excalidraw { | ||||
|   &.excalidraw-modal-container { | ||||
|     position: absolute; | ||||
|     z-index: 10; | ||||
|   } | ||||
|  | ||||
|   .Modal { | ||||
|     position: fixed; | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
|     right: 0; | ||||
| @@ -15,7 +20,7 @@ | ||||
|   } | ||||
|  | ||||
|   .Modal__background { | ||||
|     position: fixed; | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
|     right: 0; | ||||
| @@ -42,9 +47,10 @@ | ||||
|     background: var(--island-bg-color); | ||||
|     backdrop-filter: none; | ||||
|  | ||||
|     border: 1px solid var(--dialog-border); | ||||
|     border: 1px solid var(--dialog-border-color); | ||||
|     box-shadow: 0 2px 10px transparentize($oc-black, 0.75); | ||||
|     border-radius: 6px; | ||||
|     box-sizing: border-box; | ||||
|  | ||||
|     @media #{$is-mobile-query} { | ||||
|       max-width: 100%; | ||||
| @@ -82,7 +88,7 @@ | ||||
|     } | ||||
|  | ||||
|     .Modal__content { | ||||
|       position: fixed; | ||||
|       position: absolute; | ||||
|       top: 0; | ||||
|       left: 0; | ||||
|       right: 0; | ||||
|   | ||||
| @@ -51,14 +51,14 @@ const useBodyRoot = () => { | ||||
|   useLayoutEffect(() => { | ||||
|     const isDarkTheme = !!document | ||||
|       .querySelector(".excalidraw") | ||||
|       ?.classList.contains("Appearance_dark"); | ||||
|       ?.classList.contains("theme--dark"); | ||||
|     const div = document.createElement("div"); | ||||
|  | ||||
|     div.classList.add("excalidraw"); | ||||
|     div.classList.add("excalidraw", "excalidraw-modal-container"); | ||||
|  | ||||
|     if (isDarkTheme) { | ||||
|       div.classList.add("Appearance_dark"); | ||||
|       div.classList.add("Appearance_dark-background-none"); | ||||
|       div.classList.add("theme--dark"); | ||||
|       div.classList.add("theme--dark-background-none"); | ||||
|     } | ||||
|     document.body.appendChild(div); | ||||
|  | ||||
|   | ||||
| @@ -1,14 +1,6 @@ | ||||
| .excalidraw { | ||||
|   .popover { | ||||
|     position: absolute; | ||||
|     position: fixed; | ||||
|     z-index: 10; | ||||
|   } | ||||
|  | ||||
|   .popover .cover { | ||||
|     position: fixed; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
|     right: 0; | ||||
|     bottom: 0; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -7,6 +7,7 @@ type Props = { | ||||
|   value: string; | ||||
|   onChange: (value: string) => void; | ||||
|   label: string; | ||||
|   isNameEditable: boolean; | ||||
| }; | ||||
|  | ||||
| export class ProjectName extends Component<Props> { | ||||
| @@ -43,7 +44,7 @@ export class ProjectName extends Component<Props> { | ||||
|   }; | ||||
|  | ||||
|   public render() { | ||||
|     return ( | ||||
|     return this.props.isNameEditable ? ( | ||||
|       <span | ||||
|         suppressContentEditableWarning | ||||
|         ref={this.makeEditable} | ||||
| @@ -57,6 +58,13 @@ export class ProjectName extends Component<Props> { | ||||
|       > | ||||
|         {this.props.value} | ||||
|       </span> | ||||
|     ) : ( | ||||
|       <span | ||||
|         className="TextInput TextInput--readonly" | ||||
|         aria-label={this.props.label} | ||||
|       > | ||||
|         {this.props.value} | ||||
|       </span> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| .excalidraw { | ||||
|   .Stats { | ||||
|     position: fixed; | ||||
|     position: absolute; | ||||
|     top: 64px; | ||||
|     right: 12px; | ||||
|     font-size: 12px; | ||||
|   | ||||
| @@ -1,4 +1,6 @@ | ||||
| import React, { useEffect, useState } from "react"; | ||||
| import { copyTextToSystemClipboard } from "../clipboard"; | ||||
| import { DEFAULT_VERSION } from "../constants"; | ||||
| import { getCommonBounds } from "../element/bounds"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
| import { | ||||
| @@ -9,7 +11,7 @@ import { t } from "../i18n"; | ||||
| import useIsMobile from "../is-mobile"; | ||||
| import { getTargetElements } from "../scene"; | ||||
| import { AppState } from "../types"; | ||||
| import { debounce, nFormatter } from "../utils"; | ||||
| import { debounce, getVersion, nFormatter } from "../utils"; | ||||
| import { close } from "./icons"; | ||||
| import { Island } from "./Island"; | ||||
| import "./Stats.scss"; | ||||
| @@ -25,6 +27,7 @@ const getStorageSizes = debounce((cb: (sizes: StorageSizes) => void) => { | ||||
|  | ||||
| export const Stats = (props: { | ||||
|   appState: AppState; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   onClose: () => void; | ||||
| }) => { | ||||
| @@ -50,6 +53,17 @@ export const Stats = (props: { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   const version = getVersion(); | ||||
|   let hash; | ||||
|   let timestamp; | ||||
|  | ||||
|   if (version !== DEFAULT_VERSION) { | ||||
|     timestamp = version.slice(0, 16).replace("T", " "); | ||||
|     hash = version.slice(21); | ||||
|   } else { | ||||
|     timestamp = t("stats.versionNotAvailable"); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <div className="Stats"> | ||||
|       <Island padding={2}> | ||||
| @@ -156,6 +170,28 @@ export const Stats = (props: { | ||||
|                 </td> | ||||
|               </tr> | ||||
|             )} | ||||
|             <tr> | ||||
|               <th colSpan={2}>{t("stats.version")}</th> | ||||
|             </tr> | ||||
|             <tr> | ||||
|               <td | ||||
|                 colSpan={2} | ||||
|                 style={{ textAlign: "center", cursor: "pointer" }} | ||||
|                 onClick={async () => { | ||||
|                   try { | ||||
|                     await copyTextToSystemClipboard(getVersion()); | ||||
|                     props.setAppState({ | ||||
|                       toastMessage: t("toast.copyToClipboard"), | ||||
|                     }); | ||||
|                   } catch {} | ||||
|                 }} | ||||
|                 title={t("stats.versionCopy")} | ||||
|               > | ||||
|                 {timestamp} | ||||
|                 <br /> | ||||
|                 {hash} | ||||
|               </td> | ||||
|             </tr> | ||||
|           </tbody> | ||||
|         </table> | ||||
|       </Island> | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| .excalidraw { | ||||
|   .TextInput { | ||||
|     color: var(--text-color-primary); | ||||
|     color: var(--text-primary-color); | ||||
|     display: inline-block; | ||||
|     border: 1.5px solid var(--button-gray-1); | ||||
|     line-height: 1; | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
|     left: 50%; | ||||
|     margin-left: -150px; | ||||
|     padding: 4px 0; | ||||
|     position: fixed; | ||||
|     position: absolute; | ||||
|     text-align: center; | ||||
|     width: 300px; | ||||
|     z-index: 999999; | ||||
| @@ -19,6 +19,7 @@ | ||||
|  | ||||
|   .Toast__message { | ||||
|     color: var(--popup-text-color); | ||||
|     white-space: pre-wrap; | ||||
|   } | ||||
|  | ||||
|   @keyframes fade-in { | ||||
|   | ||||
| @@ -58,6 +58,7 @@ export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => { | ||||
|             "ToolIcon--selected": props.selected, | ||||
|           }, | ||||
|         )} | ||||
|         data-testid={props["data-testid"]} | ||||
|         hidden={props.hidden} | ||||
|         title={props.title} | ||||
|         aria-label={props["aria-label"]} | ||||
|   | ||||
| @@ -48,15 +48,7 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // the following 3 rules ensure that the tooltip doesn't show (nor affect | ||||
|   // the cursor) when you drag over when you draw on canvas, but at the same | ||||
|   // time it still works when clicking on the link/shield | ||||
|  | ||||
|   body:active & .Tooltip:not(:hover) { | ||||
|     pointer-events: none; | ||||
|   } | ||||
|  | ||||
|   body:not(:active) & .Tooltip:hover .Tooltip__label { | ||||
|   .Tooltip:hover .Tooltip__label { | ||||
|     visibility: visible; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import React from "react"; | ||||
| import * as Sentry from "@sentry/browser"; | ||||
| import { resetCursor } from "../utils"; | ||||
| import { t } from "../i18n"; | ||||
|  | ||||
| interface TopErrorBoundaryState { | ||||
| @@ -24,7 +23,6 @@ export class TopErrorBoundary extends React.Component< | ||||
|   } | ||||
|  | ||||
|   componentDidCatch(error: Error, errorInfo: any) { | ||||
|     resetCursor(); | ||||
|     const _localStorage: any = {}; | ||||
|     for (const [key, value] of Object.entries({ ...localStorage })) { | ||||
|       try { | ||||
|   | ||||
| @@ -11,12 +11,12 @@ import React from "react"; | ||||
| import oc from "open-color"; | ||||
| import clsx from "clsx"; | ||||
|  | ||||
| const activeElementColor = (appearance: "light" | "dark") => | ||||
|   appearance === "light" ? oc.orange[4] : oc.orange[9]; | ||||
| const iconFillColor = (appearance: "light" | "dark") => | ||||
|   appearance === "light" ? oc.black : oc.gray[4]; | ||||
| const handlerColor = (appearance: "light" | "dark") => | ||||
|   appearance === "light" ? oc.white : "#1e1e1e"; | ||||
| const activeElementColor = (theme: "light" | "dark") => | ||||
|   theme === "light" ? oc.orange[4] : oc.orange[9]; | ||||
| const iconFillColor = (theme: "light" | "dark") => | ||||
|   theme === "light" ? oc.black : oc.gray[4]; | ||||
| const handlerColor = (theme: "light" | "dark") => | ||||
|   theme === "light" ? oc.white : "#1e1e1e"; | ||||
|  | ||||
| type Opts = { | ||||
|   width?: number; | ||||
| @@ -113,6 +113,16 @@ export const questionCircle = createIcon( | ||||
|   { mirror: true }, | ||||
| ); | ||||
|  | ||||
| export const share = createIcon( | ||||
|   "M18 16.08c-.76 0-1.44.3-1.96.77L8.91 12.7c.05-.23.09-.46.09-.7s-.04-.47-.09-.7l7.05-4.11c.54.5 1.25.81 2.04.81 1.66 0 3-1.34 3-3s-1.34-3-3-3-3 1.34-3 3c0 .24.04.47.09.7L8.04 9.81C7.5 9.31 6.79 9 6 9c-1.66 0-3 1.34-3 3s1.34 3 3 3c.79 0 1.5-.31 2.04-.81l7.12 4.16c-.05.21-.08.43-.08.65 0 1.61 1.31 2.92 2.92 2.92 1.61 0 2.92-1.31 2.92-2.92s-1.31-2.92-2.92-2.92z", | ||||
|   { width: 24, height: 24 }, | ||||
| ); | ||||
|  | ||||
| export const shareIOS = createIcon( | ||||
|   "M16 5l-1.42 1.42-1.59-1.59V16h-1.98V4.83L9.42 6.42 8 5l4-4 4 4zm4 5v11c0 1.1-.9 2-2 2H6c-1.11 0-2-.9-2-2V10c0-1.11.89-2 2-2h3v2H6v11h12V10h-3V8h3c1.1 0 2 .89 2 2z", | ||||
|   { width: 24, height: 24 }, | ||||
| ); | ||||
|  | ||||
| // Icon imported form Storybook | ||||
| // Storybook is licensed under MIT https://github.com/storybookjs/storybook/blob/next/LICENSE | ||||
| export const resetZoom = createIcon( | ||||
| @@ -126,19 +136,19 @@ export const resetZoom = createIcon( | ||||
| ); | ||||
|  | ||||
| export const BringForwardIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M22 9.556C22 8.696 21.303 8 20.444 8H16v8H8v4.444C8 21.304 8.697 22 9.556 22h10.888c.86 0 1.556-.697 1.556-1.556V9.556z" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path | ||||
|           d="M16 3.556C16 2.696 15.303 2 14.444 2H3.556C2.696 2 2 2.697 2 3.556v10.888C2 15.304 2.697 16 3.556 16h10.888c.86 0 1.556-.697 1.556-1.556V3.556z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -147,19 +157,19 @@ export const BringForwardIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const SendBackwardIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M16 3.556C16 2.696 15.303 2 14.444 2H3.556C2.696 2 2 2.697 2 3.556v10.888C2 15.304 2.697 16 3.556 16h10.888c.86 0 1.556-.697 1.556-1.556V3.556z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path | ||||
|           d="M22 9.556C22 8.696 21.303 8 20.444 8H9.556C8.696 8 8 8.697 8 9.556v10.888C8 21.304 8.697 22 9.556 22h10.888c.86 0 1.556-.697 1.556-1.556V9.556z" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -168,19 +178,19 @@ export const SendBackwardIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const BringToFrontIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M13 21a1 1 0 001 1h7a1 1 0 001-1v-7a1 1 0 00-1-1h-3v5h-5v3zM11 3a1 1 0 00-1-1H3a1 1 0 00-1 1v7a1 1 0 001 1h3V6h5V3z" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path | ||||
|           d="M18 7.333C18 6.597 17.403 6 16.667 6H7.333C6.597 6 6 6.597 6 7.333v9.334C6 17.403 6.597 18 7.333 18h9.334c.736 0 1.333-.597 1.333-1.333V7.333z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -189,20 +199,20 @@ export const BringToFrontIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const SendToBackIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M18 7.333C18 6.597 17.403 6 16.667 6H7.333C6.597 6 6 6.597 6 7.333v9.334C6 17.403 6.597 18 7.333 18h9.334c.736 0 1.333-.597 1.333-1.333V7.333z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeLinejoin="round" | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path | ||||
|           d="M11 3a1 1 0 00-1-1H3a1 1 0 00-1 1v7a1 1 0 001 1h8V3zM22 14a1 1 0 00-1-1h-7a1 1 0 00-1 1v7a1 1 0 001 1h8v-8z" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeLinejoin="round" | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
| @@ -218,20 +228,20 @@ export const SendToBackIcon = React.memo( | ||||
| // that would make them lie about their function. | ||||
| // | ||||
| export const AlignTopIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M 2,5 H 22" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeLinecap="round" | ||||
|         /> | ||||
|         <path | ||||
|           d="M 6,7 C 5.446,7 5,7.446 5,8 v 9.999992 c 0,0.554 0.446,1 1,1 h 3.0000001 c 0.554,0 0.9999999,-0.446 0.9999999,-1 V 8 C 10,7.446 9.5540001,7 9.0000001,7 Z m 9,0 c -0.554,0 -1,0.446 -1,1 v 5.999992 c 0,0.554 0.446,1 1,1 h 3 c 0.554,0 1,-0.446 1,-1 V 8 C 19,7.446 18.554,7 18,7 Z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -240,20 +250,20 @@ export const AlignTopIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const AlignBottomIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M 2,19 H 22" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeLinecap="round" | ||||
|         /> | ||||
|         <path | ||||
|           d="m 6,16.999992 c -0.554,0 -1,-0.446 -1,-1 V 6 C 5,5.446 5.446,5 6,5 H 9.0000001 C 9.5540001,5 10,5.446 10,6 v 9.999992 c 0,0.554 -0.4459999,1 -0.9999999,1 z m 9,0 c -0.554,0 -1,-0.446 -1,-1 V 10 c 0,-0.554 0.446,-1 1,-1 h 3 c 0.554,0 1,0.446 1,1 v 5.999992 c 0,0.554 -0.446,1 -1,1 z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -262,20 +272,20 @@ export const AlignBottomIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const AlignLeftIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M 5,2 V 22" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeLinecap="round" | ||||
|         /> | ||||
|         <path | ||||
|           d="m 7.000004,5.999996 c 0,-0.554 0.446,-1 1,-1 h 9.999992 c 0.554,0 1,0.446 1,1 v 3.0000001 c 0,0.554 -0.446,0.9999999 -1,0.9999999 H 8.000004 c -0.554,0 -1,-0.4459999 -1,-0.9999999 z m 0,9 c 0,-0.554 0.446,-1 1,-1 h 5.999992 c 0.554,0 1,0.446 1,1 v 3 c 0,0.554 -0.446,1 -1,1 H 8.000004 c -0.554,0 -1,-0.446 -1,-1 z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -284,20 +294,20 @@ export const AlignLeftIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const AlignRightIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M 19,2 V 22" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeLinecap="round" | ||||
|         /> | ||||
|         <path | ||||
|           d="m 16.999996,5.999996 c 0,-0.554 -0.446,-1 -1,-1 H 6.000004 c -0.554,0 -1,0.446 -1,1 v 3.0000001 c 0,0.554 0.446,0.9999999 1,0.9999999 h 9.999992 c 0.554,0 1,-0.4459999 1,-0.9999999 z m 0,9 c 0,-0.554 -0.446,-1 -1,-1 h -5.999992 c -0.554,0 -1,0.446 -1,1 v 3 c 0,0.554 0.446,1 1,1 h 5.999992 c 0.554,0 1,-0.446 1,-1 z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -306,20 +316,20 @@ export const AlignRightIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const DistributeHorizontallyIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path d="M5 5V19Z" fill="black" /> | ||||
|         <path | ||||
|           d="M19 5V19M5 5V19" | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeLinecap="round" | ||||
|         /> | ||||
|         <path | ||||
|           d="M15 9C15.554 9 16 9.446 16 10V14C16 14.554 15.554 15 15 15H9C8.446 15 8 14.554 8 14V10C8 9.446 8.446 9 9 9H15Z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -336,20 +346,20 @@ export const DistributeHorizontallyIcon = React.memo( | ||||
| ></svg>; | ||||
|  | ||||
| export const DistributeVerticallyIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M5 5L19 5M5 19H19" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeLinecap="round" | ||||
|         /> | ||||
|         <path | ||||
|           d="M15 9C15.554 9 16 9.446 16 10V14C16 14.554 15.554 15 15 15H9C8.446 15 8 14.554 8 14V10C8 9.446 8.446 9 9 9H15Z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -358,19 +368,19 @@ export const DistributeVerticallyIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const CenterVerticallyIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="m 5.000004,16.999996 c 0,0.554 0.446,1 1,1 h 3 c 0.554,0 1,-0.446 1,-1 v -10 c 0,-0.554 -0.446,-1 -1,-1 h -3 c -0.554,0 -1,0.446 -1,1 z m 9,-2 c 0,0.554 0.446,1 1,1 h 3 c 0.554,0 1,-0.446 1,-1 v -6 c 0,-0.554 -0.446,-1 -1,-1 h -3 c -0.554,0 -1,0.446 -1,1 z" | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path | ||||
|           d="M 2,12 H 22" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeDasharray="1, 2.8" | ||||
|           strokeLinecap="round" | ||||
| @@ -381,19 +391,19 @@ export const CenterVerticallyIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const CenterHorizontallyIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path | ||||
|           d="M 7 5 C 6.446 5 6 5.446 6 6 L 6 9 C 6 9.554 6.446 10 7 10 L 17 10 C 17.554 10 18 9.554 18 9 L 18 6 C 18 5.446 17.554 5 17 5 L 7 5 z M 9 14 C 8.446 14 8 14.446 8 15 L 8 18 C 8 18.554 8.446 19 9 19 L 15 19 C 15.554 19 16 18.554 16 18 L 16 15 C 16 14.446 15.554 14 15 14 L 9 14 z " | ||||
|           fill={activeElementColor(appearance)} | ||||
|           stroke={activeElementColor(appearance)} | ||||
|           fill={activeElementColor(theme)} | ||||
|           stroke={activeElementColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path | ||||
|           d="M 12,2 V 22" | ||||
|           fill={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={iconFillColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|           strokeDasharray="1, 2.8" | ||||
|           strokeLinecap="round" | ||||
| @@ -438,77 +448,76 @@ export const shield = createIcon( | ||||
|   { width: 24 }, | ||||
| ); | ||||
|  | ||||
| export const GroupIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path d="M25 26H111V111H25" fill={iconFillColor(appearance)} /> | ||||
|         <path | ||||
|           d="M25 111C25 80.2068 25 49.4135 25 26M25 26C48.6174 26 72.2348 26 111 26H25ZM25 26C53.3671 26 81.7343 26 111 26H25ZM111 26C111 52.303 111 78.606 111 111V26ZM111 26C111 51.2947 111 76.5893 111 111V26ZM111 111C87.0792 111 63.1585 111 25 111H111ZM111 111C87.4646 111 63.9293 111 25 111H111ZM25 111C25 81.1514 25 51.3028 25 26V111Z" | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path d="M100 100H160V160H100" fill={iconFillColor(appearance)} /> | ||||
|         <path | ||||
|           d="M100 160C100 144.106 100 128.211 100 100M100 100C117.706 100 135.412 100 160 100H100ZM100 100C114.214 100 128.428 100 160 100H100ZM160 100C160 120.184 160 140.369 160 160V100ZM160 100C160 113.219 160 126.437 160 160V100ZM160 160C145.534 160 131.068 160 100 160H160ZM160 160C143.467 160 126.934 160 100 160H160ZM100 160C100 143.661 100 127.321 100 100V160Z" | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <rect | ||||
|           x="2.5" | ||||
|           y="2.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
|           x="2.5" | ||||
|           y="149.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
|           x="147.5" | ||||
|           y="149.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
|           x="147.5" | ||||
|           y="2.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|       </>, | ||||
|       { width: 182, height: 182, mirror: true }, | ||||
|     ), | ||||
| export const GroupIcon = React.memo(({ theme }: { theme: "light" | "dark" }) => | ||||
|   createIcon( | ||||
|     <> | ||||
|       <path d="M25 26H111V111H25" fill={iconFillColor(theme)} /> | ||||
|       <path | ||||
|         d="M25 111C25 80.2068 25 49.4135 25 26M25 26C48.6174 26 72.2348 26 111 26H25ZM25 26C53.3671 26 81.7343 26 111 26H25ZM111 26C111 52.303 111 78.606 111 111V26ZM111 26C111 51.2947 111 76.5893 111 111V26ZM111 111C87.0792 111 63.1585 111 25 111H111ZM111 111C87.4646 111 63.9293 111 25 111H111ZM25 111C25 81.1514 25 51.3028 25 26V111Z" | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth="2" | ||||
|       /> | ||||
|       <path d="M100 100H160V160H100" fill={iconFillColor(theme)} /> | ||||
|       <path | ||||
|         d="M100 160C100 144.106 100 128.211 100 100M100 100C117.706 100 135.412 100 160 100H100ZM100 100C114.214 100 128.428 100 160 100H100ZM160 100C160 120.184 160 140.369 160 160V100ZM160 100C160 113.219 160 126.437 160 160V100ZM160 160C145.534 160 131.068 160 100 160H160ZM160 160C143.467 160 126.934 160 100 160H160ZM100 160C100 143.661 100 127.321 100 100V160Z" | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth="2" | ||||
|       /> | ||||
|       <rect | ||||
|         x="2.5" | ||||
|         y="2.5" | ||||
|         width="30" | ||||
|         height="30" | ||||
|         fill={handlerColor(theme)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth="6" | ||||
|       /> | ||||
|       <rect | ||||
|         x="2.5" | ||||
|         y="149.5" | ||||
|         width="30" | ||||
|         height="30" | ||||
|         fill={handlerColor(theme)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth="6" | ||||
|       /> | ||||
|       <rect | ||||
|         x="147.5" | ||||
|         y="149.5" | ||||
|         width="30" | ||||
|         height="30" | ||||
|         fill={handlerColor(theme)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth="6" | ||||
|       /> | ||||
|       <rect | ||||
|         x="147.5" | ||||
|         y="2.5" | ||||
|         width="30" | ||||
|         height="30" | ||||
|         fill={handlerColor(theme)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth="6" | ||||
|       /> | ||||
|     </>, | ||||
|     { width: 182, height: 182, mirror: true }, | ||||
|   ), | ||||
| ); | ||||
|  | ||||
| export const UngroupIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <> | ||||
|         <path d="M25 26H111V111H25" fill={iconFillColor(appearance)} /> | ||||
|         <path d="M25 26H111V111H25" fill={iconFillColor(theme)} /> | ||||
|         <path | ||||
|           d="M25 111C25 80.2068 25 49.4135 25 26M25 26C48.6174 26 72.2348 26 111 26H25ZM25 26C53.3671 26 81.7343 26 111 26H25ZM111 26C111 52.303 111 78.606 111 111V26ZM111 26C111 51.2947 111 76.5893 111 111V26ZM111 111C87.0792 111 63.1585 111 25 111H111ZM111 111C87.4646 111 63.9293 111 25 111H111ZM25 111C25 81.1514 25 51.3028 25 26V111Z" | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <path d="M100 100H160V160H100" fill={iconFillColor(appearance)} /> | ||||
|         <path d="M100 100H160V160H100" fill={iconFillColor(theme)} /> | ||||
|         <path | ||||
|           d="M100 160C100 144.106 100 128.211 100 100M100 100C117.706 100 135.412 100 160 100H100ZM100 100C114.214 100 128.428 100 160 100H100ZM160 100C160 120.184 160 140.369 160 160V100ZM160 100C160 113.219 160 126.437 160 160V100ZM160 160C145.534 160 131.068 160 100 160H160ZM160 160C143.467 160 126.934 160 100 160H160ZM100 160C100 143.661 100 127.321 100 100V160Z" | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="2" | ||||
|         /> | ||||
|         <rect | ||||
| @@ -516,8 +525,8 @@ export const UngroupIcon = React.memo( | ||||
|           y="2.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={handlerColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
| @@ -525,8 +534,8 @@ export const UngroupIcon = React.memo( | ||||
|           y="149.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={handlerColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
| @@ -534,8 +543,8 @@ export const UngroupIcon = React.memo( | ||||
|           y="149.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={handlerColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
| @@ -543,8 +552,8 @@ export const UngroupIcon = React.memo( | ||||
|           y="78.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={handlerColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
| @@ -552,8 +561,8 @@ export const UngroupIcon = React.memo( | ||||
|           y="2.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={handlerColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|         <rect | ||||
| @@ -561,8 +570,8 @@ export const UngroupIcon = React.memo( | ||||
|           y="102.5" | ||||
|           width="30" | ||||
|           height="30" | ||||
|           fill={handlerColor(appearance)} | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           fill={handlerColor(theme)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth="6" | ||||
|         /> | ||||
|       </>, | ||||
| @@ -571,22 +580,22 @@ export const UngroupIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const FillHachureIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         fillRule="evenodd" | ||||
|         clipRule="evenodd" | ||||
|         d="M20.101 16H28.0934L36 8.95989V4H33.5779L20.101 16ZM30.5704 4L17.0935 16H9.10101L22.5779 4H30.5704ZM19.5704 4L6.09349 16H4V10.7475L11.5779 4H19.5704ZM8.57036 4H4V8.06952L8.57036 4ZM36 11.6378L31.101 16H36V11.6378ZM2 2V18H38V2H2Z" | ||||
|         fill={iconFillColor(appearance)} | ||||
|         fill={iconFillColor(theme)} | ||||
|       />, | ||||
|       { width: 40, height: 20 }, | ||||
|     ), | ||||
| ); | ||||
|  | ||||
| export const FillCrossHatchIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <g fill={iconFillColor(appearance)} fillRule="evenodd" clipRule="evenodd"> | ||||
|       <g fill={iconFillColor(theme)} fillRule="evenodd" clipRule="evenodd"> | ||||
|         <path d="M20.101 16H28.0934L36 8.95989V4H33.5779L20.101 16ZM30.5704 4L17.0935 16H9.10101L22.5779 4H30.5704ZM19.5704 4L6.09349 16H4V10.7475L11.5779 4H19.5704ZM8.57036 4H4V8.06952L8.57036 4ZM36 11.6378L31.101 16H36V11.6378ZM2 2V18H38V2H2Z" /> | ||||
|         <path d="M14.0001 18L3.00006 4.00002L4.5727 2.76438L15.5727 16.7644L14.0001 18ZM25.0001 18L14.0001 4.00002L15.5727 2.76438L26.5727 16.7644L25.0001 18ZM36.0001 18L25.0001 4.00002L26.5727 2.76438L37.5727 16.7644L36.0001 18Z" /> | ||||
|       </g>, | ||||
| @@ -595,25 +604,19 @@ export const FillCrossHatchIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const FillSolidIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|     createIcon(<path d="M2 2H38V18H2V2Z" fill={iconFillColor(appearance)} />, { | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon(<path d="M2 2H38V18H2V2Z" fill={iconFillColor(theme)} />, { | ||||
|       width: 40, | ||||
|       height: 20, | ||||
|     }), | ||||
| ); | ||||
|  | ||||
| export const StrokeWidthIcon = React.memo( | ||||
|   ({ | ||||
|     appearance, | ||||
|     strokeWidth, | ||||
|   }: { | ||||
|     appearance: "light" | "dark"; | ||||
|     strokeWidth: number; | ||||
|   }) => | ||||
|   ({ theme, strokeWidth }: { theme: "light" | "dark"; strokeWidth: number }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M6 10H34" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={strokeWidth} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -622,11 +625,11 @@ export const StrokeWidthIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const StrokeStyleSolidIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M6 10H34" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -638,11 +641,11 @@ export const StrokeStyleSolidIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const StrokeStyleDashedIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M6 10H34" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2.5} | ||||
|         strokeDasharray={"10, 8"} | ||||
|         fill="none" | ||||
| @@ -652,11 +655,11 @@ export const StrokeStyleDashedIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const StrokeStyleDottedIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M6 10H34" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2.5} | ||||
|         strokeDasharray={"4, 4"} | ||||
|         fill="none" | ||||
| @@ -666,11 +669,11 @@ export const StrokeStyleDottedIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const SloppinessArchitectIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M3.00098 16.1691C6.28774 13.9744 19.6399 2.8905 22.7215 3.00082C25.8041 3.11113 19.1158 15.5488 21.4962 16.8309C23.8757 18.1131 34.4155 11.7148 37.0001 10.6919" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -679,11 +682,11 @@ export const SloppinessArchitectIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const SloppinessArtistIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M3 17C6.68158 14.8752 16.1296 9.09849 22.0648 6.54922C28 3.99995 22.2896 13.3209 25 14C27.7104 14.6791 36.3757 9.6471 36.3757 9.6471M6.40706 15C13 11.1918 20.0468 1.51045 23.0234 3.0052C26 4.49995 20.457 12.8659 22.7285 16.4329C25 20 36.3757 13 36.3757 13" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -692,11 +695,11 @@ export const SloppinessArtistIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const SloppinessCartoonistIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M3 15.6468C6.93692 13.5378 22.5544 2.81528 26.6206 3.00242C30.6877 3.18956 25.6708 15.3346 27.4009 16.7705C29.1309 18.2055 35.4001 12.4762 37 11.6177M3.97143 10.4917C6.61158 9.24563 16.3706 2.61886 19.8104 3.01724C23.2522 3.41472 22.0773 12.2013 24.6181 12.8783C27.1598 13.5536 33.3179 8.04068 35.0571 7.07244" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -705,11 +708,11 @@ export const SloppinessCartoonistIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const EdgeSharpIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M10 17L10 5L35 5" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -718,11 +721,11 @@ export const EdgeSharpIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const EdgeRoundIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M10 17V15C10 8 13 5 21 5L33.5 5" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -731,11 +734,11 @@ export const EdgeRoundIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const ArrowheadNoneIcon = React.memo( | ||||
|   ({ appearance }: { appearance: "light" | "dark" }) => | ||||
|   ({ theme }: { theme: "light" | "dark" }) => | ||||
|     createIcon( | ||||
|       <path | ||||
|         d="M6 10H34" | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       />, | ||||
| @@ -747,17 +750,11 @@ export const ArrowheadNoneIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const ArrowheadArrowIcon = React.memo( | ||||
|   ({ | ||||
|     appearance, | ||||
|     flip = false, | ||||
|   }: { | ||||
|     appearance: "light" | "dark"; | ||||
|     flip?: boolean; | ||||
|   }) => | ||||
|   ({ theme, flip = false }: { theme: "light" | "dark"; flip?: boolean }) => | ||||
|     createIcon( | ||||
|       <g | ||||
|         transform={flip ? "translate(40, 0) scale(-1, 1)" : ""} | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         strokeWidth={2} | ||||
|         fill="none" | ||||
|       > | ||||
| @@ -769,17 +766,11 @@ export const ArrowheadArrowIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const ArrowheadDotIcon = React.memo( | ||||
|   ({ | ||||
|     appearance, | ||||
|     flip = false, | ||||
|   }: { | ||||
|     appearance: "light" | "dark"; | ||||
|     flip?: boolean; | ||||
|   }) => | ||||
|   ({ theme, flip = false }: { theme: "light" | "dark"; flip?: boolean }) => | ||||
|     createIcon( | ||||
|       <g | ||||
|         stroke={iconFillColor(appearance)} | ||||
|         fill={iconFillColor(appearance)} | ||||
|         stroke={iconFillColor(theme)} | ||||
|         fill={iconFillColor(theme)} | ||||
|         transform={flip ? "translate(40, 0) scale(-1, 1)" : ""} | ||||
|       > | ||||
|         <path d="M32 10L6 10" strokeWidth={2} /> | ||||
| @@ -790,18 +781,12 @@ export const ArrowheadDotIcon = React.memo( | ||||
| ); | ||||
|  | ||||
| export const ArrowheadBarIcon = React.memo( | ||||
|   ({ | ||||
|     appearance, | ||||
|     flip = false, | ||||
|   }: { | ||||
|     appearance: "light" | "dark"; | ||||
|     flip?: boolean; | ||||
|   }) => | ||||
|   ({ theme, flip = false }: { theme: "light" | "dark"; flip?: boolean }) => | ||||
|     createIcon( | ||||
|       <g transform={flip ? "translate(40, 0) scale(-1, 1)" : ""}> | ||||
|         <path | ||||
|           d="M34 10H5.99996M34 10L34 5M34 10L34 15" | ||||
|           stroke={iconFillColor(appearance)} | ||||
|           stroke={iconFillColor(theme)} | ||||
|           strokeWidth={2} | ||||
|           fill="none" | ||||
|         /> | ||||
|   | ||||
| @@ -1,51 +1,54 @@ | ||||
| import { FontFamily } from "./element/types"; | ||||
| import cssVariables from "./css/variables.module.scss"; | ||||
|  | ||||
| export const APP_NAME = "Excalidraw"; | ||||
|  | ||||
| export const DRAGGING_THRESHOLD = 10; | ||||
| export const LINE_CONFIRM_THRESHOLD = 10; | ||||
| export const DRAGGING_THRESHOLD = 10; // px | ||||
| export const LINE_CONFIRM_THRESHOLD = 8; // px | ||||
| export const ELEMENT_SHIFT_TRANSLATE_AMOUNT = 5; | ||||
| export const ELEMENT_TRANSLATE_AMOUNT = 1; | ||||
| export const TEXT_TO_CENTER_SNAP_THRESHOLD = 30; | ||||
| export const SHIFT_LOCKING_ANGLE = Math.PI / 12; | ||||
| export const CURSOR_TYPE = { | ||||
|   AUTO: "", | ||||
|   TEXT: "text", | ||||
|   CROSSHAIR: "crosshair", | ||||
|   GRABBING: "grabbing", | ||||
|   MOVE: "move", | ||||
|   POINTER: "pointer", | ||||
|   TEXT: "text", | ||||
|   MOVE: "move", | ||||
|   AUTO: "", | ||||
| }; | ||||
| export const POINTER_BUTTON = { | ||||
|   MAIN: 0, | ||||
|   WHEEL: 1, | ||||
|   SECONDARY: 2, | ||||
|   TOUCH: -1, | ||||
|   WHEEL: 1, | ||||
| }; | ||||
|  | ||||
| export enum EVENT { | ||||
|   BEFORE_UNLOAD = "beforeunload", | ||||
|   BLUR = "blur", | ||||
|   COPY = "copy", | ||||
|   PASTE = "paste", | ||||
|   CUT = "cut", | ||||
|   DRAG_OVER = "dragover", | ||||
|   DROP = "drop", | ||||
|   GESTURE_CHANGE = "gesturechange", | ||||
|   GESTURE_END = "gestureend", | ||||
|   GESTURE_START = "gesturestart", | ||||
|   HASHCHANGE = "hashchange", | ||||
|   KEYDOWN = "keydown", | ||||
|   KEYUP = "keyup", | ||||
|   MOUSE_MOVE = "mousemove", | ||||
|   PASTE = "paste", | ||||
|   RESIZE = "resize", | ||||
|   UNLOAD = "unload", | ||||
|   BLUR = "blur", | ||||
|   DRAG_OVER = "dragover", | ||||
|   DROP = "drop", | ||||
|   GESTURE_END = "gestureend", | ||||
|   BEFORE_UNLOAD = "beforeunload", | ||||
|   GESTURE_START = "gesturestart", | ||||
|   GESTURE_CHANGE = "gesturechange", | ||||
|   POINTER_MOVE = "pointermove", | ||||
|   POINTER_UP = "pointerup", | ||||
|   RESIZE = "resize", | ||||
|   STATE_CHANGE = "statechange", | ||||
|   TOUCH_END = "touchend", | ||||
|   TOUCH_START = "touchstart", | ||||
|   UNLOAD = "unload", | ||||
|   WHEEL = "wheel", | ||||
|   TOUCH_START = "touchstart", | ||||
|   TOUCH_END = "touchend", | ||||
|   HASHCHANGE = "hashchange", | ||||
|   VISIBILITY_CHANGE = "visibilitychange", | ||||
|   SCROLL = "scroll", | ||||
| } | ||||
|  | ||||
| export const ENV = { | ||||
| @@ -66,11 +69,11 @@ export const FONT_FAMILY = { | ||||
|  | ||||
| export const WINDOWS_EMOJI_FALLBACK_FONT = "Segoe UI Emoji"; | ||||
|  | ||||
| export const DEFAULT_FONT_FAMILY: FontFamily = 1; | ||||
| export const DEFAULT_FONT_SIZE = 20; | ||||
| export const DEFAULT_FONT_FAMILY: FontFamily = 1; | ||||
| export const DEFAULT_TEXT_ALIGN = "left"; | ||||
| export const DEFAULT_VERSION = "{version}"; | ||||
| export const DEFAULT_VERTICAL_ALIGN = "top"; | ||||
| export const DEFAULT_VERSION = "{version}"; | ||||
|  | ||||
| export const CANVAS_ONLY_ACTIONS = ["selectAll"]; | ||||
|  | ||||
| @@ -85,11 +88,25 @@ export const STORAGE_KEYS = { | ||||
|   LOCAL_STORAGE_LIBRARY: "excalidraw-library", | ||||
| }; | ||||
|  | ||||
| // Time in milliseconds | ||||
| // time in milliseconds | ||||
| export const TAP_TWICE_TIMEOUT = 300; | ||||
| export const TOUCH_CTX_MENU_TIMEOUT = 500; | ||||
| export const TITLE_TIMEOUT = 10000; | ||||
| export const TOAST_TIMEOUT = 5000; | ||||
| export const TOUCH_CTX_MENU_TIMEOUT = 500; | ||||
| export const VERSION_TIMEOUT = 15000; | ||||
| export const VERSION_TIMEOUT = 30000; | ||||
| export const SCROLL_TIMEOUT = 100; | ||||
|  | ||||
| export const ZOOM_STEP = 0.1; | ||||
|  | ||||
| // Report a user inactive after IDLE_THRESHOLD milliseconds | ||||
| export const IDLE_THRESHOLD = 60_000; | ||||
| // Report a user active each ACTIVE_THRESHOLD milliseconds | ||||
| export const ACTIVE_THRESHOLD = 3_000; | ||||
|  | ||||
| export const MODES = { | ||||
|   VIEW: "viewMode", | ||||
|   ZEN: "zenMode", | ||||
|   GRID: "gridMode", | ||||
| }; | ||||
|  | ||||
| export const THEME_FILTER = cssVariables.themeFilter; | ||||
|   | ||||
| @@ -1,15 +1,29 @@ | ||||
| @import "./variables.module"; | ||||
| @import "./theme"; | ||||
|  | ||||
| :root { | ||||
|   --zIndex-canvas: 1; | ||||
|   --zIndex-wysiwyg: 2; | ||||
|   --zIndex-layerUI: 3; | ||||
| } | ||||
|  | ||||
| .excalidraw { | ||||
|   color: var(--text-color-primary); | ||||
|   position: relative; | ||||
|   overflow: hidden; | ||||
|   color: var(--text-primary-color); | ||||
|   display: flex; | ||||
|   position: fixed; | ||||
|   top: 0; | ||||
|   bottom: 0; | ||||
|   left: 0; | ||||
|   right: 0; | ||||
|  | ||||
|   // serves 2 purposes: | ||||
|   // 1. prevent selecting text outside the component when double-clicking or | ||||
|   //    dragging inside it (e.g. on canvas) | ||||
|   // 2. prevent selecting UI, both from the inside, and from outside the | ||||
|   //    component (e.g. if you select text in a sidebar) | ||||
|   user-select: none; | ||||
|  | ||||
|   a { | ||||
|     font-weight: 500; | ||||
|     text-decoration: none; | ||||
| @@ -22,7 +36,6 @@ | ||||
|  | ||||
|   canvas { | ||||
|     touch-action: none; | ||||
|     user-select: none; | ||||
|  | ||||
|     // following props improve blurriness at certain devicePixelRatios. | ||||
|     // AFAIK it doesn't affect export (in fact, export seems sharp either way). | ||||
| @@ -30,15 +43,17 @@ | ||||
|     image-rendering: pixelated; // chromium | ||||
|     // NOTE: must be declared *after* the above | ||||
|     image-rendering: -moz-crisp-edges; // FF | ||||
|  | ||||
|     z-index: var(--zIndex-canvas); | ||||
|   } | ||||
|  | ||||
|   &.Appearance_dark { | ||||
|   &.theme--dark { | ||||
|     // The percentage is inspired by | ||||
|     // https://material.io/design/color/dark-theme.html#properties, which | ||||
|     // recommends surface color of #121212, 93% yields #111111 for #FFF | ||||
|  | ||||
|     canvas { | ||||
|       filter: var(--appearance-filter); | ||||
|       filter: var(--theme-filter); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -64,7 +79,7 @@ | ||||
|       margin-top: 0.333rem; | ||||
|       margin-bottom: 0.333rem; | ||||
|       font-size: 0.75rem; | ||||
|       color: var(--text-color-primary); | ||||
|       color: var(--text-primary-color); | ||||
|       font-weight: bold; | ||||
|       display: block; | ||||
|     } | ||||
| @@ -216,6 +231,13 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .App-top-bar { | ||||
|     z-index: var(--zIndex-layerUI); | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|   } | ||||
|  | ||||
|   .App-bottom-bar { | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
| @@ -223,7 +245,7 @@ | ||||
|     left: 0; | ||||
|     right: 0; | ||||
|     --bar-padding: calc(4 * var(--space-factor)); | ||||
|     padding-top: #{"max(var(--bar-padding), var(--sat, 0))"}; | ||||
|     padding-top: #{"max(var(--bar-padding), var(--sat,0))"}; | ||||
|     padding-right: var(--sar, 0); | ||||
|     padding-bottom: var(--sab, 0); | ||||
|     padding-left: var(--sal, 0); | ||||
| @@ -282,7 +304,7 @@ | ||||
|     pointer-events: none !important; | ||||
|   } | ||||
|  | ||||
|   .App-menu_top > * { | ||||
|   .layer-ui__wrapper:not(.disable-pointerEvents) .App-menu_top > * { | ||||
|     pointer-events: all; | ||||
|   } | ||||
|  | ||||
| @@ -323,7 +345,7 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .App-menu_bottom > * { | ||||
|   .layer-ui__wrapper:not(.disable-pointerEvents) .App-menu_bottom > * { | ||||
|     pointer-events: all; | ||||
|   } | ||||
|  | ||||
| @@ -347,7 +369,6 @@ | ||||
|  | ||||
|   .App-menu__left { | ||||
|     overflow-y: auto; | ||||
|     max-height: calc(100vh - 236px); | ||||
|   } | ||||
|  | ||||
|   .dropdown-select { | ||||
| @@ -419,7 +440,7 @@ | ||||
|  | ||||
|   .scroll-back-to-content { | ||||
|     color: var(--popup-text-color); | ||||
|     position: fixed; | ||||
|     position: absolute; | ||||
|     left: 50%; | ||||
|     bottom: 30px; | ||||
|     transform: translateX(-50%); | ||||
| @@ -492,6 +513,13 @@ | ||||
|     pointer-events: none !important; | ||||
|   } | ||||
|  | ||||
|   &.excalidraw--view-mode { | ||||
|     .App-menu { | ||||
|       display: flex; | ||||
|       justify-content: space-between; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @media print { | ||||
|     .App-bottom-bar, | ||||
|     .FixedSideContainer, | ||||
|   | ||||
| @@ -1,14 +1,15 @@ | ||||
| @import "open-color/open-color.scss"; | ||||
| @import "./variables.module.scss"; | ||||
|  | ||||
| :root { | ||||
|   --appearance-filter: none; | ||||
| .excalidraw { | ||||
|   --theme-filter: none; | ||||
|   --button-destructive-bg-color: #{$oc-red-1}; | ||||
|   --button-destructive-color: #{$oc-red-9}; | ||||
|   --button-gray-1: #{$oc-gray-2}; | ||||
|   --button-gray-2: #{$oc-gray-4}; | ||||
|   --button-gray-3: #{$oc-gray-5}; | ||||
|   --button-special-active-bg-color: #{$oc-green-0}; | ||||
|   --dialog-border: #{$oc-gray-6}; | ||||
|   --dialog-border-color: #{$oc-gray-6}; | ||||
|   --dropdown-icon: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" width="292.4" height="292.4" viewBox="0 0 292 292"><path d="M287 197L159 69c-4-3-8-5-13-5s-9 2-13 5L5 197c-3 4-5 8-5 13s2 9 5 13c4 4 8 5 13 5h256c5 0 9-1 13-5s5-8 5-13-1-9-5-13z"/></svg>'); | ||||
|   --focus-highlight-color: #{$oc-blue-2}; | ||||
|   --icon-fill-color: #{$oc-black}; | ||||
| @@ -17,7 +18,7 @@ | ||||
|   --input-border-color: #{$oc-gray-3}; | ||||
|   --input-hover-bg-color: #{$oc-gray-1}; | ||||
|   --input-label-color: #{$oc-gray-7}; | ||||
|   --island-bg-color: #{transparentize($oc-white, 0.12)}; | ||||
|   --island-bg-color: rgba(255, 255, 255, 0.9); | ||||
|   --keybinding-color: #{$oc-gray-5}; | ||||
|   --link-color: #{$oc-blue-7}; | ||||
|   --overlay-bg-color: #{transparentize($oc-white, 0.12)}; | ||||
| @@ -32,27 +33,25 @@ | ||||
|   --select-highlight-color: #{$oc-blue-5}; | ||||
|   --shadow-island: 0 1px 5px #{transparentize($oc-black, 0.85)}; | ||||
|   --space-factor: 0.25rem; | ||||
|   --text-color-primary: #{$oc-gray-8}; | ||||
| } | ||||
|   --text-primary-color: #{$oc-gray-8}; | ||||
|  | ||||
| .excalidraw { | ||||
|   &.Appearance_dark { | ||||
|   &.theme--dark { | ||||
|     background: $oc-black; | ||||
|  | ||||
|     &.Appearance_dark-background-none { | ||||
|     &.theme--dark-background-none { | ||||
|       background: none; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &.Appearance_dark { | ||||
|     --appearance-filter: invert(93%) hue-rotate(180deg); | ||||
|   &.theme--dark { | ||||
|     --theme-filter: #{$theme-filter}; | ||||
|     --button-destructive-bg-color: #5a0000; | ||||
|     --button-destructive-color: #{$oc-red-3}; | ||||
|     --button-gray-1: #363636; | ||||
|     --button-gray-2: #272727; | ||||
|     --button-gray-3: #222; | ||||
|     --button-special-active-bg-color: #204624; | ||||
|     --dialog-border: #{$oc-gray-9}; | ||||
|     --dialog-border-color: #{$oc-gray-9}; | ||||
|     --dropdown-icon: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" width="292.4" height="292.4" viewBox="0 0 292 292"><path fill="%23ced4da" d="M287 197L159 69c-4-3-8-5-13-5s-9 2-13 5L5 197c-3 4-5 8-5 13s2 9 5 13c4 4 8 5 13 5h256c5 0 9-1 13-5s5-8 5-13-1-9-5-13z"/></svg>'); | ||||
|     --focus-highlight-color: #{$oc-blue-6}; | ||||
|     --icon-fill-color: #{$oc-gray-4}; | ||||
| @@ -63,11 +62,13 @@ | ||||
|     --input-label-color: #{$oc-gray-2}; | ||||
|     --island-bg-color: #1e1e1e; | ||||
|     --keybinding-color: #{$oc-gray-6}; | ||||
|     --overlay-bg-color: rgba(30, 30, 30, 0.88); | ||||
|     --overlay-bg-color: #{transparentize($oc-gray-8, 0.88)}; | ||||
|     --popup-bg-color: #2c2c2c; | ||||
|     --popup-secondary-bg-color: #222; | ||||
|     --popup-text-color: #{$oc-gray-4}; | ||||
|     --popup-text-inverted-color: #2c2c2c; | ||||
|     --select-highlight-color: #{$oc-blue-4}; | ||||
|     --shadow-island: 0 1px 5px #{transparentize($oc-black, 0.7)}; | ||||
|     --text-primary-color: #{$oc-gray-4}; | ||||
|   } | ||||
| } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user