Compare commits
	
		
			478 Commits
		
	
	
		
			v0.17.4
			...
			dwelle/pas
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d921887e2a | ||
|   | 14d512f321 | ||
|   | 41c036e1a5 | ||
|   | 91d36e9b81 | ||
|   | 27522110df | ||
|   | 712f267519 | ||
|   | 41a7613dff | ||
|   | 95d89a751a | ||
|   | 6b5fb30d69 | ||
|   | d92a849038 | ||
|   | 0a534f1bc6 | ||
|   | 4ca5f53b1f | ||
|   | f7dcc893ea | ||
|   | 4dfb8a3f8e | ||
|   | 298812e1d0 | ||
|   | 35bb449a4b | ||
|   | c4c064982f | ||
|   | 51dbd4831b | ||
|   | 7e41026812 | ||
|   | a8ebe514da | ||
|   | a30e1b25c6 | ||
|   | ff2ed5d26a | ||
|   | e058a08b33 | ||
|   | a306a909a0 | ||
|   | 3dc54a724a | ||
|   | a7c61319dd | ||
|   | cec5232a7a | ||
|   | d4f70e9f31 | ||
|   | e19fd1332a | ||
|   | 6e655cdb24 | ||
|   | 192c4e7658 | ||
|   | 195a743874 | ||
|   | 4a60fe3d22 | ||
|   | 2a0d15799c | ||
|   | a18b139a60 | ||
|   | 1913599594 | ||
|   | debf2ad608 | ||
|   | 8fb2f70414 | ||
|   | 5fc13e4309 | ||
|   | b5d60973b7 | ||
|   | a5d6939826 | ||
|   | 0cf36d6b30 | ||
|   | 58f7d33d80 | ||
|   | 6fe7de8020 | ||
|   | 01304aac49 | ||
|   | dff69e9191 | ||
|   | 6fc85022ae | ||
|   | e48b63a0ae | ||
|   | c2caf78e95 | ||
|   | ce267aa0d3 | ||
|   | 6e47fadb59 | ||
|   | b3d5ba0567 | ||
|   | c79e892e55 | ||
|   | 57a9e301d4 | ||
|   | 7c58477382 | ||
|   | 83fac6d0db | ||
|   | f2e8404c7b | ||
|   | d797c2e210 | ||
|   | 0cd5a259ae | ||
|   | 432a46ef9e | ||
|   | a18f059188 | ||
|   | ab89d4c16f | ||
|   | 6c3a434f2a | ||
|   | e1bb59fb8f | ||
|   | 77aca48c84 | ||
|   | 58990b41ae | ||
|   | 99d8bff175 | ||
|   | 30983d801a | ||
|   | 21ffaf4d76 | ||
|   | 82b9a6b464 | ||
|   | 817d8c553c | ||
|   | 69bc5bdaab | ||
|   | d587b8a3de | ||
|   | 4ec812bc18 | ||
|   | a9e2d2348b | ||
|   | 70c3e921bb | ||
|   | d92384b77d | ||
|   | c5d3bb0b6a | ||
|   | d21c6a1bc6 | ||
|   | d1112bbf4f | ||
|   | 2523624f15 | ||
|   | 68578556ff | ||
|   | ecef5d12f4 | ||
|   | 392118bf26 | ||
|   | 0ffeaeaecf | ||
|   | 31e8476c78 | ||
|   | 9ee0b8ffcb | ||
|   | 16b86d7d16 | ||
|   | f12b92ce9d | ||
|   | 77dc055d81 | ||
|   | 26f02bebea | ||
|   | e3060dfb8f | ||
|   | c329470b73 | ||
|   | c8f4a4cb41 | ||
|   | 9e49c9254b | ||
|   | b0c8c5f7a7 | ||
|   | 4f64372506 | ||
|   | 424e94a403 | ||
|   | 302664e500 | ||
|   | 86c67bd37f | ||
|   | 511433988c | ||
|   | 9b6edc767a | ||
|   | 6cdb683410 | ||
|   | 84bab403ff | ||
|   | 61e0bb83d0 | ||
|   | bd1590fc74 | ||
|   | d29c3db7f6 | ||
|   | a58822c1c1 | ||
|   | a3e1619635 | ||
|   | 52eaf64591 | ||
|   | 7028daa44a | ||
|   | 65f218b100 | ||
|   | 807b3c59f2 | ||
|   | b8da5065fd | ||
|   | 49f1276ef2 | ||
|   | 8f20b29b73 | ||
|   | f87c2cde09 | ||
|   | 0bf234fcc9 | ||
|   | dd1b45a25a | ||
|   | ec06fbc1fc | ||
|   | fa05ae1230 | ||
|   | 91ebf8b0ea | ||
|   | 8551823da9 | ||
|   | ae6bee3403 | ||
|   | 46f42ef8d7 | ||
|   | 00b5b0a0ca | ||
|   | c92f3bebf5 | ||
|   | 2ac55067cd | ||
|   | 78ab12c7e6 | ||
|   | f2f8219917 | ||
|   | 12c39d1034 | ||
|   | d33e42e3a1 | ||
|   | 3b9ffd9586 | ||
|   | b63689c230 | ||
|   | c84babf574 | ||
|   | 36274f1f3e | ||
|   | 798c795405 | ||
|   | 107eae3916 | ||
|   | 56fca30bd0 | ||
|   | 1e3399eac8 | ||
|   | 873698a1a2 | ||
|   | 606ac6c743 | ||
|   | d99e4a23ca | ||
|   | 551bae07a7 | ||
|   | 2af3221974 | ||
|   | 9b401f6ea3 | ||
|   | 8a1152ce36 | ||
|   | b5652b8e36 | ||
|   | 31e2a0cb4a | ||
|   | c0b80a03bd | ||
|   | a758aaf8f6 | ||
|   | b2a6a87b10 | ||
|   | ab8b3537b3 | ||
|   | d21e0008dd | ||
|   | 840f1428c4 | ||
|   | 2db5bbcb29 | ||
|   | 0927431d0d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 98c0a67333 | ||
|   | 57cf577376 | ||
|   | 6e0ee89ee4 | ||
|   | 35f778a734 | ||
|   | ee091d0dbd | ||
|   | ef9ea14a75 | ||
|   | df168a6883 | ||
|   | 798f5f4dfb | ||
|   | d9ad7c039b | ||
|   | 7c0239e693 | ||
|   | da33481fa3 | ||
|   | 70e0e8dc29 | ||
|   | 2734e646ca | ||
|   | dfaaff4432 | ||
|   | 03028eaa8c | ||
|   | 79b181bcdc | ||
|   | f9815b8b4f | ||
|   | 96ed8a4331 | ||
|   | 33b01d4e80 | ||
|   | 7d52176fea | ||
|   | 958e03fcc6 | ||
|   | 4cedf3d966 | ||
|   | e957c8e9ee | ||
|   | eb09b48ae6 | ||
|   | 61623bbeba | ||
|   | 15ca182333 | ||
|   | b479f3bd65 | ||
|   | 21815fb930 | ||
|   | 47ee8a0094 | ||
|   | a977dd1bf5 | ||
|   | 3fe1883f3f | ||
|   | a80cb5896a | ||
|   | 6dfa18414a | ||
|   | 8ca4cf3260 | ||
|   | f3f0ab7c83 | ||
|   | 44a1c8d857 | ||
|   | e0a22edfbd | ||
|   | c07f5a0c80 | ||
|   | 508f16dc04 | ||
|   | c1b310c56b | ||
|   | d4900e8f19 | ||
|   | caf2db934c | ||
|   | 60e3801691 | ||
|   | 80f3b75d42 | ||
|   | dc812bee19 | ||
|   | 01e83cc9a5 | ||
|   | 813f9b702e | ||
|   | fd39712ba6 | ||
|   | b46ca0192b | ||
|   | 72b7c937b1 | ||
|   | d107215564 | ||
|   | 6959a363f0 | ||
|   | 5a11c70714 | ||
|   | 6ff56c36e3 | ||
|   | 51ea184938 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 21fff26d31 | ||
|   | f4dd23fc31 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e3d1dee9d0 | ||
|   | e3f31df747 | ||
|   | 60e75406e0 | ||
|   | b396e07b90 | ||
|   | 2d1d84a47b | ||
|   | ee30225062 | ||
|   | 16cae4fc07 | ||
|   | 576bc0dbe5 | ||
|   | 00af35c692 | ||
|   | ea7c702cfc | ||
|   | 26d2296578 | ||
|   | afb68a6467 | ||
|   | b459e5cfd2 | ||
|   | 5facc0d6da | ||
|   | 824ad603e1 | ||
|   | 5e1ff7cafe | ||
|   | b5d7f5b4ba | ||
|   | fb4bb29aa5 | ||
|   | 3cfcc7b489 | ||
|   | 4320a3cf41 | ||
|   | 8420e1aa13 | ||
|   | 5daf1a1b4e | ||
|   | 97981804d7 | ||
|   | f7b3befd0a | ||
|   | 7b2bee9746 | ||
|   | 88014ace4a | ||
|   | 87a9430809 | ||
|   | 99b91c46f7 | ||
|   | 1ea5b26f25 | ||
|   | d5f4ee7b3f | ||
|   | 261304c1a4 | ||
|   | 84398a7e5c | ||
|   | 54491d13d4 | ||
|   | dd1370381d | ||
|   | 72d6ee48fc | ||
|   | 232242d2e9 | ||
|   | f19ce30dfe | ||
|   | 3cf14c73a3 | ||
|   | 8d530cf102 | ||
|   | b87925d253 | ||
|   | a6684b09f2 | ||
|   | b427617f66 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2907fbe34b | ||
|   | c67815f7b0 | ||
|   | c641860cb1 | ||
|   | 84d89b9a8a | ||
|   | e63dd025c9 | ||
|   | 15e019706d | ||
|   | a133a70e87 | ||
|   | 80ea7ca23f | ||
|   | e844580b14 | ||
|   | 5a0771ad9c | ||
|   | adcdbe2907 | ||
|   | 230d0edc44 | ||
|   | d0a380758e | ||
|   | 7b36de0476 | ||
|   | 2427e622b0 | ||
|   | 62228e0bbb | ||
|   | 4c5408263c | ||
|   | bd7b778f41 | ||
|   | 43b2476dfe | ||
|   | df8875a497 | ||
|   | 6fbc44fd1f | ||
|   | d25a7d365b | ||
|   | e52c2cd0b6 | ||
|   | 96eeec5119 | ||
|   | f5221d521b | ||
|   | db2c235cd4 | ||
|   | 148b895f46 | ||
|   | d9258a736b | ||
|   | 2e1f08c796 | ||
|   | 1d5b41dabb | ||
|   | 66a2f24296 | ||
|   | 04668d8263 | ||
|   | abbeed3d5f | ||
|   | ba8c09d529 | ||
|   | 744b3e5d09 | ||
|   | 6ba9bd60e8 | ||
|   | a1ffa064df | ||
|   | 4dc4590f24 | ||
|   | d2f67e619f | ||
|   | 22b39277f5 | ||
|   | 63dee03ef0 | ||
|   | 08b13f971d | ||
|   | 69f4cc70cb | ||
|   | 860308eb27 | ||
|   | 4eb9463f26 | ||
|   | 6ed6131169 | ||
|   | 1ed98f9c93 | ||
|   | a71bb63d1f | ||
|   | 661d6a4a75 | ||
|   | defd34923a | ||
|   | c540bd68aa | ||
|   | eddbe55f50 | ||
|   | 2f9526da24 | ||
|   | 1b6e3fe05b | ||
|   | afe52c89a7 | ||
|   | be4e127f6c | ||
|   | ff0b4394b1 | ||
|   | 7d8b7fc14d | ||
|   | 971b4d4ae6 | ||
|   | cc4c51996c | ||
|   | 79257a1923 | ||
|   | dc66261c19 | ||
|   | 273ba803d9 | ||
|   | 301e83805d | ||
|   | ed5ce8d3de | ||
|   | 1ed53b153c | ||
|   | c1926f33bb | ||
|   | 6539029d2a | ||
|   | d1f37cc64f | ||
|   | f0d25e34c3 | ||
|   | d9bbf1eda6 | ||
|   | f79fb9aae2 | ||
|   | 275f6fbe24 | ||
|   | 88812e0386 | ||
|   | 6e5aeb112d | ||
|   | 4d83d1c91e | ||
|   | a04676d423 | ||
|   | c851aaaf7b | ||
|   | 1bd2b1fe55 | ||
|   | 015b46ab23 | ||
|   | 530617be90 | ||
|   | 5211b003b8 | ||
|   | bbcca06b94 | ||
|   | f92f04c13c | ||
|   | 890ed9f31f | ||
|   | da2e507298 | ||
|   | f59b4f6af4 | ||
|   | afcde542f9 | ||
|   | 4689a6b300 | ||
|   | 0ae9b383d6 | ||
|   | f597bd3e01 | ||
|   | 4987cc53d0 | ||
|   | d917db438e | ||
|   | a33a400f01 | ||
|   | 8a162a4cb4 | ||
|   | c6a045d092 | ||
|   | cd50aa719f | ||
|   | 92bc08207c | ||
|   | 32df5502ae | ||
|   | bbdcd30a73 | ||
|   | 3e334a67ed | ||
|   | 1d71f84515 | ||
|   | 550a388b2b | ||
|   | 6b523563d8 | ||
|   | 65bc500598 | ||
|   | 7949aa1f1c | ||
|   | 15bfa626b4 | ||
|   | 068895db0e | ||
|   | b7babe554b | ||
|   | 6a385d6663 | ||
|   | 2382fad4f6 | ||
|   | 480572f893 | ||
|   | 68b1fdb20e | ||
|   | a38e82f999 | ||
|   | a07f6e9e3a | ||
|   | 7e471b55eb | ||
|   | 160440b860 | ||
|   | f207bd0a1c | ||
|   | 99601baffc | ||
|   | af1a3d5b76 | ||
|   | 36e56267c9 | ||
|   | b09b5cb5f4 | ||
|   | dd8529743a | ||
|   | f639d44a95 | ||
|   | f5ab3e4e12 | ||
|   | 361a9449bb | ||
|   | 2e719ff671 | ||
|   | 79d9dc2f8f | ||
|   | 9013c84524 | ||
|   | 47f87f4ecb | ||
|   | 73bf50e8a8 | ||
|   | 48c3465b19 | ||
|   | adc4c9f484 | ||
|   | def1df2c68 | ||
|   | 0513b647ec | ||
|   | a289c42830 | ||
|   | d67eaa8710 | ||
|   | 0c3dffb082 | ||
|   | 0e0f34edd8 | ||
|   | 4888d9d355 | ||
|   | 1c39bd5781 | ||
|   | 90ad885446 | ||
|   | 1741c234a6 | ||
|   | 63b50b3586 | ||
|   | e0fefa8025 | ||
|   | d426cc968d | ||
|   | 2409c091ff | ||
|   | 626fe252ab | ||
|   | 10bd08ef19 | ||
|   | 2789d08154 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 678bb2b819 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 966f9aead9 | ||
|   | 4f0a2a9593 | ||
|   | f3f8217125 | ||
|   | 89bd6181f2 | ||
|   | c6fdac131b | ||
|   | 0415c616b1 | ||
|   | 740a165452 | ||
|   | 4997624a3a | ||
|   | b66daae1f3 | ||
|   | 1e7df58b5b | ||
|   | 46da032626 | ||
|   | 3b0593baa7 | ||
|   | dd530737a2 | ||
|   | a4e5e46dd1 | ||
|   | 0fa5f5de4c | ||
|   | 41cc746885 | ||
|   | 8ead8559e0 | ||
|   | 5245276409 | ||
|   | 0c24a7042f | ||
|   | 86cfeb714c | ||
|   | 872973f145 | ||
|   | 3ecf72a507 | ||
|   | 1aaa400876 | ||
|   | 65047cc2cb | ||
|   | 8b993d409e | ||
|   | 1cb350b2aa | ||
|   | 43ccc875fb | ||
|   | 4249b7dec8 | ||
|   | 49f15c736b | ||
|   | a8064ba3ee | ||
|   | e6c3c06c2e | ||
|   | d19b51d4f8 | ||
|   | c72e853c85 | ||
|   | 5f40a4cad4 | ||
|   | d91c98b82e | ||
|   | 57ea4e61d1 | ||
|   | 0808532b49 | ||
|   | 2a0fe2584e | ||
|   | 7bd6496854 | ||
|   | 537f6e7f68 | ||
|   | 6dfa89e846 | ||
|   | 561e919a2e | ||
|   | 20e3acf7a6 | ||
|   | 2c0929e537 | ||
|   | aad8ab0123 | ||
|   | 88a2b286c7 | ||
|   | b635b10b59 | ||
|   | 7ebda02b81 | ||
|   | d6cd8b78f1 | ||
|   | b7d7ccc929 | ||
|   | f14ad61bd0 | ||
|   | 8963baf5ad | ||
|   | 557add5bf7 | ||
|   | b9cfbc2077 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a04cc707c3 | ||
|   | 72ea8022bf | ||
|   | 4bdeaf999b | ||
|   | 42d8c5a040 | ||
|   | f299514e44 | ||
|   | dd220bcaea | ||
|   | fe75f29c15 | ||
|   | 14845a343b | ||
|   | dd8a7d41e2 | ||
|   | fda5c6fdf7 | ||
|   | 3d1631f375 | ||
|   | c7ee46e7f8 | ||
|   | d1e4421823 | ||
|   | 7c9cf30909 | ||
|   | 1e37dbd60e | ||
|   | f8d5c2a1b6 | ||
|   | 23b24ea5c3 | 
| @@ -4,8 +4,16 @@ | ||||
| !.eslintrc.json | ||||
| !.npmrc | ||||
| !.prettierrc | ||||
| !excalidraw-app/ | ||||
| !package.json | ||||
| !public/ | ||||
| !src/ | ||||
| !packages/ | ||||
| !scripts/ | ||||
| !tsconfig.json | ||||
| !yarn.lock | ||||
|  | ||||
| # keep (sub)sub directories at the end to exclude from explicit included | ||||
| # e.g. ./packages/excalidraw/{dist,node_modules} | ||||
| **/build | ||||
| **/dist | ||||
| **/node_modules | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| MODE="development" | ||||
|  | ||||
| VITE_APP_BACKEND_V2_GET_URL=https://json-dev.excalidraw.com/api/v2/ | ||||
| VITE_APP_BACKEND_V2_POST_URL=https://json-dev.excalidraw.com/api/v2/post/ | ||||
|  | ||||
| @@ -7,23 +9,20 @@ VITE_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfu | ||||
| # collaboration WebSocket server (https://github.com/excalidraw/excalidraw-room) | ||||
| VITE_APP_WS_SERVER_URL=http://localhost:3002 | ||||
|  | ||||
| # set this only if using the collaboration workflow we use on excalidraw.com | ||||
| VITE_APP_PORTAL_URL= | ||||
|  | ||||
| VITE_APP_PLUS_LP=https://plus.excalidraw.com | ||||
| VITE_APP_PLUS_APP=https://app.excalidraw.com | ||||
| VITE_APP_PLUS_APP=http://localhost:3000 | ||||
|  | ||||
| VITE_APP_AI_BACKEND=http://localhost:3015 | ||||
|  | ||||
| VITE_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8","authDomain":"excalidraw-oss-dev.firebaseapp.com","projectId":"excalidraw-oss-dev","storageBucket":"excalidraw-oss-dev.appspot.com","messagingSenderId":"664559512677","appId":"1:664559512677:web:a385181f2928d328a7aa8c"}' | ||||
|  | ||||
| # put these in your .env.local, or make sure you don't commit! | ||||
| # must be lowercase `true` when turned on | ||||
| # | ||||
| # whether to enable Service Workers in development | ||||
| VITE_APP_DEV_ENABLE_SW= | ||||
| # whether to disable live reload / HMR. Usuaully what you want to do when | ||||
| # debugging Service Workers. | ||||
| VITE_APP_DEV_DISABLE_LIVE_RELOAD= | ||||
| VITE_APP_DISABLE_TRACKING=true | ||||
| VITE_APP_ENABLE_TRACKING=true | ||||
|  | ||||
| FAST_REFRESH=false | ||||
|  | ||||
| @@ -40,3 +39,17 @@ VITE_APP_COLLAPSE_OVERLAY=true | ||||
|  | ||||
| # Set this flag to false to disable eslint | ||||
| VITE_APP_ENABLE_ESLINT=true | ||||
|  | ||||
| # Enable PWA in dev server | ||||
| VITE_APP_ENABLE_PWA=false | ||||
|  | ||||
| VITE_APP_PLUS_EXPORT_PUBLIC_KEY='MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAm2g5T+Rub6Kbf1Mf57t0 | ||||
| 7r2zeHuVg4dla3r5ryXMswtzz6x767octl6oLThn33mQsPSy3GKglFZoCTXJR4ij | ||||
| ba8SxB04sL/N8eRrKja7TFWjCVtRwTTfyy771NYYNFVJclkxHyE5qw4m27crHF1y | ||||
| UNWEjuqNMi/lwAErS9fFa2oJlWyT8U7zzv/5kQREkxZI6y9v0AF3qcbsy2731FnD | ||||
| s9ChJvOUW9toIab2gsIdrKW8ZNpu084ZFVKb6LNjvIXI1Se4oMTHeszXzNptzlot | ||||
| kdxxjOoaQMAyfljFSot1F1FlU6MQlag7UnFGvFjRHN1JI5q4K+n3a67DX+TMyRqS | ||||
| HQIDAQAB' | ||||
|  | ||||
| # set to true in .env.development.local to disable the prevent unload dialog | ||||
| VITE_APP_DISABLE_PREVENT_UNLOAD= | ||||
|   | ||||
| @@ -1,18 +1,34 @@ | ||||
| MODE="production" | ||||
|  | ||||
| VITE_APP_BACKEND_V2_GET_URL=https://json.excalidraw.com/api/v2/ | ||||
| VITE_APP_BACKEND_V2_POST_URL=https://json.excalidraw.com/api/v2/post/ | ||||
|  | ||||
| VITE_APP_LIBRARY_URL=https://libraries.excalidraw.com | ||||
| VITE_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | ||||
|  | ||||
| VITE_APP_PORTAL_URL=https://portal.excalidraw.com | ||||
|  | ||||
| VITE_APP_PLUS_LP=https://plus.excalidraw.com | ||||
| VITE_APP_PLUS_APP=https://app.excalidraw.com | ||||
|  | ||||
| # Fill to set socket server URL used for collaboration. | ||||
| # Meant for forks only: excalidraw.com uses custom VITE_APP_PORTAL_URL flow | ||||
| VITE_APP_WS_SERVER_URL= | ||||
| VITE_APP_AI_BACKEND=https://oss-ai.excalidraw.com | ||||
|  | ||||
| # socket server URL used for collaboration | ||||
| VITE_APP_WS_SERVER_URL=https://oss-collab.excalidraw.com | ||||
|  | ||||
| VITE_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyAd15pYlMci_xIp9ko6wkEsDzAAA0Dn0RU","authDomain":"excalidraw-room-persistence.firebaseapp.com","databaseURL":"https://excalidraw-room-persistence.firebaseio.com","projectId":"excalidraw-room-persistence","storageBucket":"excalidraw-room-persistence.appspot.com","messagingSenderId":"654800341332","appId":"1:654800341332:web:4a692de832b55bd57ce0c1"}' | ||||
|  | ||||
| VITE_APP_DISABLE_TRACKING= | ||||
| VITE_APP_ENABLE_TRACKING=false | ||||
|  | ||||
| VITE_APP_PLUS_EXPORT_PUBLIC_KEY='MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEApQ0jM9Qz8TdFLzcuAZZX | ||||
| /WvuKSOJxiw6AR/ZcE3eFQWM/mbFdhQgyK8eHGkKQifKzH1xUZjCxyXcxW6ZO02t | ||||
| kPOPxhz+nxUrIoWCD/V4NGmUA1lxwHuO21HN1gzKrN3xHg5EGjyouR9vibT9VDGF | ||||
| gq6+4Ic/kJX+AD2MM7Yre2+FsOdysrmuW2Fu3ahuC1uQE7pOe1j0k7auNP0y1q53 | ||||
| PrB8Ts2LUpepWC1l7zIXFm4ViDULuyWXTEpUcHSsEH8vpd1tckjypxCwkipfZsXx | ||||
| iPszy0o0Dx2iArPfWMXlFAI9mvyFCyFC3+nSvfyAUb2C4uZgCwAuyFh/ydPF4DEE | ||||
| PQIDAQAB' | ||||
|  | ||||
| # Set the below flags explicitly to false in production mode since vite loads and merges .env.local vars when  running the build command | ||||
| VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX=false | ||||
| VITE_APP_COLLAPSE_OVERLAY=false | ||||
| # Enable eslint in dev server | ||||
| VITE_APP_ENABLE_ESLINT=false | ||||
|  | ||||
|   | ||||
| @@ -5,4 +5,7 @@ package-lock.json | ||||
| firebase/ | ||||
| dist/ | ||||
| public/workbox | ||||
| src/packages/excalidraw/types | ||||
| packages/excalidraw/types | ||||
| examples/**/public | ||||
| dev-dist | ||||
| coverage | ||||
|   | ||||
| @@ -1,7 +1,43 @@ | ||||
| { | ||||
|   "extends": ["@excalidraw/eslint-config", "react-app"], | ||||
|   "rules": { | ||||
|     "import/order": [ | ||||
|       "warn", | ||||
|       { | ||||
|         "groups": ["builtin", "external", "internal", "parent", "sibling", "index", "object", "type"], | ||||
|         "pathGroups": [ | ||||
|           { | ||||
|             "pattern": "@excalidraw/**", | ||||
|             "group": "external", | ||||
|             "position": "after" | ||||
|           } | ||||
|         ], | ||||
|         "newlines-between": "always-and-inside-groups", | ||||
|         "warnOnUnassignedImports": true | ||||
|       } | ||||
|     ], | ||||
|     "import/no-anonymous-default-export": "off", | ||||
|     "no-restricted-globals": "off" | ||||
|     "no-restricted-globals": "off", | ||||
|     "@typescript-eslint/consistent-type-imports": [ | ||||
|       "error", | ||||
|       { | ||||
|         "prefer": "type-imports", | ||||
|         "disallowTypeAnnotations": false, | ||||
|         "fixStyle": "separate-type-imports" | ||||
|       } | ||||
|     ], | ||||
|     "no-restricted-imports": [ | ||||
|       "error", | ||||
|       { | ||||
|         "name": "jotai", | ||||
|         "message": "Do not import from \"jotai\" directly. Use our app-specific modules (\"editor-jotai\" or \"app-jotai\")." | ||||
|       } | ||||
|     ], | ||||
|     "react/jsx-no-target-blank": [ | ||||
|       "error", | ||||
|       { | ||||
|         "allowReferrer": true | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -23,5 +23,5 @@ jobs: | ||||
|           NPM_TOKEN: ${{ secrets.NPM_TOKEN }} | ||||
|       - name: Auto release | ||||
|         run: | | ||||
|           yarn add @actions/core | ||||
|           yarn add @actions/core -W | ||||
|           yarn autorelease | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -44,7 +44,7 @@ jobs: | ||||
|       - name: Auto release preview | ||||
|         id: "autorelease" | ||||
|         run: | | ||||
|           yarn add @actions/core | ||||
|           yarn add @actions/core -W | ||||
|           yarn autorelease preview ${{ github.event.issue.number }} | ||||
|       - name: Post comment post release | ||||
|         if: always() | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,7 +16,7 @@ jobs: | ||||
|  | ||||
|       - name: Install and lint | ||||
|         run: | | ||||
|           yarn --frozen-lockfile | ||||
|           yarn install | ||||
|           yarn test:other | ||||
|           yarn test:code | ||||
|           yarn test:typecheck | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/locales-coverage.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -10,7 +10,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} | ||||
|  | ||||
| @@ -22,11 +22,11 @@ jobs: | ||||
|       - name: Create report file | ||||
|         run: | | ||||
|           yarn locales-coverage | ||||
|           FILE_CHANGED=$(git diff src/locales/percentages.json) | ||||
|           FILE_CHANGED=$(git diff packages/excalidraw/locales/percentages.json) | ||||
|           if [ ! -z "${FILE_CHANGED}" ]; then | ||||
|             git config --global user.name 'Excalidraw Bot' | ||||
|             git config --global user.email 'bot@excalidraw.com' | ||||
|             git add src/locales/percentages.json | ||||
|             git add packages/excalidraw/locales/percentages.json | ||||
|             git commit -am "Auto commit: Calculate translation coverage" | ||||
|             git push | ||||
|           fi | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/semantic-pr-title.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -11,6 +11,6 @@ jobs: | ||||
|   semantic: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: amannn/action-semantic-pull-request@v3.0.0 | ||||
|       - uses: amannn/action-semantic-pull-request@v5 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/workflows/size-limit.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -15,16 +15,14 @@ jobs: | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: 18.x | ||||
|       - name: Install | ||||
|         run: yarn --frozen-lockfile | ||||
|       - name: Install in src/packages/excalidraw | ||||
|         run: yarn --frozen-lockfile | ||||
|         working-directory: src/packages/excalidraw | ||||
|       - name: Install in packages/excalidraw | ||||
|         run: yarn | ||||
|         working-directory: packages/excalidraw | ||||
|         env: | ||||
|           CI: true | ||||
|       - uses: andresz1/size-limit-action@v1 | ||||
|         with: | ||||
|           github_token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           build_script: build:umd | ||||
|           build_script: build:esm | ||||
|           skip_step: install | ||||
|           directory: src/packages/excalidraw | ||||
|           directory: packages/excalidraw | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/test-coverage-pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,7 +16,7 @@ jobs: | ||||
|         with: | ||||
|           node-version: "18.x" | ||||
|       - name: "Install Deps" | ||||
|         run: yarn --frozen-lockfile | ||||
|         run: yarn install | ||||
|       - name: "Test Coverage" | ||||
|         run: yarn test:coverage | ||||
|       - name: "Report Coverage" | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,17 +1,19 @@ | ||||
| name: Tests | ||||
|  | ||||
| on: pull_request | ||||
| on: | ||||
|   push: | ||||
|     branches: master | ||||
|  | ||||
| jobs: | ||||
|   test: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - uses: actions/checkout@v4 | ||||
|       - name: Setup Node.js 18.x | ||||
|         uses: actions/setup-node@v2 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 18.x | ||||
|       - name: Install and test | ||||
|         run: | | ||||
|           yarn --frozen-lockfile | ||||
|           yarn install | ||||
|           yarn test:app | ||||
|   | ||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -21,10 +21,8 @@ npm-debug.log* | ||||
| package-lock.json | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| src/packages/excalidraw/types | ||||
| src/packages/excalidraw/example/public/bundle.js | ||||
| src/packages/excalidraw/example/public/excalidraw-assets-dev | ||||
| src/packages/excalidraw/example/public/excalidraw.development.js | ||||
| packages/excalidraw/types | ||||
| coverage | ||||
| dev-dist | ||||
| html | ||||
| meta*.json | ||||
							
								
								
									
										12
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						| @@ -2,16 +2,18 @@ FROM node:18 AS build | ||||
|  | ||||
| WORKDIR /opt/node_app | ||||
|  | ||||
| COPY package.json yarn.lock ./ | ||||
| RUN yarn --ignore-optional --network-timeout 600000 | ||||
| COPY . . | ||||
|  | ||||
| # do not ignore optional dependencies: | ||||
| # Error: Cannot find module @rollup/rollup-linux-x64-gnu | ||||
| RUN yarn --network-timeout 600000 | ||||
|  | ||||
| ARG NODE_ENV=production | ||||
|  | ||||
| COPY . . | ||||
| RUN yarn build:app:docker | ||||
|  | ||||
| FROM nginx:1.21-alpine | ||||
| FROM nginx:1.27-alpine | ||||
|  | ||||
| COPY --from=build /opt/node_app/build /usr/share/nginx/html | ||||
| COPY --from=build /opt/node_app/excalidraw-app/build /usr/share/nginx/html | ||||
|  | ||||
| HEALTHCHECK CMD wget -q -O /dev/null http://localhost || exit 1 | ||||
|   | ||||
							
								
								
									
										21
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -7,7 +7,7 @@ | ||||
|  | ||||
| <h4 align="center"> | ||||
|   <a href="https://excalidraw.com">Excalidraw Editor</a> | | ||||
|   <a href="https://blog.excalidraw.com">Blog</a> | | ||||
|   <a href="https://plus.excalidraw.com/blog">Blog</a> | | ||||
|   <a href="https://docs.excalidraw.com">Documentation</a> | | ||||
|   <a href="https://plus.excalidraw.com">Excalidraw+</a> | ||||
| </h4> | ||||
| @@ -34,6 +34,9 @@ | ||||
|   <a href="https://discord.gg/UexuTaE"> | ||||
|     <img alt="Chat on Discord" src="https://img.shields.io/discord/723672430744174682?color=738ad6&label=Chat%20on%20Discord&logo=discord&logoColor=ffffff&widge=false"/> | ||||
|   </a> | ||||
|   <a href="https://deepwiki.com/excalidraw/excalidraw"> | ||||
|     <img alt="Ask DeepWiki" src="https://deepwiki.com/badge.svg" /> | ||||
|   </a> | ||||
|   <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"/> | ||||
|   </a> | ||||
| @@ -63,7 +66,7 @@ The Excalidraw editor (npm package) supports: | ||||
| - 🏗️ Customizable. | ||||
| - 📷 Image support. | ||||
| - 😀 Shape libraries support. | ||||
| - 👅 Localization (i18n) support. | ||||
| - 🌐 Localization (i18n) support. | ||||
| - 🖼️ Export to PNG, SVG & clipboard. | ||||
| - 💾 Open format - export drawings as an `.excalidraw` json file. | ||||
| - ⚒️ Wide range of tools - rectangle, circle, diamond, arrow, line, free-draw, eraser... | ||||
| @@ -85,19 +88,17 @@ We'll be adding these features as drop-in plugins for the npm package in the fut | ||||
|  | ||||
| ## Quick start | ||||
|  | ||||
| Install the [Excalidraw npm package](https://www.npmjs.com/package/@excalidraw/excalidraw): | ||||
| **Note:** following instructions are for installing the Excalidraw [npm package](https://www.npmjs.com/package/@excalidraw/excalidraw) when integrating Excalidraw into your own app. To run the repository locally for development, please refer to our [Development Guide](https://docs.excalidraw.com/docs/introduction/development). | ||||
|  | ||||
| ``` | ||||
| Use `npm` or `yarn` to install the package. | ||||
|  | ||||
| ```bash | ||||
| npm install react react-dom @excalidraw/excalidraw | ||||
| ``` | ||||
|  | ||||
| or via yarn | ||||
|  | ||||
| ``` | ||||
| # or | ||||
| yarn add react react-dom @excalidraw/excalidraw | ||||
| ``` | ||||
|  | ||||
| Don't forget to check out our [Documentation](https://docs.excalidraw.com)! | ||||
| Check out our [documentation](https://docs.excalidraw.com/docs/@excalidraw/excalidraw/installation) for more details! | ||||
|  | ||||
| ## Contributing | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,3 @@ | ||||
| files: | ||||
|   - source: /src/locales/en.json | ||||
|     translation: /src/locales/%locale%.json | ||||
|   - source: /packages/excalidraw/locales/en.json | ||||
|     translation: /packages/excalidraw/locales/%locale%.json | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| Earlier we were using `renderFooter` prop to render custom footer which was removed in [#5970](https://github.com/excalidraw/excalidraw/pull/5970). Now you can pass a `Footer` component instead to render the custom UI for footer. | ||||
|  | ||||
| You will need to import the `Footer` component from the package and wrap your component with the Footer component. The `Footer` should a valid React Node. | ||||
| You will need to import the `Footer` component from the package and wrap your component with the Footer component. The `Footer` should be a valid React Node. | ||||
|  | ||||
| **Usage** | ||||
|  | ||||
| @@ -25,7 +25,7 @@ function App() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| This will only for `Desktop` devices. | ||||
| This will only work for `Desktop` devices. | ||||
|  | ||||
| For `mobile` you will need to render it inside the [MainMenu](#mainmenu). You can use the [`useDevice`](#useDevice) hook to check the type of device, this will be available only inside the `children` of `Excalidraw` component. | ||||
|  | ||||
| @@ -65,4 +65,4 @@ const App = () => ( | ||||
| // Need to render when code is span across multiple components | ||||
| // in Live Code blocks editor | ||||
| render(<App />); | ||||
| ``` | ||||
| ``` | ||||
|   | ||||
| @@ -133,7 +133,7 @@ function App() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Here is a [complete list](https://github.com/excalidraw/excalidraw/blob/master/src/components/mainMenu/DefaultItems.tsx) of the default items. | ||||
| Here is a [complete list](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/components/main-menu/DefaultItems.tsx) of the default items. | ||||
|  | ||||
| ### MainMenu.Group | ||||
|  | ||||
|   | ||||
| @@ -31,7 +31,7 @@ The welcome screen consists of two main groups of subcomponents: | ||||
|  | ||||
| <img | ||||
|   src={require("@site/static/img/welcome-screen-overview.png").default} | ||||
|   alt="Excalidraw logo: Sketch handrawn like diagrams." | ||||
|   alt="Excalidraw logo: Sketch hand-drawn like diagrams." | ||||
| /> | ||||
|  | ||||
| ### Center | ||||
|   | ||||
| @@ -8,15 +8,15 @@ | ||||
| import { FONT_FAMILY } from "@excalidraw/excalidraw"; | ||||
| ``` | ||||
|  | ||||
| `FONT_FAMILY` contains all the font families used in `Excalidraw` as explained below | ||||
| `FONT_FAMILY` contains all the font families used in `Excalidraw`. The default families are the following: | ||||
|  | ||||
| | Font Family | Description            | | ||||
| | ----------- | ---------------------- | | ||||
| | `Virgil`    | The `handwritten` font | | ||||
| | `Helvetica` | The `Normal` Font      | | ||||
| | `Cascadia`  | The `Code` Font        | | ||||
| | `Excalifont`    | The `Hand-drawn` font | | ||||
| | `Nunito` | The `Normal` Font      | | ||||
| | `Comic Shanns`  | The `Code` Font        | | ||||
|  | ||||
| Defaults to `FONT_FAMILY.Virgil` unless passed in `initialData.appState.currentItemFontFamily`. | ||||
| Pre-selected family is `FONT_FAMILY.Excalifont`, unless it's overriden with `initialData.appState.currentItemFontFamily`. | ||||
|  | ||||
| ### THEME | ||||
|  | ||||
| @@ -37,7 +37,7 @@ Defaults to `THEME.LIGHT` unless passed in `initialData.appState.theme` | ||||
|  | ||||
| ### MIME_TYPES | ||||
|  | ||||
| [`MIME_TYPES`](https://github.com/excalidraw/excalidraw/blob/master/src/constants.ts#L101) contains all the mime types supported by `Excalidraw`. | ||||
| [`MIME_TYPES`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/constants.ts#L101) contains all the mime types supported by `Excalidraw`. | ||||
|  | ||||
| **How to use ** | ||||
|  | ||||
|   | ||||
| @@ -2,9 +2,9 @@ | ||||
|  | ||||
| We support a simplified API to make it easier to generate Excalidraw elements programmatically. This API is in beta and subject to change before stable. You can check the [PR](https://github.com/excalidraw/excalidraw/pull/6546) for more details. | ||||
|  | ||||
| For this purpose we introduced a new type [`ExcalidrawElementSkeleton`](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133). This is the simplified version of [`ExcalidrawElement`](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L134) type with the minimum possible attributes so that creating elements programmatically is much easier (especially for cases like binding arrows or creating text containers). | ||||
| For this purpose we introduced a new type [`ExcalidrawElementSkeleton`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133). This is the simplified version of [`ExcalidrawElement`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L134) type with the minimum possible attributes so that creating elements programmatically is much easier (especially for cases like binding arrows or creating text containers). | ||||
|  | ||||
| The [`ExcalidrawElementSkeleton`](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133) can be converted to fully qualified Excalidraw elements by using [`convertToExcalidrawElements`](/docs/@excalidraw/excalidraw/api/excalidraw-element-skeleton#converttoexcalidrawelements). | ||||
| The [`ExcalidrawElementSkeleton`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133) can be converted to fully qualified Excalidraw elements by using [`convertToExcalidrawElements`](/docs/@excalidraw/excalidraw/api/excalidraw-element-skeleton#converttoexcalidrawelements). | ||||
|  | ||||
| ## convertToExcalidrawElements | ||||
|  | ||||
| @@ -19,7 +19,7 @@ convertToExcalidrawElements( | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | `elements` | [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L137) |  | The Excalidraw element Skeleton which needs to be converted to Excalidraw elements. | | ||||
| | `elements` | [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L137) |  | The Excalidraw element Skeleton which needs to be converted to Excalidraw elements. | | ||||
| | `opts` | `{ regenerateIds: boolean }` | ` {regenerateIds: true}` | By default `id` will be regenerated for all the elements irrespective of whether you pass the `id` so if you don't want the ids to regenerated, you can set this attribute to `false`. | | ||||
|  | ||||
| **_How to use_** | ||||
| @@ -71,7 +71,7 @@ function App() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| You can pass additional [`properties`](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L27) as well to decorate the shapes. | ||||
| You can pass additional [`properties`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L27) as well to decorate the shapes. | ||||
|  | ||||
| :::info | ||||
|  | ||||
| @@ -192,7 +192,7 @@ convertToExcalidrawElements([ | ||||
|  | ||||
| ### Text Containers | ||||
|  | ||||
| In addition to `type`, `x` and `y` properties, [`label`](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L124C7-L130C59) property is required for text containers. The `text` property in `label` is required, rest of the attributes are optional. | ||||
| In addition to `type`, `x` and `y` properties, [`label`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L124C7-L130C59) property is required for text containers. The `text` property in `label` is required, rest of the attributes are optional. | ||||
|  | ||||
| If you don't provide the dimensions of container, we calculate it based of the label dimensions. | ||||
|  | ||||
| @@ -326,7 +326,7 @@ convertToExcalidrawElements([ | ||||
|  | ||||
| ### Arrow bindings | ||||
|  | ||||
| To bind arrow to a shape you need to specify its [`start`](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L86) and [`end`](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L54) properties. You need to pass either `type` or `id` property in `start` and `end` properties, rest of the attributes are optional | ||||
| To bind arrow to a shape you need to specify its [`start`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L86) and [`end`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L54) properties. You need to pass either `type` or `id` property in `start` and `end` properties, rest of the attributes are optional | ||||
|  | ||||
| ```js | ||||
| convertToExcalidrawElements([ | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| <pre> | ||||
|   (api:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L616"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L616"> | ||||
|     ExcalidrawAPI | ||||
|   </a> | ||||
|   ) => void; | ||||
| @@ -13,16 +13,16 @@ Once the callback is triggered, you will need to store the api in state to acces | ||||
| ```jsx showLineNumbers | ||||
| export default function App() { | ||||
|   const [excalidrawAPI, setExcalidrawAPI] = useState(null); | ||||
|   return <Excalidraw excalidrawAPI={{(api)=> setExcalidrawAPI(api)}} />; | ||||
|   return <Excalidraw excalidrawAPI={(api)=> setExcalidrawAPI(api)} />; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| You can use this prop when you want to access some [Excalidraw APIs](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L616). We expose the below APIs :point_down: | ||||
| You can use this prop when you want to access some [Excalidraw APIs](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L616). We expose the below APIs :point_down: | ||||
|  | ||||
| | API | Signature | Usage | | ||||
| | --- | --- | --- | | ||||
| | [updateScene](#updatescene) | `function` | updates the scene with the sceneData | | ||||
| | [updateLibrary](#updatelibrary) | `function` | updates the scene with the sceneData | | ||||
| | [updateLibrary](#updatelibrary) | `function` | updates the library | | ||||
| | [addFiles](#addfiles) | `function` | add files data to the appState | | ||||
| | [resetScene](#resetscene) | `function` | Resets the scene. If `resetLoadingState` is passed as true then it will also force set the loading state to false. | | ||||
| | [getSceneElementsIncludingDeleted](#getsceneelementsincludingdeleted) | `function` | Returns all the elements including the deleted in the scene | | ||||
| @@ -37,7 +37,7 @@ You can use this prop when you want to access some [Excalidraw APIs](https://git | ||||
| | [setActiveTool](#setactivetool) | `function` | This API can be used to set the active tool | | ||||
| | [setCursor](#setcursor) | `function` | This API can be used to set customise the mouse cursor on the canvas | | ||||
| | [resetCursor](#resetcursor) | `function` | This API can be used to reset to default mouse cursor on the canvas | | ||||
| | [toggleMenu](#togglemenu) | `function` | Toggles specific menus on/off | | ||||
| | [toggleSidebar](#toggleSidebar) | `function` | Toggles specific sidebar on/off | | ||||
| | [onChange](#onChange) | `function` | Subscribes to change events | | ||||
| | [onPointerDown](#onPointerDown) | `function` | Subscribes to `pointerdown` events | | ||||
| | [onPointerUp](#onPointerUp) | `function` | Subscribes to `pointerup` events | | ||||
| @@ -52,7 +52,7 @@ Additionally `ready` and `readyPromise` from the API have been discontinued. The | ||||
|  | ||||
| <pre> | ||||
|   (scene:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L339"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L339"> | ||||
|     sceneData | ||||
|   </a> | ||||
|   ) => void | ||||
| @@ -62,10 +62,10 @@ You can use this function to update the scene with the sceneData. It accepts the | ||||
|  | ||||
| | Name | Type | Description | | ||||
| | --- | --- | --- | | ||||
| | `elements` | [`ImportedDataState["elements"]`](https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L38) | The `elements` to be updated in the scene | | ||||
| | `appState` | [`ImportedDataState["appState"]`](https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L39) | The `appState` to be updated in the scene. | | ||||
| | `collaborators` | <code>Map<string, <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L37">Collaborator></a></code> | The list of collaborators to be updated in the scene. | | ||||
| | `commitToHistory` | `boolean` | Implies if the `history (undo/redo)` should be recorded. Defaults to `false`. | | ||||
| | `elements` | [`ImportedDataState["elements"]`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L38) | The `elements` to be updated in the scene | | ||||
| | `appState` | [`ImportedDataState["appState"]`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L39) | The `appState` to be updated in the scene. | | ||||
| | `collaborators` | <code>Map<string, <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L37">Collaborator></a></code> | The list of collaborators to be updated in the scene. | | ||||
| | `captureUpdate` | [`CaptureUpdateAction`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/store.ts#L40) | Controls which updates should be captured by the `Store`. Captured updates are emmitted and listened to by other components, such as `History` for undo / redo purposes. | | ||||
|  | ||||
| ```jsx live | ||||
| function App() { | ||||
| @@ -105,6 +105,7 @@ function App() { | ||||
|       appState: { | ||||
|         viewBackgroundColor: "#edf2ff", | ||||
|       }, | ||||
|       captureUpdate: CaptureUpdateAction.IMMEDIATELY, | ||||
|     }; | ||||
|     excalidrawAPI.updateScene(sceneData); | ||||
|   }; | ||||
| @@ -115,23 +116,35 @@ function App() { | ||||
|       <button className="custom-button" onClick={updateScene}> | ||||
|         Update Scene | ||||
|       </button> | ||||
|       <Excalidraw ref={(api) => setExcalidrawAPI(api)} /> | ||||
|       <Excalidraw excalidrawAPI={(api) => setExcalidrawAPI(api)} /> | ||||
|     </div> | ||||
|   ); | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### captureUpdate | ||||
|  | ||||
| You can use the `captureUpdate` to influence undo / redo behaviour. | ||||
|  | ||||
| > **NOTE**: Some updates are not observed by the store / history - i.e. updates to `collaborators` object or parts of `AppState` which are not observed (not `ObservedAppState`). Such updates will never make it to the undo / redo stacks, regardless of the passed `captureUpdate` value. | ||||
|  | ||||
| |  | `captureUpdate` value | Notes | | ||||
| | --- | --- | --- | | ||||
| | _Immediately undoable_ | `CaptureUpdateAction.IMMEDIATELY` | Use for updates which should be captured. Should be used for most of the local updates. These updates will _immediately_ make it to the local undo / redo stacks. | | ||||
| | _Eventually undoable_ | `CaptureUpdateAction.EVENTUALLY` | Use for updates which should not be captured immediately - likely exceptions which are part of some async multi-step process. Otherwise, all such updates would end up being captured with the next `CaptureUpdateAction.IMMEDIATELY` - triggered either by the next `updateScene` or internally by the editor. These updates will _eventually_ make it to the local undo / redo stacks. | | ||||
| | _Never undoable_ | `CaptureUpdateAction.NEVER` | Use for updates which should never be recorded, such as remote updates or scene initialization. These updates will _never_ make it to the local undo / redo stacks. | | ||||
|  | ||||
| ### updateLibrary | ||||
|  | ||||
| <pre> | ||||
|   (opts: { <br /> libraryItems:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L249"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L249"> | ||||
|     LibraryItemsSource | ||||
|   </a> | ||||
|   ;<br /> merge?: boolean; <br /> prompt?: boolean; | ||||
|   <br /> openLibraryMenu?: boolean; | ||||
|   <br /> defaultStatus?: "unpublished" | "published"; <br /> }) => Promise< | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L246"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L246"> | ||||
|     LibraryItems | ||||
|   </a> | ||||
|   > | ||||
| @@ -141,7 +154,7 @@ You can use this function to update the library. It accepts the below attributes | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | `libraryItems` | [LibraryItemsSource](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L249) | \_ | The `libraryItems` to be replaced/merged with current library | | ||||
| | `libraryItems` | [LibraryItemsSource](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L249) | \_ | The `libraryItems` to be replaced/merged with current library | | ||||
| | `merge` | boolean | `false` | Whether to merge with existing library items. | | ||||
| | `prompt` | boolean | `false` | Whether to prompt user for confirmation. | | ||||
| | `openLibraryMenu` | boolean | `false` | Keep the library menu open after library is updated. | | ||||
| @@ -188,8 +201,8 @@ function App() { | ||||
|         Update Library | ||||
|       </button> | ||||
|       <Excalidraw | ||||
|         ref={(api) => setExcalidrawAPI(api)} | ||||
|         // initial data retrieved from https://github.com/excalidraw/excalidraw/blob/master/dev-docs/src/initialData.js | ||||
|         excalidrawAPI={(api) => setExcalidrawAPI(api)} | ||||
|         // initial data retrieved from https://github.com/excalidraw/excalidraw/blob/master/dev-docs/packages/excalidraw/initialData.js | ||||
|         initialData={{ | ||||
|           libraryItems: initialData.libraryItems, | ||||
|           appState: { openSidebar: "library" }, | ||||
| @@ -204,7 +217,7 @@ function App() { | ||||
|  | ||||
| <pre> | ||||
|   (files:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L59"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L59"> | ||||
|     BinaryFileData | ||||
|   </a> | ||||
|   ) => void | ||||
| @@ -224,7 +237,7 @@ Resets the scene. If `resetLoadingState` is passed as true then it will also for | ||||
|  | ||||
| <pre> | ||||
|   () =>{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L115"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L115"> | ||||
|     ExcalidrawElement[] | ||||
|   </a> | ||||
| </pre> | ||||
| @@ -235,7 +248,7 @@ Returns all the elements including the deleted in the scene. | ||||
|  | ||||
| <pre> | ||||
|   () => NonDeleted< | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L115"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L115"> | ||||
|     ExcalidrawElement | ||||
|   </a> | ||||
|   []> | ||||
| @@ -247,7 +260,7 @@ Returns all the elements excluding the deleted in the scene | ||||
|  | ||||
| <pre> | ||||
|   () =>{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95"> | ||||
|     AppState | ||||
|   </a> | ||||
| </pre> | ||||
| @@ -288,7 +301,7 @@ Scroll the nearest element out of the elements supplied to the center of the vie | ||||
|  | ||||
| | Attribute | type | default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | target | [ExcalidrawElement](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L115) | [ExcalidrawElement[]](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L115) | All scene elements | The element(s) to scroll to. | | ||||
| | target | [ExcalidrawElement](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L115) | [ExcalidrawElement[]](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L115) | All scene elements | The element(s) to scroll to. | | ||||
| | opts.fitToContent | boolean | false | Whether to fit the elements to viewport by automatically changing zoom as needed. Note that the zoom range is between 10%-100%. | | ||||
| | opts.fitToViewport | boolean | false | Similar to fitToContent but the zoom range is not limited. If elements are smaller than the viewport, zoom will go above 100%. | | ||||
| | opts.viewportZoomFactor | number | 0.7 | when fitToViewport=true, how much screen should the content cover, between 0.1 (10%) and 1 (100%) | | ||||
| @@ -336,7 +349,7 @@ The unique id of the excalidraw component. This can be used to identify the exca | ||||
|  | ||||
| <pre> | ||||
|   () =>{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L82"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L82"> | ||||
|     files | ||||
|   </a> | ||||
| </pre> | ||||
| @@ -364,7 +377,7 @@ This API has the below signature. It sets the `tool` passed in param as the acti | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | `type` | [ToolType](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L91) | `selection` | The tool type which should be set as active tool. When setting `image` as active tool, the insertion onto canvas when using image tool is disabled by default, so you can enable it by setting `insertOnCanvasDirectly` to `true` | | ||||
| | `type` | [ToolType](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L91) | `selection` | The tool type which should be set as active tool. When setting `image` as active tool, the insertion onto canvas when using image tool is disabled by default, so you can enable it by setting `insertOnCanvasDirectly` to `true` | | ||||
| | `locked` | `boolean` | `false` | Indicates whether the the active tool should be locked. It behaves the same way when using the `lock` tool in the editor interface | | ||||
|  | ||||
| ## setCursor | ||||
|   | ||||
| @@ -1,18 +1,18 @@ | ||||
| # initialData | ||||
|  | ||||
| <pre> | ||||
| { elements?: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a>, appState?: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a> } | ||||
| { elements?: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a>, appState?: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a> } | ||||
| </pre> | ||||
|  | ||||
| This helps to load Excalidraw with `initialData`. It must be an object or a [promise](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise/Promise) which resolves to an object containing the below optional fields. | ||||
|  | ||||
| | Name | Type | Description | | ||||
| | --- | --- | --- | | ||||
| | `elements` | [ExcalidrawElement[]](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114) | The `elements` with which `Excalidraw` should be mounted. | | ||||
| | `appState` | [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95) | The `AppState` with which `Excalidraw` should be mounted. | | ||||
| | `elements` | [ExcalidrawElement[]](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114) | The `elements` with which `Excalidraw` should be mounted. | | ||||
| | `appState` | [AppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95) | The `AppState` with which `Excalidraw` should be mounted. | | ||||
| | `scrollToContent` | `boolean` | This attribute indicates whether to `scroll` to the nearest element to center once `Excalidraw` is mounted. By default, it will not scroll the nearest element to the center. Make sure you pass `initialData.appState.scrollX` and `initialData.appState.scrollY` when `scrollToContent` is false so that scroll positions are retained | | ||||
| | `libraryItems` | [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L247) | Promise<[LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200)> | This library items with which `Excalidraw` should be mounted. | | ||||
| | `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L82) | The `files` added to the scene. | | ||||
| | `libraryItems` | [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L247) | Promise<[LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L200)> | This library items with which `Excalidraw` should be mounted. | | ||||
| | `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L82) | The `files` added to the scene. | | ||||
|  | ||||
| You might want to use this when you want to load excalidraw with some initial elements and app state. | ||||
|  | ||||
|   | ||||
| @@ -5,35 +5,37 @@ All `props` are _optional_. | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | [`initialData`](/docs/@excalidraw/excalidraw/api/props/initialdata) | `object` | `null` | <code>Promise<object | null></code> | `null` | The initial data with which app loads. | | ||||
| | [`excalidrawAPI`](/docs/@excalidraw/excalidraw/api/props/excalidraw-api) | `function` | _ | Callback triggered with the excalidraw api once rendered | | ||||
| | [`isCollaborating`](#iscollaborating) | `boolean` | _ | This indicates if the app is in `collaboration` mode | | ||||
| | [`onChange`](#onchange) | `function` | _ | This callback is triggered whenever the component updates due to any change. This callback will receive the excalidraw `elements` and the current `app state`. | | ||||
| | [`onPointerUpdate`](#onpointerupdate) | `function` | _ | Callback triggered when mouse pointer is updated. | | ||||
| | [`onPointerDown`](#onpointerdown) | `function` | _ | This prop if passed gets triggered on pointer down evenets | | ||||
| | [`onScrollChange`](#onscrollchange) | `function` | _ | This prop if passed gets triggered when scrolling the canvas. | | ||||
| | [`onPaste`](#onpaste) | `function` | _ | Callback to be triggered if passed when the something is pasted in to the scene | | ||||
| | [`onLibraryChange`](#onlibrarychange) | `function` | _ | The callback if supplied is triggered when the library is updated and receives the library items. | | ||||
| | [`onLinkOpen`](#onlinkopen) | `function` | _ | The callback if supplied is triggered when any link is opened. | | ||||
| | [`excalidrawAPI`](/docs/@excalidraw/excalidraw/api/props/excalidraw-api) | `function` | \_ | Callback triggered with the excalidraw api once rendered | | ||||
| | [`isCollaborating`](#iscollaborating) | `boolean` | \_ | This indicates if the app is in `collaboration` mode | | ||||
| | [`onChange`](#onchange) | `function` | \_ | This callback is triggered whenever the component updates due to any change. This callback will receive the excalidraw `elements` and the current `app state`. | | ||||
| | [`onPointerUpdate`](#onpointerupdate) | `function` | \_ | Callback triggered when mouse pointer is updated. | | ||||
| | [`onPointerDown`](#onpointerdown) | `function` | \_ | This prop if passed gets triggered on pointer down events | | ||||
| | [`onScrollChange`](#onscrollchange) | `function` | \_ | This prop if passed gets triggered when scrolling the canvas. | | ||||
| | [`onPaste`](#onpaste) | `function` | \_ | Callback to be triggered if passed when something is pasted into the scene | | ||||
| | [`onLibraryChange`](#onlibrarychange) | `function` | \_ | The callback if supplied is triggered when the library is updated and receives the library items. | | ||||
| | [`generateLinkForSelection`](#generatelinkforselection) | `function` | \_ | Allows you to override `url` generation when linking to Excalidraw elements. | | ||||
| | [`onLinkOpen`](#onlinkopen) | `function` | \_ | The callback if supplied is triggered when any link is opened. | | ||||
| | [`langCode`](#langcode) | `string` | `en` | Language code string to be used in Excalidraw | | ||||
| | [`renderTopRightUI`](/docs/@excalidraw/excalidraw/api/props/render-props#rendertoprightui) | `function` | _ | Render function that renders custom UI in top right corner | | ||||
| | [`renderCustomStats`](/docs/@excalidraw/excalidraw/api/props/render-props#rendercustomstats) | `function` | _ | Render function that can be used to render custom stats on the stats dialog. | | ||||
| | [`viewModeEnabled`](#viewmodeenabled) | `boolean` | _ | This indicates if the app is in `view` mode. | | ||||
| | [`zenModeEnabled`](#zenmodeenabled) | `boolean` | _ | This indicates if the `zen` mode is enabled | | ||||
| | [`gridModeEnabled`](#gridmodeenabled) | `boolean` | _ | This indicates if the `grid` mode is enabled | | ||||
| | [`libraryReturnUrl`](#libraryreturnurl) | `string` | _ | What URL should [libraries.excalidraw.com](https://libraries.excalidraw.com) be installed to | | ||||
| | [`renderTopRightUI`](/docs/@excalidraw/excalidraw/api/props/render-props#rendertoprightui) | `function` | \_ | Render function that renders custom UI in top right corner | | ||||
| | [`renderCustomStats`](/docs/@excalidraw/excalidraw/api/props/render-props#rendercustomstats) | `function` | \_ | Render function that can be used to render custom stats on the stats dialog. | | ||||
| | [`viewModeEnabled`](#viewmodeenabled) | `boolean` | \_ | This indicates if the app is in `view` mode. | | ||||
| | [`zenModeEnabled`](#zenmodeenabled) | `boolean` | \_ | This indicates if the `zen` mode is enabled | | ||||
| | [`gridModeEnabled`](#gridmodeenabled) | `boolean` | \_ | This indicates if the `grid` mode is enabled | | ||||
| | [`libraryReturnUrl`](#libraryreturnurl) | `string` | \_ | What URL should [libraries.excalidraw.com](https://libraries.excalidraw.com) be installed to | | ||||
| | [`theme`](#theme) | `"light"` | `"dark"` | `"light"` | The theme of the Excalidraw component | | ||||
| | [`name`](#name) | `string` |  | Name of the drawing | | ||||
| | [`UIOptions`](/docs/@excalidraw/excalidraw/api/props/ui-options) | `object` | [DEFAULT UI OPTIONS](https://github.com/excalidraw/excalidraw/blob/master/src/constants.ts#L151) | To customise UI options. Currently we support customising [`canvas actions`](#canvasactions) | | ||||
| | [`UIOptions`](/docs/@excalidraw/excalidraw/api/props/ui-options) | `object` | [DEFAULT UI OPTIONS](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/constants.ts#L151) | To customise UI options. Currently we support customising [`canvas actions`](/docs/@excalidraw/excalidraw/api/props/ui-options#canvasactions) | | ||||
| | [`detectScroll`](#detectscroll) | `boolean` | `true` | Indicates whether to update the offsets when nearest ancestor is scrolled. | | ||||
| | [`handleKeyboardGlobally`](#handlekeyboardglobally) | `boolean` | `false` | Indicates whether to bind the keyboard events to document. | | ||||
| | [`autoFocus`](#autofocus) | `boolean` | `false` | indicates whether to focus the Excalidraw component on page load | | ||||
| | [`generateIdForFile`](#generateidforfile) | `function` | _ | Allows you to override `id` generation for files added on canvas | | ||||
| | [`validateEmbeddable`](#validateEmbeddable) | string[] | `boolean | RegExp | RegExp[] | ((link: string) => boolean | undefined)` | \_ | use for custom src url validation | | ||||
| | [`autoFocus`](#autofocus) | `boolean` | `false` | Indicates whether to focus the Excalidraw component on page load | | ||||
| | [`generateIdForFile`](#generateidforfile) | `function` | \_ | Allows you to override `id` generation for files added on canvas | | ||||
| | [`validateEmbeddable`](#validateembeddable) | `string[]` \| `boolean` \| `RegExp` \| `RegExp[]` \| <code>((link: string) => boolean | undefined)</code> | \_ | use for custom src url validation | | ||||
| | [`renderEmbeddable`](/docs/@excalidraw/excalidraw/api/props/render-props#renderEmbeddable) | `function` | \_ | Render function that can override the built-in `<iframe>` | | ||||
| | [`renderScrollbars`] | `boolean`| | `false` | Indicates whether scrollbars will be shown | ||||
|  | ||||
| ### Storing custom data on Excalidraw elements | ||||
|  | ||||
| Beyond attributes that Excalidraw elements already support, you can store `custom` data on each `element` in a `customData` object. The type of the attribute is [`Record<string, any>`](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L66) and is optional. | ||||
| Beyond attributes that Excalidraw elements already support, you can store `custom` data on each `element` in a `customData` object. The type of the attribute is [`Record<string, any>`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L66) and is optional. | ||||
|  | ||||
| You can use this to add any extra information you need to keep track of. | ||||
|  | ||||
| @@ -59,11 +61,11 @@ Every time component updates, this callback if passed will get triggered and has | ||||
| (excalidrawElements, appState, files) => void; | ||||
| ``` | ||||
|  | ||||
| 1. `excalidrawElements`: Array of [excalidrawElements](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114) in the scene. | ||||
| 1. `excalidrawElements`: Array of [excalidrawElements](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114) in the scene. | ||||
|  | ||||
| 2. `appState`: [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95) of the scene. | ||||
| 2. `appState`: [AppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95) of the scene. | ||||
|  | ||||
| 3. `files`: The [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L64) which are added to the scene. | ||||
| 3. `files`: The [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L64) which are added to the scene. | ||||
|  | ||||
| Here you can try saving the data to your backend or local storage for example. | ||||
|  | ||||
| @@ -79,28 +81,27 @@ This callback is triggered when mouse pointer is updated. | ||||
|  | ||||
| 2.`button`: The position of the button. This will be one of `["down", "up"]` | ||||
|  | ||||
| 3.`pointersMap`: [`pointers`](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L131) map of the scene | ||||
| 3.`pointersMap`: [`pointers`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L131) map of the scene | ||||
|  | ||||
| ```js | ||||
| (exportedElements, appState, canvas) => void | ||||
| ``` | ||||
|  | ||||
| 1. `exportedElements`: An array of [non deleted elements](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L87) which needs to be exported. | ||||
| 2. `appState`: [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95) of the scene. | ||||
| 1. `exportedElements`: An array of [non deleted elements](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L87) which needs to be exported. | ||||
| 2. `appState`: [AppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95) of the scene. | ||||
| 3. `canvas`: The `HTMLCanvasElement` of the scene. | ||||
|  | ||||
| ### onPointerDown | ||||
|  | ||||
| This prop if passed will be triggered on pointer down events and has the below signature. | ||||
|  | ||||
|  | ||||
| <pre> | ||||
| (activeTool:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L115"> | ||||
|   (activeTool:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L115"> | ||||
|     {" "} | ||||
|     AppState["activeTool"] | ||||
|   </a> | ||||
|   , pointerDownState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L424"> | ||||
|   , pointerDownState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L424"> | ||||
|     PointerDownState | ||||
|   </a>) => void | ||||
| </pre> | ||||
| @@ -119,7 +120,7 @@ This callback is triggered if passed when something is pasted into the scene. Yo | ||||
|  | ||||
| <pre> | ||||
|   (data:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/clipboard.ts#L18"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/clipboard.ts#L18"> | ||||
|     ClipboardData | ||||
|   </a> | ||||
|   , event: ClipboardEvent | null) => boolean | ||||
| @@ -135,7 +136,7 @@ This callback if supplied will get triggered when the library is updated and has | ||||
|  | ||||
| <pre> | ||||
|   (items:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L200"> | ||||
|     LibraryItems | ||||
|   </a> | ||||
|   ) => void | Promise<any> | ||||
| @@ -143,13 +144,21 @@ This callback if supplied will get triggered when the library is updated and has | ||||
|  | ||||
| It is invoked with empty items when user clears the library. You can use this callback when you want to do something additional when library is updated for example persisting it to local storage. | ||||
|  | ||||
| ### generateLinkForSelection | ||||
|  | ||||
| This prop if passed will be used to replace the default link generation function. The idea is that the host app can take over the creation of element links, which can be used to navigate to a particular element or a group. If the host app chooses a different key for element link id, then the host app should also take care of the handling and the navigation in `onLinkOpen`. | ||||
|  | ||||
| ```tsx | ||||
| (id: string, type: "element" | "group") => string; | ||||
| ``` | ||||
|  | ||||
| ### onLinkOpen | ||||
|  | ||||
| This prop if passed will be triggered when clicked on `link`. To handle the redirect yourself (such as when using your own router for internal links), you must call `event.preventDefault()`. | ||||
|  | ||||
| <pre> | ||||
|   (element:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114"> | ||||
|     ExcalidrawElement | ||||
|   </a> | ||||
|   , event: CustomEvent<{ nativeEvent: MouseEvent }>) => void | ||||
| @@ -182,7 +191,7 @@ const onLinkOpen: ExcalidrawProps["onLinkOpen"] = useCallback( | ||||
|  | ||||
| ### langCode | ||||
|  | ||||
| Determines the `language` of the UI. It should be one of the [available language codes](https://github.com/excalidraw/excalidraw/blob/master/src/i18n.ts#L14). Defaults to `en` (English). We also export default language and supported languages which you can import as shown below. | ||||
| Determines the `language` of the UI. It should be one of the [available language codes](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/i18n.ts#L14). Defaults to `en` (English). We also export default language and supported languages which you can import as shown below. | ||||
|  | ||||
| ```js | ||||
| import { defaultLang, languages } from "@excalidraw/excalidraw"; | ||||
| @@ -191,7 +200,7 @@ import { defaultLang, languages } from "@excalidraw/excalidraw"; | ||||
| | name | type | | ||||
| | --- | --- | | ||||
| | `defaultLang` | `string` | | ||||
| | `languages` | [`Language[]`](https://github.com/excalidraw/excalidraw/blob/master/src/i18n.ts#L15) | | ||||
| | `languages` | [`Language[]`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/i18n.ts#L15) | | ||||
|  | ||||
| ### viewModeEnabled | ||||
|  | ||||
| @@ -207,8 +216,7 @@ This prop indicates whether the shows the grid. When supplied, the value takes p | ||||
|  | ||||
| ### libraryReturnUrl | ||||
|  | ||||
| If supplied, this URL will be used when user tries to install a library from [libraries.excalidraw.com](https://libraries.excalidraw.com).   | ||||
| Defaults to _window.location.origin + window.location.pathname_. To install the libraries in the same tab from which it was opened, you need to set `window.name` (to any alphanumeric string) — if it's not set it will open in a new tab. | ||||
| If supplied, this URL will be used when user tries to install a library from [libraries.excalidraw.com](https://libraries.excalidraw.com). Defaults to _window.location.origin + window.location.pathname_. To install the libraries in the same tab from which it was opened, you need to set `window.name` (to any alphanumeric string) — if it's not set it will open in a new tab. | ||||
|  | ||||
| ### theme | ||||
|  | ||||
| @@ -220,7 +228,6 @@ You can use [`THEME`](/docs/@excalidraw/excalidraw/api/utils#theme) to specify t | ||||
|  | ||||
| This prop sets the `name` of the drawing which will be used when exporting the drawing. When supplied, the value takes precedence over _intialData.appState.name_, the `name` will be fully controlled by host app and the users won't be able to edit from within Excalidraw. | ||||
|  | ||||
|  | ||||
| ### detectScroll | ||||
|  | ||||
| Indicates whether Excalidraw should listen for `scroll` event on the nearest scrollable container in the DOM tree and recompute the coordinates (e.g. to correctly handle the cursor) when the component's position changes. You can disable this when you either know this doesn't affect your app or you want to take care of it yourself (calling the [`refresh()`](#ref) method). | ||||
|   | ||||
| @@ -4,7 +4,7 @@ | ||||
|  | ||||
| <pre> | ||||
|   (isMobile: boolean, appState: | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95"> | ||||
|     AppState | ||||
|   </a>) => JSX | null | ||||
| </pre> | ||||
| @@ -66,7 +66,7 @@ function App() { | ||||
|  | ||||
| <pre> | ||||
|   (element: NonDeleted<ExcalidrawEmbeddableElement>, appState:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95"> | ||||
|     AppState | ||||
|   </a> | ||||
|   ) => JSX.Element | null | ||||
|   | ||||
| @@ -4,7 +4,7 @@ This prop can be used to customise UI of Excalidraw. Currently we support custom | ||||
|  | ||||
| <pre> | ||||
|   { | ||||
|   <br /> canvasActions?: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L372"> | ||||
|   <br /> canvasActions?: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L372"> | ||||
|     CanvasActions | ||||
|   </a>, <br /> dockedSidebarBreakpoint?: number, <br /> welcomeScreen?: boolean <br /> | ||||
|  | ||||
| @@ -55,7 +55,7 @@ If `UIOptions.canvasActions.export` is `false` the export button will not be ren | ||||
|  | ||||
| ## dockedSidebarBreakpoint | ||||
|  | ||||
| This prop indicates at what point should we break to a docked, permanent sidebar. If not passed it defaults to [`MQ_RIGHT_SIDEBAR_MAX_WIDTH_PORTRAIT`](https://github.com/excalidraw/excalidraw/blob/master/src/constants.ts#L161).   | ||||
| This prop indicates at what point should we break to a docked, permanent sidebar. If not passed it defaults to [`MQ_RIGHT_SIDEBAR_MAX_WIDTH_PORTRAIT`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/constants.ts#L161).   | ||||
| If the _width_ of the _excalidraw_ container exceeds _dockedSidebarBreakpoint_, the sidebar will be `dockable` and the button to `dock` the sidebar will be shown   | ||||
| If user choses to `dock` the sidebar, it will push the right part of the UI towards the left, making space for the sidebar as shown below. | ||||
|  | ||||
| @@ -73,9 +73,9 @@ function App() { | ||||
|  | ||||
| ## tools | ||||
|  | ||||
| This `prop ` controls the visibility of the tools in the editor. | ||||
| This `prop` controls the visibility of the tools in the editor. | ||||
| Currently you can control the visibility of `image` tool via this prop. | ||||
|  | ||||
| | Prop | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | image | boolean | true | Decides whether `image` tool should be visible. | ||||
| | image | boolean | true | Decides whether `image` tool should be visible. | ||||
|   | ||||
| @@ -20,16 +20,16 @@ exportToCanvas({<br/>  | ||||
|   getDimensions,<br/>  | ||||
|   files,<br/>  | ||||
|   exportPadding?: number;<br/> | ||||
| }: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L21">ExportOpts</a> | ||||
| }: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/utils/export.ts#L24">ExportOpts</a> | ||||
| </pre> | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | `elements` | [Excalidraw Element []](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114) |  | The elements to be exported to canvas. | | ||||
| | `appState` | [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L23) | [Default App State](https://github.com/excalidraw/excalidraw/blob/master/src/appState.ts#L17) | The app state of the scene. | | ||||
| | `elements` | [Excalidraw Element []](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114) |  | The elements to be exported to canvas. | | ||||
| | `appState` | [AppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/packages/utils.ts#L23) | [Default App State](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/appState.ts#L17) | The app state of the scene. | | ||||
| | [`getDimensions`](#getdimensions) | `function` | _ | A function which returns the `width`, `height`, and optionally `scale` (defaults to  `1`), with which canvas is to be exported. | | ||||
| | `maxWidthOrHeight` | `number` | _ | The maximum `width` or `height` of the exported image. If provided, `getDimensions` is ignored. | | ||||
| | `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L59) | _ | The files added to the scene. | | ||||
| | `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L59) | _ | The files added to the scene. | | ||||
| | `exportPadding` | `number` | `10` | The `padding` to be added on canvas. | | ||||
|  | ||||
|  | ||||
| @@ -90,7 +90,7 @@ function App() { | ||||
|         <img src={canvasUrl} alt="" /> | ||||
|       </div> | ||||
|       <div style={{ height: "400px" }}> | ||||
|         <Excalidraw ref={(api) => setExcalidrawAPI(api)} | ||||
|         <Excalidraw excalidrawAPI={(api) => setExcalidrawAPI(api)} | ||||
| /> | ||||
|       </div> | ||||
|     </> | ||||
| @@ -105,7 +105,7 @@ function App() { | ||||
|  | ||||
| <pre> | ||||
| exportToBlob(<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L14">ExportOpts</a> & {<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/packages/utils.ts#L14">ExportOpts</a> & {<br/>  | ||||
|   mimeType?: string,<br/>  | ||||
|   quality?: number,<br/>  | ||||
|   exportPadding?: number;<br/> | ||||
| @@ -134,16 +134,16 @@ Returns a promise which resolves with a [blob](https://developer.mozilla.org/en- | ||||
| <pre> | ||||
| exportToSvg({<br/>  | ||||
|   elements:   | ||||
|     <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114"> | ||||
|     <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114"> | ||||
|       ExcalidrawElement[] | ||||
|     </a>,<br/>  | ||||
|   appState: | ||||
|     <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95"> AppState | ||||
|     <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95"> AppState | ||||
|     </a>,<br/>  | ||||
|   exportPadding: number,<br/>  | ||||
|   metadata: string,<br/>  | ||||
|   files:  | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L59"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L59"> | ||||
|       BinaryFiles | ||||
|     </a>,<br/> | ||||
| }); | ||||
| @@ -151,10 +151,10 @@ exportToSvg({<br/>  | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | elements | [Excalidraw Element []](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114) |  | The elements to exported as `svg `| | ||||
| | appState | [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95) | [defaultAppState](https://github.com/excalidraw/excalidraw/blob/master/src/appState.ts#L11) | The `appState` of the scene | | ||||
| | elements | [Excalidraw Element []](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114) |  | The elements to exported as `svg `| | ||||
| | appState | [AppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95) | [defaultAppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/appState.ts#L11) | The `appState` of the scene | | ||||
| | exportPadding | number | 10 | The `padding` to be added on canvas | | ||||
| | files | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L64) | undefined | The `files` added to the scene. | | ||||
| | files | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L64) | undefined | The `files` added to the scene. | | ||||
|  | ||||
| This function returns a promise which resolves to `svg` of the exported drawing. | ||||
|  | ||||
| @@ -164,7 +164,7 @@ This function returns a promise which resolves to `svg` of the exported drawing. | ||||
|  | ||||
| <pre> | ||||
| exportToClipboard(<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L21">ExportOpts</a> & {<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/packages/utils.ts#L21">ExportOpts</a> & {<br/>  | ||||
|   mimeType?: string,<br/>  | ||||
|   quality?: number;<br/>  | ||||
|   type: 'png' | 'svg' |'json'<br/> | ||||
|   | ||||
| @@ -8,7 +8,7 @@ id: "restore" | ||||
| **_Signature_** | ||||
|  | ||||
| <pre> | ||||
| restoreAppState(appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L34">ImportedDataState["appState"]</a>,<br/>  localAppState: Partial<<a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a>> | null): <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a> | ||||
| restoreAppState(appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L34">ImportedDataState["appState"]</a>,<br/>  localAppState: Partial<<a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a>> | null): <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a> | ||||
| </pre> | ||||
|  | ||||
| **_How to use_** | ||||
| @@ -17,7 +17,7 @@ restoreAppState(appState: <a href="https://github.com/excalidraw/excalidraw/blob | ||||
| import { restoreAppState } from "@excalidraw/excalidraw"; | ||||
| ``` | ||||
|  | ||||
| This function will make sure all the `keys` have appropriate `values` in [appState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95) and if any key is missing, it will be set to its `default` value. | ||||
| This function will make sure all the `keys` have appropriate `values` in [appState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95) and if any key is missing, it will be set to its `default` value. | ||||
|  | ||||
| When `localAppState` is supplied, it's used in place of values that are missing (`undefined`) in `appState` instead of the defaults.   | ||||
| Use this as a way to not override user's defaults if you persist them. | ||||
| @@ -29,16 +29,16 @@ You can pass `null` / `undefined` if not applicable. | ||||
|  | ||||
| <pre> | ||||
| restoreElements( | ||||
|   elements: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ImportedDataState["elements"]</a>,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a> | null | undefined): <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a>,<br/>  | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean }<br/> | ||||
|   elements: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ImportedDataState["elements"]</a>,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a> | null | undefined): <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a>,<br/>  | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean, normalizeIndices?: boolean }<br/> | ||||
| ) | ||||
| </pre> | ||||
|  | ||||
| | Prop | Type | Description | | ||||
| | ---- | ---- | ---- | | ||||
| | `elements` | <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ImportedDataState["elements"]</a> | The `elements` to be restored | | ||||
| | [`localElements`](#localelements) | <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a> | null | undefined |  When `localElements` are supplied, they are used to ensure that existing restored elements reuse `version` (and increment it), and regenerate `versionNonce`. | | ||||
| | `elements` | <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ImportedDataState["elements"]</a> | The `elements` to be restored | | ||||
| | [`localElements`](#localelements) | <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a> | null | undefined |  When `localElements` are supplied, they are used to ensure that existing restored elements reuse `version` (and increment it), and regenerate `versionNonce`. | | ||||
| | [`opts`](#opts) | `Object` | The extra optional parameter to configure restored elements | ||||
|  | ||||
| #### localElements | ||||
| @@ -51,8 +51,9 @@ The extra optional parameter to configure restored elements. It has the followin | ||||
|  | ||||
| | Prop | Type | Description| | ||||
| | --- | --- | ------| | ||||
| | `refreshDimensions` | `boolean` | Indicates whether we should also `recalculate` text element dimensions. Since this is a potentially costly operation, you may want to disable it if you restore elements in tight loops, such as during collaboration. | | ||||
| | `repairBindings` |`boolean` | Indicates whether the `bindings` for the elements should be repaired. This is to make sure there are no containers with non existent bound text element id and no bound text elements with non existent container id. | | ||||
| | `refreshDimensions` | `boolean` | Indicates whether we should also _recalculate_ text element dimensions. Since this is a potentially costly operation, you may want to disable it if you restore elements in tight loops, such as during collaboration. | | ||||
| | `repairBindings` |`boolean` | Indicates whether the _bindings_ for the elements should be repaired. This is to make sure there are no containers with non existent bound text element id and no bound text elements with non existent container id. | | ||||
| | `normalizeIndices` |`boolean` | Indicates whether _fractional indices_ for the elements should be normalized. This is to prevent possible issues caused by using stale (too old, too long) indices. | | ||||
|  | ||||
| **_How to use_** | ||||
|  | ||||
| @@ -70,15 +71,15 @@ Parameter `refreshDimensions` indicates whether we should also `recalculate` tex | ||||
|  | ||||
| <pre> | ||||
| restore( | ||||
|   data: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L34">ImportedDataState</a>,<br/>  | ||||
|   localAppState: Partial<<a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a>> | null | undefined,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a> | null | undefined<br/>): <a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L4">DataState</a><br/> | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean }<br/> | ||||
|   data: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L34">ImportedDataState</a>,<br/>  | ||||
|   localAppState: Partial<<a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a>> | null | undefined,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a> | null | undefined<br/>): <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L4">DataState</a><br/> | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean, normalizeIndices?: boolean }<br/> | ||||
|  | ||||
| ) | ||||
| </pre> | ||||
|  | ||||
| See [`restoreAppState()`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#restoreAppState) about `localAppState`, and [`restoreElements()`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#restoreElements) about `localElements`. | ||||
| See [`restoreAppState()`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/packages/excalidraw/README.md#restoreAppState) about `localAppState`, and [`restoreElements()`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/packages/excalidraw/README.md#restoreElements) about `localElements`. | ||||
|  | ||||
| **_How to use_** | ||||
|  | ||||
| @@ -93,7 +94,7 @@ This function makes sure elements and state is set to appropriate values and set | ||||
| **_Signature_** | ||||
|  | ||||
| <pre> | ||||
| restoreLibraryItems(libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L34">ImportedDataState["libraryItems"]</a>,<br/>  | ||||
| restoreLibraryItems(libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L34">ImportedDataState["libraryItems"]</a>,<br/>  | ||||
| defaultStatus: "published" | "unpublished") | ||||
| </pre> | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ These are pure Javascript functions exported from the @excalidraw/excalidraw [`@ | ||||
|  | ||||
| ### serializeAsJSON | ||||
|  | ||||
| Takes the scene elements and state and returns a JSON string. `Deleted` elements as well as most properties from `AppState` are removed from the resulting JSON. (see [`serializeAsJSON()`](https://github.com/excalidraw/excalidraw/blob/master/src/data/json.ts#L42) source for details). | ||||
| Takes the scene elements and state and returns a JSON string. `Deleted` elements as well as most properties from `AppState` are removed from the resulting JSON. (see [`serializeAsJSON()`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/json.ts#L42) source for details). | ||||
|  | ||||
| If you want to overwrite the `source` field in the `JSON` string, you can set `window.EXCALIDRAW_EXPORT_SOURCE` to the desired value. | ||||
|  | ||||
| @@ -16,8 +16,8 @@ If you want to overwrite the `source` field in the `JSON` string, you can set `w | ||||
|  | ||||
| <pre> | ||||
| serializeAsJSON({<br/>  | ||||
|   elements: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a>,<br/>  | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a>,<br/> | ||||
|   elements: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a>,<br/>  | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a>,<br/> | ||||
| }): string | ||||
| </pre> | ||||
|  | ||||
| @@ -37,7 +37,7 @@ If you want to overwrite the source field in the JSON string, you can set `windo | ||||
|  | ||||
| <pre> | ||||
| serializeLibraryAsJSON( | ||||
|   libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200">LibraryItems[]</a>) | ||||
|   libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L200">LibraryItems[]</a>) | ||||
| </pre> | ||||
|  | ||||
| **How to use** | ||||
| @@ -53,7 +53,7 @@ Returns `true` if element is invisibly small (e.g. width & height are zero). | ||||
| **_Signature_** | ||||
|  | ||||
| <pre> | ||||
| isInvisiblySmallElement(element:  <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement</a>): boolean | ||||
| isInvisiblySmallElement(element:  <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement</a>): boolean | ||||
| </pre> | ||||
|  | ||||
| **How to use** | ||||
| @@ -80,10 +80,10 @@ excalidrawAPI.updateScene(scene); | ||||
| <pre> | ||||
| loadFromBlob(<br/>  | ||||
|   blob: <a href="https://developer.mozilla.org/en-US/docs/Web/API/Blob">Blob</a>,<br/>  | ||||
|   localAppState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a> | null,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a> | null,<br/>  | ||||
|   localAppState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a> | null,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a> | null,<br/>  | ||||
|   fileHandle?: FileSystemHandle | null <br/> | ||||
| ) => Promise<<a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/restore.ts#L61">RestoredDataState</a>> | ||||
| ) => Promise<<a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/restore.ts#L61">RestoredDataState</a>> | ||||
| </pre> | ||||
|  | ||||
| ### loadLibraryFromBlob | ||||
| @@ -130,10 +130,10 @@ if (contents.type === MIME_TYPES.excalidraw) { | ||||
| <pre> | ||||
| loadSceneOrLibraryFromBlob(<br/>  | ||||
|   blob: <a href="https://developer.mozilla.org/en-US/docs/Web/API/Blob">Blob</a>,<br/>  | ||||
|   localAppState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a> | null,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a> | null,<br/>  | ||||
|   localAppState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a> | null,<br/>  | ||||
|   localElements: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a> | null,<br/>  | ||||
|   fileHandle?: FileSystemHandle | null<br/> | ||||
| ) => Promise<{ type: string, data: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/restore.ts#L53">RestoredDataState</a> | <a href="https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L33">ImportedLibraryState</a>}> | ||||
| ) => Promise<{ type: string, data: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/restore.ts#L53">RestoredDataState</a> | <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/types.ts#L33">ImportedLibraryState</a>}> | ||||
| </pre> | ||||
|  | ||||
| ### getFreeDrawSvgPath | ||||
| @@ -149,7 +149,7 @@ import { getFreeDrawSvgPath } from "@excalidraw/excalidraw"; | ||||
| **Signature** | ||||
|  | ||||
| <pre> | ||||
| getFreeDrawSvgPath(element: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L182">ExcalidrawFreeDrawElement</a>) | ||||
| getFreeDrawSvgPath(element: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L182">ExcalidrawFreeDrawElement</a>) | ||||
| </pre> | ||||
|  | ||||
| ### isLinearElement | ||||
| @@ -165,7 +165,7 @@ import { isLinearElement } from "@excalidraw/excalidraw"; | ||||
| **Signature** | ||||
|  | ||||
| <pre> | ||||
| isLinearElement(elementType?: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L80">ExcalidrawElement</a>): boolean | ||||
| isLinearElement(elementType?: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L80">ExcalidrawElement</a>): boolean | ||||
| </pre> | ||||
|  | ||||
| ### getNonDeletedElements | ||||
| @@ -181,7 +181,7 @@ import { getNonDeletedElements } from "@excalidraw/excalidraw"; | ||||
| **Signature** | ||||
|  | ||||
| <pre> | ||||
| getNonDeletedElements(elements:<a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114"> readonly ExcalidrawElement[]</a>): as readonly <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L125">NonDeletedExcalidrawElement[]</a> | ||||
| getNonDeletedElements(elements:<a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114"> readonly ExcalidrawElement[]</a>): as readonly <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L125">NonDeletedExcalidrawElement[]</a> | ||||
| </pre> | ||||
|  | ||||
| ### mergeLibraryItems | ||||
| @@ -196,9 +196,9 @@ import { mergeLibraryItems } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| <pre> | ||||
| mergeLibraryItems(<br/>  | ||||
|   localItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L250">LibraryItems</a>,<br/>  | ||||
|   otherItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200">LibraryItems</a><br/> | ||||
| ): <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L250">LibraryItems</a> | ||||
|   localItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L250">LibraryItems</a>,<br/>  | ||||
|   otherItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L200">LibraryItems</a><br/> | ||||
| ): <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L250">LibraryItems</a> | ||||
| </pre> | ||||
|  | ||||
| ### parseLibraryTokensFromUrl | ||||
| @@ -239,8 +239,8 @@ export const App = () => { | ||||
|  | ||||
| <pre> | ||||
| useHandleLibrary(opts: {<br/>  | ||||
|   excalidrawAPI: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L494">ExcalidrawAPI</a>,<br/>  | ||||
|   getInitialLibraryItems?: () => <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L253">LibraryItemsSource</a><br/> | ||||
|   excalidrawAPI: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L494">ExcalidrawAPI</a>,<br/>  | ||||
|   getInitialLibraryItems?: () => <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L253">LibraryItemsSource</a><br/> | ||||
| }); | ||||
| </pre> | ||||
|  | ||||
| @@ -253,7 +253,7 @@ This function returns the current `scene` version. | ||||
| **_Signature_** | ||||
|  | ||||
| <pre> | ||||
| getSceneVersion(elements:  <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114">ExcalidrawElement[]</a>) | ||||
| getSceneVersion(elements:  <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a>) | ||||
| </pre> | ||||
|  | ||||
| **How to use** | ||||
| @@ -274,7 +274,7 @@ import { sceneCoordsToViewportCoords } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| <pre> | ||||
| sceneCoordsToViewportCoords({ sceneX: number, sceneY: number },<br/>  | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a><br/>): { x: number, y: number } | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a><br/>): { x: number, y: number } | ||||
| </pre> | ||||
|  | ||||
| ### viewportCoordsToSceneCoords | ||||
| @@ -289,7 +289,7 @@ import { viewportCoordsToSceneCoords } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| <pre> | ||||
| viewportCoordsToSceneCoords({ clientX: number, clientY: number },<br/>  | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95">AppState</a><br/>): {x: number, y: number} | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95">AppState</a><br/>): {x: number, y: number} | ||||
| </pre> | ||||
|  | ||||
| ### useDevice | ||||
| @@ -350,8 +350,8 @@ To help with localization, we export the following. | ||||
| | name | type | | ||||
| | --- | --- | | ||||
| | `defaultLang` | `string` | | ||||
| | `languages` | [`Language[]`](https://github.com/excalidraw/excalidraw/blob/master/src/i18n.ts#L15) | | ||||
| | `useI18n` | [`() => { langCode, t }`](https://github.com/excalidraw/excalidraw/blob/master/src/i18n.ts#L15) | | ||||
| | `languages` | [`Language[]`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/i18n.ts#L15) | | ||||
| | `useI18n` | [`() => { langCode, t }`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/i18n.ts#L15) | | ||||
|  | ||||
| ```js | ||||
| import { defaultLang, languages, useI18n } from "@excalidraw/excalidraw"; | ||||
|   | ||||
| @@ -21,7 +21,7 @@ Most notably, you can customize the primary colors, by overriding these variable | ||||
| - `--color-primary-light` | ||||
| - `--color-primary-contrast-offset` — a slightly darker (in light mode), or lighter (in dark mode) `--color-primary` color to fix contrast issues (see [Chubb illusion](https://en.wikipedia.org/wiki/Chubb_illusion)). It will fall back to `--color-primary` if not present. | ||||
|  | ||||
| For a complete list of variables, check [theme.scss](https://github.com/excalidraw/excalidraw/blob/master/src/css/theme.scss), though most of them will not make sense to override. | ||||
| For a complete list of variables, check [theme.scss](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/css/theme.scss), though most of them will not make sense to override. | ||||
|  | ||||
| ```css showLineNumbers | ||||
| .custom-styles .excalidraw { | ||||
|   | ||||
| @@ -13,18 +13,18 @@ To start the example app using the `@excalidraw/excalidraw` package, follow the | ||||
| 1. Install the dependencies | ||||
|  | ||||
|    ```bash | ||||
|    cd src/packages/excalidraw && yarn | ||||
|    yarn | ||||
|    ``` | ||||
|  | ||||
| 2. Start the example app | ||||
|  | ||||
|    ```bash | ||||
|    yarn start | ||||
|    yarn start:example | ||||
|    ``` | ||||
|  | ||||
|    [http://localhost:3001](http://localhost:3001) will open in your default browser. | ||||
|  | ||||
|    The example is same as the [codesandbox example](https://ehlz3.csb.app/) | ||||
|     | ||||
|    This is the same example as the [CodeSandbox](https://codesandbox.io/p/sandbox/github/excalidraw/excalidraw/tree/master/examples/with-script-in-browser) example. | ||||
|  | ||||
| ## Releasing | ||||
|  | ||||
|   | ||||
| @@ -17,11 +17,9 @@ We strongly recommend turning it off. You can follow the steps below on how to d | ||||
| <div style={{width:'30rem'}}> | ||||
|  | ||||
| 2. Once opened, look for **Aggressively Block Fingerprinting** | ||||
|  | ||||
|  | ||||
|  | ||||
| 3. Switch to **Block Fingerprinting** | ||||
|  | ||||
|  | ||||
|  | ||||
| 4. Thats all. All text elements should be fixed now 🎉 | ||||
|   | ||||
| @@ -1,16 +1,12 @@ | ||||
| # Installation | ||||
|  | ||||
| **Excalidraw** is published to npm as a component you can directly embed in your projects. | ||||
| **Excalidraw** is exported as a component to be directly embedded in your project. | ||||
|  | ||||
| Using `npm`: | ||||
| Use `npm` or `yarn` to install the package. | ||||
|  | ||||
| ```bash | ||||
| npm install react react-dom @excalidraw/excalidraw | ||||
| ``` | ||||
|  | ||||
| or `yarn`: | ||||
|  | ||||
| ```bash | ||||
| # or | ||||
| yarn add react react-dom @excalidraw/excalidraw | ||||
| ``` | ||||
|  | ||||
| @@ -20,24 +16,40 @@ yarn add react react-dom @excalidraw/excalidraw | ||||
|  | ||||
| ::: | ||||
|  | ||||
| ### Static assets | ||||
| ### Self-hosting fonts | ||||
|  | ||||
| Excalidraw depends on assets such as localization files (if you opt to use them), fonts, and others. | ||||
| By default, Excalidraw will try to download all the used fonts from the [CDN](https://esm.run/@excalidraw/excalidraw/dist/prod). | ||||
|  | ||||
| By default these assets are loaded from a public CDN [`https://unpkg.com/@excalidraw/excalidraw/dist/`](https://unpkg.com/@excalidraw/excalidraw/dist), so you don't need to do anything on your end. | ||||
|  | ||||
| However, if you want to host these files yourself, you can find them in your `node_modules/@excalidraw/excalidraw/dist` directory, in folders `excalidraw-assets` (for production) and `excalidraw-assets-dev` (for development). | ||||
|  | ||||
| Copy these folders to your static assets directory, and add a `window.EXCALIDRAW_ASSET_PATH` variable in your `index.html` or `index.js` entry file pointing to your public assets path (relative). For example, if you serve your assets from the root of your hostname, you would do: | ||||
| For self-hosting purposes, you'll have to copy the content of the folder `node_modules/@excalidraw/excalidraw/dist/prod/fonts` to the path where your assets should be served from (i.e. `public/` directory in your project). In that case, you should also set `window.EXCALIDRAW_ASSET_PATH` to the very same path, i.e. `/` in case it's in the root: | ||||
|  | ||||
| ```js | ||||
| window.EXCALIDRAW_ASSET_PATH = "/"; | ||||
| ``` | ||||
|  | ||||
| or, if you serve your assets from the root of your CDN, you would do: | ||||
|  | ||||
| ```js | ||||
| // Vanilla | ||||
| <head> | ||||
|     <script> | ||||
|         window.EXCALIDRAW_ASSET_PATH = "https://my.cdn.com/assets/"; | ||||
|     </script>     | ||||
| </head> | ||||
| ``` | ||||
|  | ||||
| or, if you prefer the path to be dynamicly set based on the `location.origin`, you could do the following: | ||||
|  | ||||
| ```jsx | ||||
| // Next.js | ||||
| <Script id="load-env-variables" strategy="beforeInteractive" > | ||||
|     { `window["EXCALIDRAW_ASSET_PATH"] = location.origin;` } // or use just "/"! | ||||
| </Script> | ||||
| ``` | ||||
|  | ||||
| ### Dimensions of Excalidraw | ||||
|  | ||||
| Excalidraw takes _100%_ of `width` and `height` of the containing block so make sure the container in which you render Excalidraw has non zero dimensions. | ||||
|  | ||||
| ### Demo | ||||
| ## Demo | ||||
|  | ||||
| [Try here](https://codesandbox.io/s/excalidraw-ehlz3). | ||||
| Go to [CodeSandbox](https://codesandbox.io/p/sandbox/github/excalidraw/excalidraw/tree/master/examples/with-script-in-browser) example. | ||||
|   | ||||
| @@ -12,7 +12,7 @@ import { Excalidraw } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| Throughout the documentation we use live, editable Excalidraw examples like the one shown below. | ||||
|  | ||||
| While we aim for the examples to closely reflect what you'd get if you rendered it yourself, we actually initialize it with some props behind the scenes.   | ||||
| While we aim for the examples to closely reflect what you'd get if you rendered it yourself, we actually initialize it with some props behind the scenes. | ||||
| For example, we're passing a `theme` prop to it based on the current color theme of the docs you're just reading. | ||||
|  | ||||
| ::: | ||||
| @@ -32,15 +32,9 @@ function App() { | ||||
|  | ||||
| ### Next.js | ||||
|  | ||||
| Since _Excalidraw_ doesn't support server side rendering, you should render the component once the host is `mounted`. | ||||
| Since Excalidraw doesn't support `server side rendering` so it should be rendered only on `client`. The way to achieve this in next.js is using `next.js dynamic import`. | ||||
|  | ||||
| Here are two ways on how you can render **Excalidraw** on **Next.js**. | ||||
|  | ||||
|  | ||||
|  | ||||
| 1. Using **Next.js Dynamic** import [Recommended]. | ||||
|  | ||||
| Since Excalidraw doesn't support server side rendering so you can also use `dynamic import` to render by setting `ssr` to `false`. | ||||
| If you want to only import `Excalidraw` component you can do :point_down: | ||||
|  | ||||
| ```jsx showLineNumbers | ||||
| import dynamic from "next/dynamic"; | ||||
| @@ -55,27 +49,89 @@ export default function App() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Here is a working [demo](https://codesandbox.io/p/sandbox/excalidraw-with-next-dynamic-k8yjq2). | ||||
| However the above component only works for named component exports. If you want to import some util / constant or something else apart from Excalidraw, then this approach will not work. Instead you can write a wrapper over Excalidraw and import the wrapper dynamically. | ||||
|  | ||||
| If you are using `pages router` then importing the wrapper dynamically would work, where as if you are using `app router` then you will have to also add `useClient` directive on top of the file in addition to dynamically importing the wrapper as shown :point_down: | ||||
|  | ||||
| 2. Importing Excalidraw once **client** is rendered. | ||||
| <Tabs> | ||||
|   <TabItem value="Excalidraw Wrapper" label="Excalidraw Wrapper" > | ||||
|  | ||||
| ```jsx showLineNumbers | ||||
| import { useState, useEffect } from "react"; | ||||
| export default function App() { | ||||
|   const [Excalidraw, setExcalidraw] = useState(null); | ||||
|   useEffect(() => { | ||||
|     import("@excalidraw/excalidraw").then((comp) => | ||||
|       setExcalidraw(comp.Excalidraw), | ||||
|   ```jsx showLineNumbers | ||||
|   "use client"; | ||||
|   import { Excalidraw, convertToExcalidrawElements } from "@excalidraw/excalidraw"; | ||||
|  | ||||
|   import "@excalidraw/excalidraw/index.css"; | ||||
|  | ||||
|   const ExcalidrawWrapper: React.FC = () => { | ||||
|     console.info(convertToExcalidrawElements([{ | ||||
|       type: "rectangle", | ||||
|       id: "rect-1", | ||||
|       width: 186.47265625, | ||||
|       height: 141.9765625, | ||||
|     },])); | ||||
|     return ( | ||||
|       <div style={{height:"500px", width:"500px"}}> | ||||
|         <Excalidraw /> | ||||
|       </div> | ||||
|     ); | ||||
|   }, []); | ||||
|   return <>{Excalidraw && <Excalidraw />}</>; | ||||
| } | ||||
| ``` | ||||
|   }; | ||||
|   export default ExcalidrawWrapper; | ||||
|   ``` | ||||
|  | ||||
| Here is a working [demo](https://codesandbox.io/p/sandbox/excalidraw-with-next-5xb3d) | ||||
|   </TabItem> | ||||
|  | ||||
| The `types` are available at `@excalidraw/excalidraw/types`, you can view [example for typescript](https://codesandbox.io/s/excalidraw-types-9h2dm) | ||||
|   <TabItem value="pages" label="Pages router"> | ||||
|  | ||||
|   ```jsx showLineNumbers | ||||
|   import dynamic from "next/dynamic"; | ||||
|  | ||||
|   // Since client components get prerenderd on server as well hence importing | ||||
|   // the excalidraw stuff dynamically with ssr false | ||||
|  | ||||
|   const ExcalidrawWrapper = dynamic( | ||||
|     async () => (await import("../excalidrawWrapper")).default, | ||||
|     { | ||||
|       ssr: false, | ||||
|     }, | ||||
|   ); | ||||
|  | ||||
|   export default function Page() { | ||||
|     return ( | ||||
|       <ExcalidrawWrapper /> | ||||
|     ); | ||||
|   } | ||||
|   ``` | ||||
|   </TabItem> | ||||
|  | ||||
|   <TabItem value="app" label="App router"> | ||||
|  | ||||
|   ```jsx showLineNumbers | ||||
|   import dynamic from "next/dynamic"; | ||||
|  | ||||
|   // Since client components get prerenderd on server as well hence importing | ||||
|   // the excalidraw stuff dynamically with ssr false | ||||
|  | ||||
|   const ExcalidrawWrapper = dynamic( | ||||
|     async () => (await import("../excalidrawWrapper")).default, | ||||
|     { | ||||
|       ssr: false, | ||||
|     }, | ||||
|   ); | ||||
|  | ||||
|   export default function Page() { | ||||
|     return ( | ||||
|       <ExcalidrawWrapper /> | ||||
|     ); | ||||
|   } | ||||
|   ``` | ||||
|  | ||||
|   </TabItem> | ||||
| </Tabs> | ||||
|  | ||||
| {/* Link  should be updated to point to the latest! */} | ||||
| Here is a [source code](https://github.com/excalidraw/excalidraw/tree/master/examples/with-nextjs) for the example with app and pages router. You you can try it out [here](https://excalidraw-package-example-with-nextjs.vercel.app/). | ||||
|  | ||||
| The `types` are available at `@excalidraw/excalidraw/types`, check [CodeSandbox](https://codesandbox.io/p/sandbox/github/excalidraw/excalidraw/tree/master/examples/with-script-in-browser) example for details. | ||||
|  | ||||
| ### Preact | ||||
|  | ||||
| @@ -96,31 +152,13 @@ Since Vite removes env variables by default, you can update the vite config to e | ||||
|     "process.env.IS_PREACT": JSON.stringify("true"), | ||||
|   }, | ||||
| ``` | ||||
| :::  | ||||
| ::: | ||||
|  | ||||
| ## Browser | ||||
|  | ||||
| To use it in a browser directly: | ||||
| To use it Excalidraw in a browser directly, use the following setup :point_down: | ||||
|  | ||||
| For development use :point_down: | ||||
|  | ||||
| ```js | ||||
| <script | ||||
|   type="text/javascript" | ||||
|   src="https://unpkg.com/@excalidraw/excalidraw/dist/excalidraw.development.js" | ||||
| ></script> | ||||
| ``` | ||||
|  | ||||
| For production use :point_down: | ||||
|  | ||||
| ```js | ||||
| <script | ||||
|   type="text/javascript" | ||||
|   src="https://unpkg.com/@excalidraw/excalidraw/dist/excalidraw.production.min.js" | ||||
| ></script> | ||||
| ``` | ||||
|  | ||||
| You will need to make sure `react`, `react-dom` is available as shown in the below example. For prod please use the production versions of `react`, `react-dom`. | ||||
| > **Note**: We rely on import maps to de-duplicate `react`, `react-dom` and `react/jsx-runtime` versions. | ||||
|  | ||||
| import Tabs from "@theme/Tabs"; | ||||
| import TabItem from "@theme/TabItem"; | ||||
| @@ -134,13 +172,23 @@ import TabItem from "@theme/TabItem"; | ||||
|   <head> | ||||
|     <title>Excalidraw in browser</title> | ||||
|     <meta charset="UTF-8" /> | ||||
|     <script src="https://unpkg.com/react@18.2.0/umd/react.development.js"></script> | ||||
|     <script src="https://unpkg.com/react-dom@18.2.0/umd/react-dom.development.js"></script> | ||||
|  | ||||
|     <script | ||||
|       type="text/javascript" | ||||
|       src="https://unpkg.com/@excalidraw/excalidraw/dist/excalidraw.development.js" | ||||
|     ></script> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="https://esm.sh/@excalidraw/excalidraw@0.18.0/dist/dev/index.css" | ||||
|     /> | ||||
|     <link rel="stylesheet" href="./index.css" /> | ||||
|     <script> | ||||
|     	window.EXCALIDRAW_ASSET_PATH = "https://esm.sh/@excalidraw/excalidraw@0.18.0/dist/prod/"; | ||||
| 		</script> | ||||
|     <script type="importmap"> | ||||
|       { | ||||
|         "imports": { | ||||
|           "react": "https://esm.sh/react@19.0.0", | ||||
|           "react/jsx-runtime": "https://esm.sh/react@19.0.0/jsx-runtime", | ||||
|           "react-dom": "https://esm.sh/react-dom@19.0.0" | ||||
|           } | ||||
|       } | ||||
|     </script> | ||||
|   </head> | ||||
|  | ||||
|   <body> | ||||
| @@ -148,7 +196,7 @@ import TabItem from "@theme/TabItem"; | ||||
|       <h1>Excalidraw Embed Example</h1> | ||||
|       <div id="app"></div> | ||||
|     </div> | ||||
|     <script type="text/javascript" src="src/index.js"></script> | ||||
|     <script type="text/javascript" src="packages/excalidraw/index.js"></script> | ||||
|   </body> | ||||
| </html> | ||||
| ``` | ||||
| @@ -157,6 +205,14 @@ import TabItem from "@theme/TabItem"; | ||||
| <TabItem value="js" label="Javascript"> | ||||
|  | ||||
| ```js showLineNumbers | ||||
| // See https://www.npmjs.com/package/@excalidraw/excalidraw documentation. | ||||
| import * as ExcalidrawLib from 'https://esm.sh/@excalidraw/excalidraw@0.18.0/dist/dev/index.js?external=react,react-dom'; | ||||
| import React from "https://esm.sh/react@19.0.0"; | ||||
| import ReactDOM from "https://esm.sh/react-dom@19.0.0" | ||||
|  | ||||
| window.ExcalidrawLib = ExcalidrawLib; | ||||
| console.log("Excalidraw library", ExcalidrawLib); | ||||
|  | ||||
| const App = () => { | ||||
|   return React.createElement( | ||||
|     React.Fragment, | ||||
| @@ -178,3 +234,5 @@ root.render(React.createElement(App)); | ||||
|  | ||||
| </TabItem> | ||||
| </Tabs> | ||||
|  | ||||
| You can try it out [here](https://jsfiddle.net/vfn6dm14/3/). | ||||
|   | ||||
| @@ -14,7 +14,7 @@ This API receives the mermaid syntax as the input, and resolves to skeleton Exca | ||||
| import { parseMermaidToExcalidraw } from "@excalidraw/mermaid-to-excalidraw"; | ||||
| import { convertToExcalidrawElements}  from "@excalidraw/excalidraw" | ||||
| try { | ||||
|   const { elements, files } = await parseMermaid(mermaidSyntax: string, { | ||||
|   const { elements, files } = await parseMermaidToExcalidraw(mermaidSyntax: string, { | ||||
|     fontSize: number, | ||||
|   }); | ||||
|   const excalidrawElements = convertToExcalidrawElements(elements); | ||||
|   | ||||
| @@ -38,9 +38,9 @@ Add the diagram type in switch case in [`parseMermaid`](https://github.com/excal | ||||
|  | ||||
| ## Writing the Excalidraw Skeleton Convertor | ||||
|  | ||||
| With the completion of previous step, we have all the data, now we need to transform it so to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133) format. | ||||
| With the completion of previous step, we have all the data, now we need to transform it so to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133) format. | ||||
|  | ||||
| Similar to [`FlowChartToExcalidrawSkeletonConverter`](https://github.com/excalidraw/mermaid-to-excalidraw/blob/master/src/converter/types/flowchart.ts#L24), you have to write the `{{diagramType}}ToExcalidrawSkeletonConverter` which parses the data received in previous step and returns the [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133). | ||||
| Similar to [`FlowChartToExcalidrawSkeletonConverter`](https://github.com/excalidraw/mermaid-to-excalidraw/blob/master/src/converter/types/flowchart.ts#L24), you have to write the `{{diagramType}}ToExcalidrawSkeletonConverter` which parses the data received in previous step and returns the [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133). | ||||
|  | ||||
| Thats it, you have added the new diagram type 🥳, now lets test it out! | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,7 @@ In this section we will be diving into how the [flowchart parser](https://github | ||||
|  | ||||
|  | ||||
|  | ||||
| We use `diagram.parser.yy` attribute to parse the data. If you want to know more about how the `diagram.parse.yy` attribute looks like, you can check it [here](https://github.com/mermaid-js/mermaid/blob/00d06c7282a701849793680c1e97da1cfdfcce62/packages/mermaid/src/diagrams/flowchart/flowDb.js#L768), however for scope of flowchart we are using **3** APIs from this parser to compute `vertices`, `edges` and `clusters` as we need these data to transform to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133C13-L133C38). | ||||
| We use `diagram.parser.yy` attribute to parse the data. If you want to know more about how the `diagram.parse.yy` attribute looks like, you can check it [here](https://github.com/mermaid-js/mermaid/blob/00d06c7282a701849793680c1e97da1cfdfcce62/packages/mermaid/src/diagrams/flowchart/flowDb.js#L768), however for scope of flowchart we are using **3** APIs from this parser to compute `vertices`, `edges` and `clusters` as we need these data to transform to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133C13-L133C38). | ||||
|  | ||||
|  | ||||
| For computing `vertices` and `edge`s lets consider the below svg generated by mermaid | ||||
| @@ -42,7 +42,7 @@ Considering the same example this is the response from the API | ||||
| 	} | ||||
| } | ||||
| ``` | ||||
| The dimensions and position is missing in this response and we need that to transform to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133C13-L133C38), for this we have our own parser [`parseVertex`](https://github.com/excalidraw/mermaid-to-excalidraw/blob/master/src/parseMermaid.ts#L178) which takes the above response and uses the `svg` together to compute position, dimensions and cleans up the response. | ||||
| The dimensions and position is missing in this response and we need that to transform to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133C13-L133C38), for this we have our own parser [`parseVertex`](https://github.com/excalidraw/mermaid-to-excalidraw/blob/master/src/parseMermaid.ts#L178) which takes the above response and uses the `svg` together to compute position, dimensions and cleans up the response. | ||||
|  | ||||
|  The final output from `parseVertex` looks like :point_down: | ||||
|  | ||||
|   | ||||
| @@ -55,11 +55,11 @@ If you want to understand how flowchart parser works, you can navigate to [Flowc | ||||
|  | ||||
| ## Converting to ExcalidrawElementSkeleton | ||||
|  | ||||
| Now we have all the data, we just need to transform it to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133C13-L133C38) API so it can be rendered in Excalidraw. | ||||
| Now we have all the data, we just need to transform it to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133C13-L133C38) API so it can be rendered in Excalidraw. | ||||
|  | ||||
| For this we have `converters` which takes the parsed mermaid data and gives back the Excalidraw Skeleton. | ||||
| For Unsupported types, we have already mentioned above that we convert it to `dataURL` and return the ExcalidrawImageSkeleton. | ||||
|  | ||||
| For supported types, currently only flowchart, we have [flowchartConverter](https://github.com/excalidraw/mermaid-to-excalidraw/blob/master/src/converter/types/flowchart.ts#L24) which parses the data and converts to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/src/data/transform.ts#L133C13-L133C38). | ||||
| For supported types, currently only flowchart, we have [flowchartConverter](https://github.com/excalidraw/mermaid-to-excalidraw/blob/master/src/converter/types/flowchart.ts#L24) which parses the data and converts to [ExcalidrawElementSkeleton](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/transform.ts#L133C13-L133C38). | ||||
|  | ||||
|  | ||||
| @@ -43,7 +43,7 @@ When saving an Excalidraw scene locally to a file, the JSON file (`.excalidraw`) | ||||
|  | ||||
|   // editor state (canvas config, preferences, ...) | ||||
|   "appState": { | ||||
|     "gridSize": null, | ||||
|     "gridSize": 20, | ||||
|     "viewBackgroundColor": "#ffffff" | ||||
|   }, | ||||
|  | ||||
|   | ||||
| @@ -52,15 +52,6 @@ Make sure the title starts with a semantic prefix: | ||||
| - **chore**: Other changes that don't modify src or test files | ||||
| - **revert**: Reverts a previous commit | ||||
|  | ||||
| ### Changelog | ||||
|  | ||||
| Add a brief description of your pull request to the changelog located here: [changelog](https://github.com/excalidraw/excalidraw/blob/master/CHANGELOG.md) | ||||
|  | ||||
| Notes: | ||||
|  | ||||
| - Make sure to prepend to the section corresponding with the semantic prefix you selected in the title | ||||
| - Link to your pull request - this will require updating the CHANGELOG _after_ creating the pull request | ||||
|  | ||||
| ### Testing | ||||
|  | ||||
| Once you submit your pull request it will automatically be tested. Be sure to check the results of the test and fix any issues that arise. | ||||
|   | ||||
| @@ -41,10 +41,7 @@ const config = { | ||||
|           showLastUpdateTime: true, | ||||
|         }, | ||||
|         theme: { | ||||
|           customCss: [ | ||||
|             require.resolve("./src/css/custom.scss"), | ||||
|             require.resolve("../src/packages/excalidraw/example/App.scss"), | ||||
|           ], | ||||
|           customCss: [require.resolve("./src/css/custom.scss")], | ||||
|         }, | ||||
|       }), | ||||
|     ], | ||||
| @@ -69,7 +66,7 @@ const config = { | ||||
|             label: "Docs", | ||||
|           }, | ||||
|           { | ||||
|             to: "https://blog.excalidraw.com", | ||||
|             to: "https://plus.excalidraw.com/blog", | ||||
|             label: "Blog", | ||||
|             position: "left", | ||||
|           }, | ||||
| @@ -114,7 +111,7 @@ const config = { | ||||
|             items: [ | ||||
|               { | ||||
|                 label: "Blog", | ||||
|                 to: "https://blog.excalidraw.com", | ||||
|                 to: "https://plus.excalidraw.com/blog", | ||||
|               }, | ||||
|               { | ||||
|                 label: "GitHub", | ||||
| @@ -152,6 +149,29 @@ const config = { | ||||
|         systemvars: true, | ||||
|       }, | ||||
|     ], | ||||
|     function () { | ||||
|       return { | ||||
|         name: "disable-fully-specified-error", | ||||
|         configureWebpack() { | ||||
|           return { | ||||
|             module: { | ||||
|               rules: [ | ||||
|                 { | ||||
|                   test: /\.m?js$/, | ||||
|                   resolve: { | ||||
|                     fullySpecified: false, | ||||
|                   }, | ||||
|                 }, | ||||
|               ], | ||||
|             }, | ||||
|             optimization: { | ||||
|               // disable terser minification | ||||
|               minimize: false, | ||||
|             }, | ||||
|           }; | ||||
|         }, | ||||
|       }; | ||||
|     }, | ||||
|   ], | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -18,13 +18,13 @@ | ||||
|     "@docusaurus/core": "2.2.0", | ||||
|     "@docusaurus/preset-classic": "2.2.0", | ||||
|     "@docusaurus/theme-live-codeblock": "2.2.0", | ||||
|     "@excalidraw/excalidraw": "0.17.0", | ||||
|     "@excalidraw/excalidraw": "0.18.0", | ||||
|     "@mdx-js/react": "^1.6.22", | ||||
|     "clsx": "^1.2.1", | ||||
|     "docusaurus-plugin-sass": "0.2.3", | ||||
|     "prism-react-renderer": "^1.3.5", | ||||
|     "react": "^17.0.2", | ||||
|     "react-dom": "^17.0.2", | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0", | ||||
|     "sass": "1.57.1" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
| import React from "react"; | ||||
|  | ||||
| import styles from "./styles.module.css"; | ||||
|  | ||||
| const FeatureList = [ | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
| import React from "react"; | ||||
|  | ||||
| import styles from "./styles.module.css"; | ||||
|  | ||||
| type FeatureItem = { | ||||
|   | ||||
| @@ -59,7 +59,7 @@ pre a { | ||||
|   padding: 5px; | ||||
|   background: #70b1ec; | ||||
|   color: white; | ||||
|   font-weight: bold; | ||||
|   font-weight: 700; | ||||
|   border: none; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,10 +1,11 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
| import Layout from "@theme/Layout"; | ||||
| import Link from "@docusaurus/Link"; | ||||
| import useDocusaurusContext from "@docusaurus/useDocusaurusContext"; | ||||
| import styles from "./index.module.css"; | ||||
| import HomepageFeatures from "@site/src/components/Homepage"; | ||||
| import Layout from "@theme/Layout"; | ||||
| import clsx from "clsx"; | ||||
| import React from "react"; | ||||
|  | ||||
| import styles from "./index.module.css"; | ||||
|  | ||||
| function HomepageHeader() { | ||||
|   const { siteConfig } = useDocusaurusContext(); | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| // Import the original mapper | ||||
| import MDXComponents from "@theme-original/MDXComponents"; | ||||
| import Highlight from "@site/src/components/Highlight"; | ||||
| import MDXComponents from "@theme-original/MDXComponents"; | ||||
|  | ||||
| export default { | ||||
|   // Re-use the default mapping | ||||
|   | ||||
| @@ -3,11 +3,18 @@ import ExecutionEnvironment from "@docusaurus/ExecutionEnvironment"; | ||||
| import initialData from "@site/src/initialData"; | ||||
| import { useColorMode } from "@docusaurus/theme-common"; | ||||
|  | ||||
| import "@excalidraw/excalidraw/index.css"; | ||||
|  | ||||
| let ExcalidrawComp = {}; | ||||
| if (ExecutionEnvironment.canUseDOM) { | ||||
|   ExcalidrawComp = require("@excalidraw/excalidraw"); | ||||
| } | ||||
| const Excalidraw = React.forwardRef((props, ref) => { | ||||
|   if (!window.EXCALIDRAW_ASSET_PATH) { | ||||
|     window.EXCALIDRAW_ASSET_PATH = | ||||
|       "https://esm.sh/@excalidraw/excalidraw@0.18.0/dist/prod/"; | ||||
|   } | ||||
|  | ||||
|   const { colorMode } = useColorMode(); | ||||
|   return <ExcalidrawComp.Excalidraw theme={colorMode} {...props} ref={ref} />; | ||||
| }); | ||||
|   | ||||
							
								
								
									
										4
									
								
								dev-docs/vercel.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| { | ||||
|   "outputDirectory": "build", | ||||
|   "installCommand": "yarn install" | ||||
| } | ||||
							
								
								
									
										1501
									
								
								dev-docs/yarn.lock
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										39
									
								
								examples/with-nextjs/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,39 @@ | ||||
| # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. | ||||
|  | ||||
| # dependencies | ||||
| /node_modules | ||||
| /.pnp | ||||
| .pnp.js | ||||
| .yarn/install-state.gz | ||||
|  | ||||
| # testing | ||||
| /coverage | ||||
|  | ||||
| # next.js | ||||
| /.next/ | ||||
| /out/ | ||||
|  | ||||
| # production | ||||
| /build | ||||
|  | ||||
| # misc | ||||
| .DS_Store | ||||
| *.pem | ||||
|  | ||||
| # debug | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
|  | ||||
| # local env files | ||||
| .env*.local | ||||
|  | ||||
| # vercel | ||||
| .vercel | ||||
|  | ||||
| # typescript | ||||
| *.tsbuildinfo | ||||
| next-env.d.ts | ||||
|  | ||||
| # copied assets | ||||
| public/**/*.woff2 | ||||
							
								
								
									
										36
									
								
								examples/with-nextjs/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,36 @@ | ||||
| This is a [Next.js](https://nextjs.org/) project bootstrapped with [`create-next-app`](https://github.com/vercel/next.js/tree/canary/packages/create-next-app). | ||||
|  | ||||
| ## Getting Started | ||||
|  | ||||
| First, run the development server: | ||||
|  | ||||
| ```bash | ||||
| npm run dev | ||||
| # or | ||||
| yarn dev | ||||
| # or | ||||
| pnpm dev | ||||
| # or | ||||
| bun dev | ||||
| ``` | ||||
|  | ||||
| Open [http://localhost:3000](http://localhost:3005) with your browser to see the result. | ||||
|  | ||||
| You can start editing the page by modifying `app/page.tsx`. The page auto-updates as you edit the file. | ||||
|  | ||||
| This project uses [`next/font`](https://nextjs.org/docs/basic-features/font-optimization) to automatically optimize and load Inter, a custom Google Font. | ||||
|  | ||||
| ## Learn More | ||||
|  | ||||
| To learn more about Next.js, take a look at the following resources: | ||||
|  | ||||
| - [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API. | ||||
| - [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial. | ||||
|  | ||||
| You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js/) - your feedback and contributions are welcome! | ||||
|  | ||||
| ## Deploy on Vercel | ||||
|  | ||||
| The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js. | ||||
|  | ||||
| Check out our [Next.js deployment documentation](https://nextjs.org/docs/deployment) for more details. | ||||
							
								
								
									
										12
									
								
								examples/with-nextjs/next.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| /** @type {import('next').NextConfig} */ | ||||
| const nextConfig = { | ||||
|   distDir: "build", | ||||
|   typescript: { | ||||
|     // The ts config doesn't work with `jsx: preserve" and if updated to `react-jsx` it gets ovewritten by next js throwing ts errors hence I am ignoring build errors until this is fixed. | ||||
|     ignoreBuildErrors: true, | ||||
|   }, | ||||
|   // This is needed as in pages router the code for importing types throws error as its outside next js app | ||||
|   transpilePackages: ["../"], | ||||
| }; | ||||
|  | ||||
| module.exports = nextConfig; | ||||
							
								
								
									
										25
									
								
								examples/with-nextjs/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | ||||
| { | ||||
|   "name": "with-nextjs", | ||||
|   "version": "0.1.0", | ||||
|   "private": true, | ||||
|   "scripts": { | ||||
|     "build:workspace": "yarn workspace @excalidraw/excalidraw run build:esm && yarn copy:assets", | ||||
|     "copy:assets": "cp -r ../../packages/excalidraw/dist/prod/fonts ./public", | ||||
|     "dev": "yarn build:workspace && next dev -p 3005", | ||||
|     "build": "yarn build:workspace && next build", | ||||
|     "start": "next start -p 3006", | ||||
|     "lint": "next lint" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "next": "14.1", | ||||
|     "react": "19.0.0", | ||||
|     "react-dom": "19.0.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/node": "^20", | ||||
|     "@types/react": "19.0.10", | ||||
|     "@types/react-dom": "19.0.4", | ||||
|     "path2d-polyfill": "2.0.1", | ||||
|     "typescript": "^5" | ||||
|   } | ||||
| } | ||||
| Before Width: | Height: | Size: 197 KiB After Width: | Height: | Size: 197 KiB | 
| Before Width: | Height: | Size: 30 KiB After Width: | Height: | Size: 30 KiB | 
| Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 6.1 KiB | 
| Before Width: | Height: | Size: 39 KiB After Width: | Height: | Size: 39 KiB | 
							
								
								
									
										
											BIN
										
									
								
								examples/with-nextjs/src/app/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 25 KiB | 
							
								
								
									
										11
									
								
								examples/with-nextjs/src/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| export default function RootLayout({ | ||||
|   children, | ||||
| }: { | ||||
|   children: React.ReactNode; | ||||
| }) { | ||||
|   return ( | ||||
|     <html lang="en"> | ||||
|       <body>{children}</body> | ||||
|     </html> | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										27
									
								
								examples/with-nextjs/src/app/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| import dynamic from "next/dynamic"; | ||||
| import Script from "next/script"; | ||||
|  | ||||
| import "../common.scss"; | ||||
|  | ||||
| // Since client components get prerenderd on server as well hence importing the excalidraw stuff dynamically | ||||
| // with ssr false | ||||
| const ExcalidrawWithClientOnly = dynamic( | ||||
|   async () => (await import("../excalidrawWrapper")).default, | ||||
|   { | ||||
|     ssr: false, | ||||
|   }, | ||||
| ); | ||||
|  | ||||
| export default function Page() { | ||||
|   return ( | ||||
|     <> | ||||
|       <a href="/excalidraw-in-pages">Switch to Pages router</a> | ||||
|       <h1 className="page-title">App Router</h1> | ||||
|       <Script id="load-env-variables" strategy="beforeInteractive"> | ||||
|         {`window["EXCALIDRAW_ASSET_PATH"] = window.origin;`} | ||||
|       </Script> | ||||
|       {/* @ts-expect-error - https://github.com/vercel/next.js/issues/42292 */} | ||||
|       <ExcalidrawWithClientOnly /> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										15
									
								
								examples/with-nextjs/src/common.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| * { | ||||
|   box-sizing: border-box; | ||||
|   font-family: sans-serif; | ||||
| } | ||||
|  | ||||
| a { | ||||
|   color: #1c7ed6; | ||||
|   font-size: 20px; | ||||
|   text-decoration: none; | ||||
|   font-weight: 500; | ||||
| } | ||||
|  | ||||
| .page-title { | ||||
|   text-align: center; | ||||
| } | ||||
							
								
								
									
										23
									
								
								examples/with-nextjs/src/excalidrawWrapper.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| "use client"; | ||||
| import * as excalidrawLib from "@excalidraw/excalidraw"; | ||||
| import { Excalidraw } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| import "@excalidraw/excalidraw/index.css"; | ||||
|  | ||||
| import App from "../../with-script-in-browser/components/ExampleApp"; | ||||
|  | ||||
| const ExcalidrawWrapper: React.FC = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <App | ||||
|         appTitle={"Excalidraw with Nextjs Example"} | ||||
|         useCustom={(api: any, args?: any[]) => {}} | ||||
|         excalidrawLib={excalidrawLib} | ||||
|       > | ||||
|         <Excalidraw /> | ||||
|       </App> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default ExcalidrawWrapper; | ||||
							
								
								
									
										23
									
								
								examples/with-nextjs/src/pages/excalidraw-in-pages.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| import dynamic from "next/dynamic"; | ||||
|  | ||||
| import "../common.scss"; | ||||
|  | ||||
| // Since client components get prerenderd on server as well hence importing the excalidraw stuff dynamically | ||||
| // with ssr false | ||||
| const Excalidraw = dynamic( | ||||
|   async () => (await import("../excalidrawWrapper")).default, | ||||
|   { | ||||
|     ssr: false, | ||||
|   }, | ||||
| ); | ||||
|  | ||||
| export default function Page() { | ||||
|   return ( | ||||
|     <> | ||||
|       <a href="/">Switch to App router</a> | ||||
|       <h1 className="page-title">Pages Router</h1> | ||||
|       {/* @ts-expect-error - https://github.com/vercel/next.js/issues/42292 */} | ||||
|       <Excalidraw /> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										28
									
								
								examples/with-nextjs/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,28 @@ | ||||
| { | ||||
|   "compilerOptions": { | ||||
|     "target": "es5", | ||||
|     "lib": ["dom", "dom.iterable", "esnext"], | ||||
|     "allowJs": true, | ||||
|     "skipLibCheck": true, | ||||
|     "strict": true, | ||||
|     "noEmit": true, | ||||
|     "esModuleInterop": true, | ||||
|     "module": "esnext", | ||||
|     "moduleResolution": "node", | ||||
|     "resolveJsonModule": true, | ||||
|     "isolatedModules": true, | ||||
|     "jsx": "preserve", | ||||
|     "incremental": true, | ||||
|     "plugins": [ | ||||
|       { | ||||
|         "name": "next" | ||||
|       } | ||||
|     ], | ||||
|     "paths": { | ||||
|       "@/*": ["./src/*"] | ||||
|     }, | ||||
|     "forceConsistentCasingInFileNames": true | ||||
|   }, | ||||
|   "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts", "build/types/**/*.ts"], | ||||
|   "exclude": ["node_modules"] | ||||
| } | ||||
							
								
								
									
										3
									
								
								examples/with-nextjs/vercel.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| { | ||||
|   "outputDirectory": "build" | ||||
| } | ||||
							
								
								
									
										252
									
								
								examples/with-nextjs/yarn.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,252 @@ | ||||
| # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. | ||||
| # yarn lockfile v1 | ||||
|  | ||||
|  | ||||
| "@excalidraw/excalidraw@workspace:^": | ||||
|   version "0.17.2" | ||||
|   resolved "https://registry.yarnpkg.com/@excalidraw/excalidraw/-/excalidraw-0.17.2.tgz#9a636a1e6bb3c88c5883347d3a7e75e9cce8ab96" | ||||
|   integrity sha512-7pqUWD8+mPjDhF4XxG3gw4rvE2JGaLW3Vss5UZfTbITPxAtFaGEc1K081bncitnaYhUwN9ENJE0i87QB3poDwQ== | ||||
|  | ||||
| "@next/env@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/env/-/env-14.0.4.tgz#d5cda0c4a862d70ae760e58c0cd96a8899a2e49a" | ||||
|   integrity sha512-irQnbMLbUNQpP1wcE5NstJtbuA/69kRfzBrpAD7Gsn8zm/CY6YQYc3HQBz8QPxwISG26tIm5afvvVbu508oBeQ== | ||||
|  | ||||
| "@next/swc-darwin-arm64@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-darwin-arm64/-/swc-darwin-arm64-14.0.4.tgz#27b1854c2cd04eb1d5e75081a1a792ad91526618" | ||||
|   integrity sha512-mF05E/5uPthWzyYDyptcwHptucf/jj09i2SXBPwNzbgBNc+XnwzrL0U6BmPjQeOL+FiB+iG1gwBeq7mlDjSRPg== | ||||
|  | ||||
| "@next/swc-darwin-x64@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-darwin-x64/-/swc-darwin-x64-14.0.4.tgz#9940c449e757d0ee50bb9e792d2600cc08a3eb3b" | ||||
|   integrity sha512-IZQ3C7Bx0k2rYtrZZxKKiusMTM9WWcK5ajyhOZkYYTCc8xytmwSzR1skU7qLgVT/EY9xtXDG0WhY6fyujnI3rw== | ||||
|  | ||||
| "@next/swc-linux-arm64-gnu@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-14.0.4.tgz#0eafd27c8587f68ace7b4fa80695711a8434de21" | ||||
|   integrity sha512-VwwZKrBQo/MGb1VOrxJ6LrKvbpo7UbROuyMRvQKTFKhNaXjUmKTu7wxVkIuCARAfiI8JpaWAnKR+D6tzpCcM4w== | ||||
|  | ||||
| "@next/swc-linux-arm64-musl@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-14.0.4.tgz#2b0072adb213f36dada5394ea67d6e82069ae7dd" | ||||
|   integrity sha512-8QftwPEW37XxXoAwsn+nXlodKWHfpMaSvt81W43Wh8dv0gkheD+30ezWMcFGHLI71KiWmHK5PSQbTQGUiidvLQ== | ||||
|  | ||||
| "@next/swc-linux-x64-gnu@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-14.0.4.tgz#68c67d20ebc8e3f6ced6ff23a4ba2a679dbcec32" | ||||
|   integrity sha512-/s/Pme3VKfZAfISlYVq2hzFS8AcAIOTnoKupc/j4WlvF6GQ0VouS2Q2KEgPuO1eMBwakWPB1aYFIA4VNVh667A== | ||||
|  | ||||
| "@next/swc-linux-x64-musl@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-14.0.4.tgz#67cd81b42fb2caf313f7992fcf6d978af55a1247" | ||||
|   integrity sha512-m8z/6Fyal4L9Bnlxde5g2Mfa1Z7dasMQyhEhskDATpqr+Y0mjOBZcXQ7G5U+vgL22cI4T7MfvgtrM2jdopqWaw== | ||||
|  | ||||
| "@next/swc-win32-arm64-msvc@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-14.0.4.tgz#be06585906b195d755ceda28f33c633e1443f1a3" | ||||
|   integrity sha512-7Wv4PRiWIAWbm5XrGz3D8HUkCVDMMz9igffZG4NB1p4u1KoItwx9qjATHz88kwCEal/HXmbShucaslXCQXUM5w== | ||||
|  | ||||
| "@next/swc-win32-ia32-msvc@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-win32-ia32-msvc/-/swc-win32-ia32-msvc-14.0.4.tgz#e76cabefa9f2d891599c3d85928475bd8d3f6600" | ||||
|   integrity sha512-zLeNEAPULsl0phfGb4kdzF/cAVIfaC7hY+kt0/d+y9mzcZHsMS3hAS829WbJ31DkSlVKQeHEjZHIdhN+Pg7Gyg== | ||||
|  | ||||
| "@next/swc-win32-x64-msvc@14.0.4": | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-14.0.4.tgz#e74892f1a9ccf41d3bf5979ad6d3d77c07b9cba1" | ||||
|   integrity sha512-yEh2+R8qDlDCjxVpzOTEpBLQTEFAcP2A8fUFLaWNap9GitYKkKv1//y2S6XY6zsR4rCOPRpU7plYDR+az2n30A== | ||||
|  | ||||
| "@swc/helpers@0.5.2": | ||||
|   version "0.5.2" | ||||
|   resolved "https://registry.yarnpkg.com/@swc/helpers/-/helpers-0.5.2.tgz#85ea0c76450b61ad7d10a37050289eded783c27d" | ||||
|   integrity sha512-E4KcWTpoLHqwPHLxidpOqQbcrZVgi0rsmmZXUle1jXmJfuIf/UWpczUJ7MZZ5tlxytgJXyp0w4PGkkeLiuIdZw== | ||||
|   dependencies: | ||||
|     tslib "^2.4.0" | ||||
|  | ||||
| "@types/node@^20": | ||||
|   version "20.11.0" | ||||
|   resolved "https://registry.yarnpkg.com/@types/node/-/node-20.11.0.tgz#8e0b99e70c0c1ade1a86c4a282f7b7ef87c9552f" | ||||
|   integrity sha512-o9bjXmDNcF7GbM4CNQpmi+TutCgap/K3w1JyKgxAjqx41zp9qlIAVFi0IhCNsJcXolEqLWhbFbEeL0PvYm4pcQ== | ||||
|   dependencies: | ||||
|     undici-types "~5.26.4" | ||||
|  | ||||
| "@types/prop-types@*": | ||||
|   version "15.7.11" | ||||
|   resolved "https://registry.yarnpkg.com/@types/prop-types/-/prop-types-15.7.11.tgz#2596fb352ee96a1379c657734d4b913a613ad563" | ||||
|   integrity sha512-ga8y9v9uyeiLdpKddhxYQkxNDrfvuPrlFb0N1qnZZByvcElJaXthF1UhvCh9TLWJBEHeNtdnbysW7Y6Uq8CVng== | ||||
|  | ||||
| "@types/react-dom@^18": | ||||
|   version "18.2.18" | ||||
|   resolved "https://registry.yarnpkg.com/@types/react-dom/-/react-dom-18.2.18.tgz#16946e6cd43971256d874bc3d0a72074bb8571dd" | ||||
|   integrity sha512-TJxDm6OfAX2KJWJdMEVTwWke5Sc/E/RlnPGvGfS0W7+6ocy2xhDVQVh/KvC2Uf7kACs+gDytdusDSdWfWkaNzw== | ||||
|   dependencies: | ||||
|     "@types/react" "*" | ||||
|  | ||||
| "@types/react@*", "@types/react@^18": | ||||
|   version "18.2.47" | ||||
|   resolved "https://registry.yarnpkg.com/@types/react/-/react-18.2.47.tgz#85074b27ab563df01fbc3f68dc64bf7050b0af40" | ||||
|   integrity sha512-xquNkkOirwyCgoClNk85BjP+aqnIS+ckAJ8i37gAbDs14jfW/J23f2GItAf33oiUPQnqNMALiFeoM9Y5mbjpVQ== | ||||
|   dependencies: | ||||
|     "@types/prop-types" "*" | ||||
|     "@types/scheduler" "*" | ||||
|     csstype "^3.0.2" | ||||
|  | ||||
| "@types/scheduler@*": | ||||
|   version "0.16.8" | ||||
|   resolved "https://registry.yarnpkg.com/@types/scheduler/-/scheduler-0.16.8.tgz#ce5ace04cfeabe7ef87c0091e50752e36707deff" | ||||
|   integrity sha512-WZLiwShhwLRmeV6zH+GkbOFT6Z6VklCItrDioxUnv+u4Ll+8vKeFySoFyK/0ctcRpOmwAicELfmys1sDc/Rw+A== | ||||
|  | ||||
| busboy@1.6.0: | ||||
|   version "1.6.0" | ||||
|   resolved "https://registry.yarnpkg.com/busboy/-/busboy-1.6.0.tgz#966ea36a9502e43cdb9146962523b92f531f6893" | ||||
|   integrity sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA== | ||||
|   dependencies: | ||||
|     streamsearch "^1.1.0" | ||||
|  | ||||
| caniuse-lite@^1.0.30001406: | ||||
|   version "1.0.30001576" | ||||
|   resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001576.tgz#893be772cf8ee6056d6c1e2d07df365b9ec0a5c4" | ||||
|   integrity sha512-ff5BdakGe2P3SQsMsiqmt1Lc8221NR1VzHj5jXN5vBny9A6fpze94HiVV/n7XRosOlsShJcvMv5mdnpjOGCEgg== | ||||
|  | ||||
| client-only@0.0.1: | ||||
|   version "0.0.1" | ||||
|   resolved "https://registry.yarnpkg.com/client-only/-/client-only-0.0.1.tgz#38bba5d403c41ab150bff64a95c85013cf73bca1" | ||||
|   integrity sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA== | ||||
|  | ||||
| csstype@^3.0.2: | ||||
|   version "3.1.3" | ||||
|   resolved "https://registry.yarnpkg.com/csstype/-/csstype-3.1.3.tgz#d80ff294d114fb0e6ac500fbf85b60137d7eff81" | ||||
|   integrity sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw== | ||||
|  | ||||
| glob-to-regexp@^0.4.1: | ||||
|   version "0.4.1" | ||||
|   resolved "https://registry.yarnpkg.com/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz#c75297087c851b9a578bd217dd59a92f59fe546e" | ||||
|   integrity sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw== | ||||
|  | ||||
| graceful-fs@^4.1.2, graceful-fs@^4.2.11: | ||||
|   version "4.2.11" | ||||
|   resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" | ||||
|   integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== | ||||
|  | ||||
| "js-tokens@^3.0.0 || ^4.0.0": | ||||
|   version "4.0.0" | ||||
|   resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499" | ||||
|   integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== | ||||
|  | ||||
| loose-envify@^1.1.0: | ||||
|   version "1.4.0" | ||||
|   resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.4.0.tgz#71ee51fa7be4caec1a63839f7e682d8132d30caf" | ||||
|   integrity sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q== | ||||
|   dependencies: | ||||
|     js-tokens "^3.0.0 || ^4.0.0" | ||||
|  | ||||
| nanoid@^3.3.6: | ||||
|   version "3.3.7" | ||||
|   resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.7.tgz#d0c301a691bc8d54efa0a2226ccf3fe2fd656bd8" | ||||
|   integrity sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g== | ||||
|  | ||||
| next@14.0.4: | ||||
|   version "14.0.4" | ||||
|   resolved "https://registry.yarnpkg.com/next/-/next-14.0.4.tgz#bf00b6f835b20d10a5057838fa2dfced1d0d84dc" | ||||
|   integrity sha512-qbwypnM7327SadwFtxXnQdGiKpkuhaRLE2uq62/nRul9cj9KhQ5LhHmlziTNqUidZotw/Q1I9OjirBROdUJNgA== | ||||
|   dependencies: | ||||
|     "@next/env" "14.0.4" | ||||
|     "@swc/helpers" "0.5.2" | ||||
|     busboy "1.6.0" | ||||
|     caniuse-lite "^1.0.30001406" | ||||
|     graceful-fs "^4.2.11" | ||||
|     postcss "8.4.31" | ||||
|     styled-jsx "5.1.1" | ||||
|     watchpack "2.4.0" | ||||
|   optionalDependencies: | ||||
|     "@next/swc-darwin-arm64" "14.0.4" | ||||
|     "@next/swc-darwin-x64" "14.0.4" | ||||
|     "@next/swc-linux-arm64-gnu" "14.0.4" | ||||
|     "@next/swc-linux-arm64-musl" "14.0.4" | ||||
|     "@next/swc-linux-x64-gnu" "14.0.4" | ||||
|     "@next/swc-linux-x64-musl" "14.0.4" | ||||
|     "@next/swc-win32-arm64-msvc" "14.0.4" | ||||
|     "@next/swc-win32-ia32-msvc" "14.0.4" | ||||
|     "@next/swc-win32-x64-msvc" "14.0.4" | ||||
|  | ||||
| path2d-polyfill@2.0.1: | ||||
|   version "2.0.1" | ||||
|   resolved "https://registry.yarnpkg.com/path2d-polyfill/-/path2d-polyfill-2.0.1.tgz#24c554a738f42700d6961992bf5f1049672f2391" | ||||
|   integrity sha512-ad/3bsalbbWhmBo0D6FZ4RNMwsLsPpL6gnvhuSaU5Vm7b06Kr5ubSltQQ0T7YKsiJQO+g22zJ4dJKNTXIyOXtA== | ||||
|  | ||||
| picocolors@^1.0.0: | ||||
|   version "1.0.0" | ||||
|   resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" | ||||
|   integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== | ||||
|  | ||||
| postcss@8.4.31: | ||||
|   version "8.4.31" | ||||
|   resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.31.tgz#92b451050a9f914da6755af352bdc0192508656d" | ||||
|   integrity sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ== | ||||
|   dependencies: | ||||
|     nanoid "^3.3.6" | ||||
|     picocolors "^1.0.0" | ||||
|     source-map-js "^1.0.2" | ||||
|  | ||||
| react-dom@^18: | ||||
|   version "18.2.0" | ||||
|   resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-18.2.0.tgz#22aaf38708db2674ed9ada224ca4aa708d821e3d" | ||||
|   integrity sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g== | ||||
|   dependencies: | ||||
|     loose-envify "^1.1.0" | ||||
|     scheduler "^0.23.0" | ||||
|  | ||||
| react@^18: | ||||
|   version "18.2.0" | ||||
|   resolved "https://registry.yarnpkg.com/react/-/react-18.2.0.tgz#555bd98592883255fa00de14f1151a917b5d77d5" | ||||
|   integrity sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ== | ||||
|   dependencies: | ||||
|     loose-envify "^1.1.0" | ||||
|  | ||||
| scheduler@^0.23.0: | ||||
|   version "0.23.0" | ||||
|   resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.23.0.tgz#ba8041afc3d30eb206a487b6b384002e4e61fdfe" | ||||
|   integrity sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw== | ||||
|   dependencies: | ||||
|     loose-envify "^1.1.0" | ||||
|  | ||||
| source-map-js@^1.0.2: | ||||
|   version "1.0.2" | ||||
|   resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.0.2.tgz#adbc361d9c62df380125e7f161f71c826f1e490c" | ||||
|   integrity sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw== | ||||
|  | ||||
| streamsearch@^1.1.0: | ||||
|   version "1.1.0" | ||||
|   resolved "https://registry.yarnpkg.com/streamsearch/-/streamsearch-1.1.0.tgz#404dd1e2247ca94af554e841a8ef0eaa238da764" | ||||
|   integrity sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg== | ||||
|  | ||||
| styled-jsx@5.1.1: | ||||
|   version "5.1.1" | ||||
|   resolved "https://registry.yarnpkg.com/styled-jsx/-/styled-jsx-5.1.1.tgz#839a1c3aaacc4e735fed0781b8619ea5d0009d1f" | ||||
|   integrity sha512-pW7uC1l4mBZ8ugbiZrcIsiIvVx1UmTfw7UkC3Um2tmfUq9Bhk8IiyEIPl6F8agHgjzku6j0xQEZbfA5uSgSaCw== | ||||
|   dependencies: | ||||
|     client-only "0.0.1" | ||||
|  | ||||
| tslib@^2.4.0: | ||||
|   version "2.6.2" | ||||
|   resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.2.tgz#703ac29425e7b37cd6fd456e92404d46d1f3e4ae" | ||||
|   integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q== | ||||
|  | ||||
| typescript@^5: | ||||
|   version "5.3.3" | ||||
|   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.3.3.tgz#b3ce6ba258e72e6305ba66f5c9b452aaee3ffe37" | ||||
|   integrity sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw== | ||||
|  | ||||
| undici-types@~5.26.4: | ||||
|   version "5.26.5" | ||||
|   resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" | ||||
|   integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== | ||||
|  | ||||
| watchpack@2.4.0: | ||||
|   version "2.4.0" | ||||
|   resolved "https://registry.yarnpkg.com/watchpack/-/watchpack-2.4.0.tgz#fa33032374962c78113f93c7f2fb4c54c9862a5d" | ||||
|   integrity sha512-Lcvm7MGST/4fup+ifyKi2hjyIAwcdI4HRgtvTpIUxBRhB+RFtUh8XtDOxUfctVCnhVi+QQj49i91OyvzkJl6cg== | ||||
|   dependencies: | ||||
|     glob-to-regexp "^0.4.1" | ||||
|     graceful-fs "^4.1.2" | ||||
							
								
								
									
										5
									
								
								examples/with-script-in-browser/.codesandbox/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| FROM node:18-bullseye | ||||
|  | ||||
| # Vite wants to open the browser using `open`, so we | ||||
| # need to install those utils. | ||||
| RUN apt update -y && apt install -y xdg-utils | ||||
							
								
								
									
										35
									
								
								examples/with-script-in-browser/.codesandbox/tasks.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | ||||
| { | ||||
|   // These tasks will run in order when initializing your CodeSandbox project. | ||||
|   "setupTasks": [ | ||||
|     { | ||||
|       "name": "Install Dependencies", | ||||
|       "command": "yarn install" | ||||
|     } | ||||
|   ], | ||||
|  | ||||
|   // These tasks can be run from CodeSandbox. Running one will open a log in the app. | ||||
|   "tasks": { | ||||
|     "build": { | ||||
|       "name": "Build", | ||||
|       "command": "yarn build", | ||||
|       "runAtStart": false | ||||
|     }, | ||||
|     "start": { | ||||
|       "name": "Start Example", | ||||
|       "command": "yarn start", | ||||
|       "runAtStart": true, | ||||
|       "preview": { | ||||
|         "port": 3001 | ||||
|       } | ||||
|     }, | ||||
|     "install-deps": { | ||||
|       "name": "Install Dependencies", | ||||
|       "command": "yarn install", | ||||
|       "restartOn": { | ||||
|         "files": ["yarn.lock"], | ||||
|         "branch": false, | ||||
|         "resume": false | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										2
									
								
								examples/with-script-in-browser/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | ||||
| # copied assets | ||||
| public/**/*.woff2 | ||||
| @@ -1,6 +1,7 @@ | ||||
| import { ExcalidrawImperativeAPI } from "../../../types"; | ||||
| import { MIME_TYPES } from "../entry"; | ||||
| import { Button } from "../../../components/Button"; | ||||
| import React from "react"; | ||||
| 
 | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
| import type { ExcalidrawImperativeAPI } from "@excalidraw/excalidraw/types"; | ||||
| 
 | ||||
| const COMMENT_SVG = ( | ||||
|   <svg | ||||
| @@ -18,24 +19,28 @@ const COMMENT_SVG = ( | ||||
|     <path d="M21 11.5a8.38 8.38 0 0 1-.9 3.8 8.5 8.5 0 0 1-7.6 4.7 8.38 8.38 0 0 1-3.8-.9L3 21l1.9-5.7a8.38 8.38 0 0 1-.9-3.8 8.5 8.5 0 0 1 4.7-7.6 8.38 8.38 0 0 1 3.8-.9h.5a8.48 8.48 0 0 1 8 8v.5z"></path> | ||||
|   </svg> | ||||
| ); | ||||
| 
 | ||||
| const CustomFooter = ({ | ||||
|   excalidrawAPI, | ||||
|   excalidrawLib, | ||||
| }: { | ||||
|   excalidrawAPI: ExcalidrawImperativeAPI; | ||||
|   excalidrawLib: typeof TExcalidraw; | ||||
| }) => { | ||||
|   const { Button, MIME_TYPES } = excalidrawLib; | ||||
| 
 | ||||
|   return ( | ||||
|     <> | ||||
|       <Button | ||||
|         onSelect={() => alert("General Kenobi!")} | ||||
|         className="you are a bold one" | ||||
|         style={{ marginLeft: "1rem" }} | ||||
|         style={{ marginLeft: "1rem", width: "auto" }} | ||||
|         title="Hello there!" | ||||
|       > | ||||
|         {COMMENT_SVG} | ||||
|         Hit me | ||||
|       </Button> | ||||
|       <button | ||||
|       <Button | ||||
|         className="custom-element" | ||||
|         onClick={() => { | ||||
|         onSelect={() => { | ||||
|           excalidrawAPI?.setActiveTool({ | ||||
|             type: "custom", | ||||
|             customType: "comment", | ||||
| @@ -58,15 +63,10 @@ const CustomFooter = ({ | ||||
|           )}`;
 | ||||
|           excalidrawAPI?.setCursor(`url(${url}), auto`); | ||||
|         }} | ||||
|         title="Comments!" | ||||
|       > | ||||
|         {COMMENT_SVG} | ||||
|       </button> | ||||
|       <button | ||||
|         className="custom-footer" | ||||
|         onClick={() => alert("This is dummy footer")} | ||||
|       > | ||||
|         custom footer | ||||
|       </button> | ||||
|       </Button> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
| @@ -15,14 +15,23 @@ | ||||
|       border-radius: 50%; | ||||
|     } | ||||
|   } | ||||
|   .app-title { | ||||
|     margin-block-start: 0.83em; | ||||
|     margin-block-end: 0.83em; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .button-wrapper button { | ||||
|   z-index: 1; | ||||
|   height: 40px; | ||||
|   max-width: 200px; | ||||
|   margin: 10px; | ||||
|   padding: 5px; | ||||
| .button-wrapper { | ||||
|   input[type="checkbox"] { | ||||
|     margin: 5px; | ||||
|   } | ||||
|   button { | ||||
|     z-index: 1; | ||||
|     height: 40px; | ||||
|     max-width: 200px; | ||||
|     margin: 10px; | ||||
|     padding: 5px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .excalidraw .App-menu_top .buttonList { | ||||
| @@ -43,7 +52,7 @@ | ||||
|   transform: none; | ||||
| } | ||||
| 
 | ||||
| .excalidraw .panelColumn { | ||||
| .excalidraw .selected-shape-actions { | ||||
|   text-align: left; | ||||
| } | ||||
| 
 | ||||
| @@ -1,23 +1,20 @@ | ||||
| import { useEffect, useState, useRef, useCallback } from "react"; | ||||
| 
 | ||||
| import ExampleSidebar from "./sidebar/ExampleSidebar"; | ||||
| 
 | ||||
| import type * as TExcalidraw from "../index"; | ||||
| 
 | ||||
| import "./App.scss"; | ||||
| import initialData from "./initialData"; | ||||
| import { nanoid } from "nanoid"; | ||||
| import { | ||||
|   resolvablePromise, | ||||
|   ResolvablePromise, | ||||
|   withBatchedUpdates, | ||||
|   withBatchedUpdatesThrottled, | ||||
| } from "../../../utils"; | ||||
| import { EVENT, ROUNDNESS } from "../../../constants"; | ||||
| import { distance2d } from "../../../math"; | ||||
| import { fileOpen } from "../../../data/filesystem"; | ||||
| import { loadSceneOrLibraryFromBlob } from "../../utils"; | ||||
| import { | ||||
| import React, { | ||||
|   useEffect, | ||||
|   useState, | ||||
|   useRef, | ||||
|   useCallback, | ||||
|   Children, | ||||
|   cloneElement, | ||||
| } from "react"; | ||||
| 
 | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
| import type { ImportedLibraryData } from "@excalidraw/excalidraw/data/types"; | ||||
| import type { | ||||
|   NonDeletedExcalidrawElement, | ||||
|   Theme, | ||||
| } from "@excalidraw/excalidraw/element/types"; | ||||
| import type { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   ExcalidrawImperativeAPI, | ||||
| @@ -25,18 +22,24 @@ import { | ||||
|   Gesture, | ||||
|   LibraryItems, | ||||
|   PointerDownState as ExcalidrawPointerDownState, | ||||
| } from "../../../types"; | ||||
| import { NonDeletedExcalidrawElement, Theme } from "../../../element/types"; | ||||
| import { ImportedLibraryData } from "../../../data/types"; | ||||
| } from "@excalidraw/excalidraw/types"; | ||||
| 
 | ||||
| import initialData from "../initialData"; | ||||
| import { | ||||
|   resolvablePromise, | ||||
|   distance2d, | ||||
|   fileOpen, | ||||
|   withBatchedUpdates, | ||||
|   withBatchedUpdatesThrottled, | ||||
| } from "../utils"; | ||||
| 
 | ||||
| import CustomFooter from "./CustomFooter"; | ||||
| import MobileFooter from "./MobileFooter"; | ||||
| import { KEYS } from "../../../keys"; | ||||
| import ExampleSidebar from "./sidebar/ExampleSidebar"; | ||||
| 
 | ||||
| declare global { | ||||
|   interface Window { | ||||
|     ExcalidrawLib: typeof TExcalidraw; | ||||
|   } | ||||
| } | ||||
| import "./ExampleApp.scss"; | ||||
| 
 | ||||
| import type { ResolvablePromise } from "../utils"; | ||||
| 
 | ||||
| type Comment = { | ||||
|   x: number; | ||||
| @@ -57,27 +60,6 @@ type PointerDownState = { | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| // This is so that we use the bundled excalidraw.development.js file instead
 | ||||
| // of the actual source code
 | ||||
| const { | ||||
|   exportToCanvas, | ||||
|   exportToSvg, | ||||
|   exportToBlob, | ||||
|   exportToClipboard, | ||||
|   Excalidraw, | ||||
|   useHandleLibrary, | ||||
|   MIME_TYPES, | ||||
|   sceneCoordsToViewportCoords, | ||||
|   viewportCoordsToSceneCoords, | ||||
|   restoreElements, | ||||
|   Sidebar, | ||||
|   Footer, | ||||
|   WelcomeScreen, | ||||
|   MainMenu, | ||||
|   LiveCollaborationTrigger, | ||||
|   convertToExcalidrawElements, | ||||
| } = window.ExcalidrawLib; | ||||
| 
 | ||||
| const COMMENT_ICON_DIMENSION = 32; | ||||
| const COMMENT_INPUT_HEIGHT = 50; | ||||
| const COMMENT_INPUT_WIDTH = 150; | ||||
| @@ -86,13 +68,43 @@ export interface AppProps { | ||||
|   appTitle: string; | ||||
|   useCustom: (api: ExcalidrawImperativeAPI | null, customArgs?: any[]) => void; | ||||
|   customArgs?: any[]; | ||||
|   children: React.ReactNode; | ||||
|   excalidrawLib: typeof TExcalidraw; | ||||
| } | ||||
| 
 | ||||
| export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
| export default function ExampleApp({ | ||||
|   appTitle, | ||||
|   useCustom, | ||||
|   customArgs, | ||||
|   children, | ||||
|   excalidrawLib, | ||||
| }: AppProps) { | ||||
|   const { | ||||
|     exportToCanvas, | ||||
|     exportToSvg, | ||||
|     exportToBlob, | ||||
|     exportToClipboard, | ||||
|     useHandleLibrary, | ||||
|     MIME_TYPES, | ||||
|     sceneCoordsToViewportCoords, | ||||
|     viewportCoordsToSceneCoords, | ||||
|     restoreElements, | ||||
|     Sidebar, | ||||
|     Footer, | ||||
|     WelcomeScreen, | ||||
|     MainMenu, | ||||
|     LiveCollaborationTrigger, | ||||
|     convertToExcalidrawElements, | ||||
|     TTDDialog, | ||||
|     TTDDialogTrigger, | ||||
|     ROUNDNESS, | ||||
|     loadSceneOrLibraryFromBlob, | ||||
|   } = excalidrawLib; | ||||
|   const appRef = useRef<any>(null); | ||||
|   const [viewModeEnabled, setViewModeEnabled] = useState(false); | ||||
|   const [zenModeEnabled, setZenModeEnabled] = useState(false); | ||||
|   const [gridModeEnabled, setGridModeEnabled] = useState(false); | ||||
|   const [renderScrollbars, setRenderScrollbars] = useState(false); | ||||
|   const [blobUrl, setBlobUrl] = useState<string>(""); | ||||
|   const [canvasUrl, setCanvasUrl] = useState<string>(""); | ||||
|   const [exportWithDarkMode, setExportWithDarkMode] = useState(false); | ||||
| @@ -150,8 +162,106 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|       }; | ||||
|     }; | ||||
|     fetchData(); | ||||
|   }, [excalidrawAPI]); | ||||
|   }, [excalidrawAPI, convertToExcalidrawElements, MIME_TYPES]); | ||||
| 
 | ||||
|   const renderExcalidraw = (children: React.ReactNode) => { | ||||
|     const Excalidraw: any = Children.toArray(children).find( | ||||
|       (child) => | ||||
|         React.isValidElement(child) && | ||||
|         typeof child.type !== "string" && | ||||
|         //@ts-ignore
 | ||||
|         child.type.displayName === "Excalidraw", | ||||
|     ); | ||||
|     if (!Excalidraw) { | ||||
|       return; | ||||
|     } | ||||
|     const newElement = cloneElement( | ||||
|       Excalidraw, | ||||
|       { | ||||
|         excalidrawAPI: (api: ExcalidrawImperativeAPI) => setExcalidrawAPI(api), | ||||
|         initialData: initialStatePromiseRef.current.promise, | ||||
|         onChange: ( | ||||
|           elements: NonDeletedExcalidrawElement[], | ||||
|           state: AppState, | ||||
|         ) => { | ||||
|           console.info("Elements :", elements, "State : ", state); | ||||
|         }, | ||||
|         onPointerUpdate: (payload: { | ||||
|           pointer: { x: number; y: number }; | ||||
|           button: "down" | "up"; | ||||
|           pointersMap: Gesture["pointers"]; | ||||
|         }) => setPointerData(payload), | ||||
|         viewModeEnabled, | ||||
|         zenModeEnabled, | ||||
|         renderScrollbars, | ||||
|         gridModeEnabled, | ||||
|         theme, | ||||
|         name: "Custom name of drawing", | ||||
|         UIOptions: { | ||||
|           canvasActions: { | ||||
|             loadScene: false, | ||||
|           }, | ||||
|           tools: { image: !disableImageTool }, | ||||
|         }, | ||||
|         renderTopRightUI, | ||||
|         onLinkOpen, | ||||
|         onPointerDown, | ||||
|         onScrollChange: rerenderCommentIcons, | ||||
|         validateEmbeddable: true, | ||||
|       }, | ||||
|       <> | ||||
|         {excalidrawAPI && ( | ||||
|           <Footer> | ||||
|             <CustomFooter | ||||
|               excalidrawAPI={excalidrawAPI} | ||||
|               excalidrawLib={excalidrawLib} | ||||
|             /> | ||||
|           </Footer> | ||||
|         )} | ||||
|         <WelcomeScreen /> | ||||
|         <Sidebar name="custom"> | ||||
|           <Sidebar.Tabs> | ||||
|             <Sidebar.Header /> | ||||
|             <Sidebar.Tab tab="one">Tab one!</Sidebar.Tab> | ||||
|             <Sidebar.Tab tab="two">Tab two!</Sidebar.Tab> | ||||
|             <Sidebar.TabTriggers> | ||||
|               <Sidebar.TabTrigger tab="one">One</Sidebar.TabTrigger> | ||||
|               <Sidebar.TabTrigger tab="two">Two</Sidebar.TabTrigger> | ||||
|             </Sidebar.TabTriggers> | ||||
|           </Sidebar.Tabs> | ||||
|         </Sidebar> | ||||
|         <Sidebar.Trigger | ||||
|           name="custom" | ||||
|           tab="one" | ||||
|           style={{ | ||||
|             position: "absolute", | ||||
|             left: "50%", | ||||
|             transform: "translateX(-50%)", | ||||
|             bottom: "20px", | ||||
|             zIndex: 9999999999999999, | ||||
|           }} | ||||
|         > | ||||
|           Toggle Custom Sidebar | ||||
|         </Sidebar.Trigger> | ||||
|         {renderMenu()} | ||||
|         {excalidrawAPI && ( | ||||
|           <TTDDialogTrigger icon={<span>😀</span>}> | ||||
|             Text to diagram | ||||
|           </TTDDialogTrigger> | ||||
|         )} | ||||
|         <TTDDialog | ||||
|           onTextSubmit={async (_) => { | ||||
|             console.info("submit"); | ||||
|             // sleep for 2s
 | ||||
|             await new Promise((resolve) => setTimeout(resolve, 2000)); | ||||
|             throw new Error("error, go away now"); | ||||
|             // return "dummy";
 | ||||
|           }} | ||||
|         /> | ||||
|       </>, | ||||
|     ); | ||||
|     return newElement; | ||||
|   }; | ||||
|   const renderTopRightUI = (isMobile: boolean) => { | ||||
|     return ( | ||||
|       <> | ||||
| @@ -335,8 +445,8 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|     pointerDownState: PointerDownState, | ||||
|   ) => { | ||||
|     return withBatchedUpdates((event) => { | ||||
|       window.removeEventListener(EVENT.POINTER_MOVE, pointerDownState.onMove); | ||||
|       window.removeEventListener(EVENT.POINTER_UP, pointerDownState.onUp); | ||||
|       window.removeEventListener("pointermove", pointerDownState.onMove); | ||||
|       window.removeEventListener("pointerup", pointerDownState.onUp); | ||||
|       excalidrawAPI?.setActiveTool({ type: "selection" }); | ||||
|       const distance = distance2d( | ||||
|         pointerDownState.x, | ||||
| @@ -400,8 +510,8 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|               onPointerMoveFromPointerDownHandler(pointerDownState); | ||||
|             const onPointerUp = | ||||
|               onPointerUpFromPointerDownHandler(pointerDownState); | ||||
|             window.addEventListener(EVENT.POINTER_MOVE, onPointerMove); | ||||
|             window.addEventListener(EVENT.POINTER_UP, onPointerUp); | ||||
|             window.addEventListener("pointermove", onPointerMove); | ||||
|             window.addEventListener("pointerup", onPointerUp); | ||||
| 
 | ||||
|             pointerDownState.onMove = onPointerMove; | ||||
|             pointerDownState.onUp = onPointerUp; | ||||
| @@ -493,7 +603,7 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|         }} | ||||
|         onBlur={saveComment} | ||||
|         onKeyDown={(event) => { | ||||
|           if (!event.shiftKey && event.key === KEYS.ENTER) { | ||||
|           if (!event.shiftKey && event.key === "Enter") { | ||||
|             event.preventDefault(); | ||||
|             saveComment(); | ||||
|           } | ||||
| @@ -526,7 +636,12 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|         </MainMenu.ItemCustom> | ||||
|         <MainMenu.DefaultItems.Help /> | ||||
| 
 | ||||
|         {excalidrawAPI && <MobileFooter excalidrawAPI={excalidrawAPI} />} | ||||
|         {excalidrawAPI && ( | ||||
|           <MobileFooter | ||||
|             excalidrawLib={excalidrawLib} | ||||
|             excalidrawAPI={excalidrawAPI} | ||||
|           /> | ||||
|         )} | ||||
|       </MainMenu> | ||||
|     ); | ||||
|   }; | ||||
| @@ -597,6 +712,14 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|             /> | ||||
|             Grid mode | ||||
|           </label> | ||||
|           <label> | ||||
|             <input | ||||
|               type="checkbox" | ||||
|               checked={renderScrollbars} | ||||
|               onChange={() => setRenderScrollbars(!renderScrollbars)} | ||||
|             /> | ||||
|             Render scrollbars | ||||
|           </label> | ||||
|           <label> | ||||
|             <input | ||||
|               type="checkbox" | ||||
| @@ -675,69 +798,7 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|           </div> | ||||
|         </div> | ||||
|         <div className="excalidraw-wrapper"> | ||||
|           <Excalidraw | ||||
|             excalidrawAPI={(api: ExcalidrawImperativeAPI) => | ||||
|               setExcalidrawAPI(api) | ||||
|             } | ||||
|             initialData={initialStatePromiseRef.current.promise} | ||||
|             onChange={(elements, state) => { | ||||
|               console.info("Elements :", elements, "State : ", state); | ||||
|             }} | ||||
|             onPointerUpdate={(payload: { | ||||
|               pointer: { x: number; y: number }; | ||||
|               button: "down" | "up"; | ||||
|               pointersMap: Gesture["pointers"]; | ||||
|             }) => setPointerData(payload)} | ||||
|             viewModeEnabled={viewModeEnabled} | ||||
|             zenModeEnabled={zenModeEnabled} | ||||
|             gridModeEnabled={gridModeEnabled} | ||||
|             theme={theme} | ||||
|             name="Custom name of drawing" | ||||
|             UIOptions={{ | ||||
|               canvasActions: { | ||||
|                 loadScene: false, | ||||
|               }, | ||||
|               tools: { image: !disableImageTool }, | ||||
|             }} | ||||
|             renderTopRightUI={renderTopRightUI} | ||||
|             onLinkOpen={onLinkOpen} | ||||
|             onPointerDown={onPointerDown} | ||||
|             onScrollChange={rerenderCommentIcons} | ||||
|             // allow all urls
 | ||||
|             validateEmbeddable={true} | ||||
|           > | ||||
|             {excalidrawAPI && ( | ||||
|               <Footer> | ||||
|                 <CustomFooter excalidrawAPI={excalidrawAPI} /> | ||||
|               </Footer> | ||||
|             )} | ||||
|             <WelcomeScreen /> | ||||
|             <Sidebar name="custom"> | ||||
|               <Sidebar.Tabs> | ||||
|                 <Sidebar.Header /> | ||||
|                 <Sidebar.Tab tab="one">Tab one!</Sidebar.Tab> | ||||
|                 <Sidebar.Tab tab="two">Tab two!</Sidebar.Tab> | ||||
|                 <Sidebar.TabTriggers> | ||||
|                   <Sidebar.TabTrigger tab="one">One</Sidebar.TabTrigger> | ||||
|                   <Sidebar.TabTrigger tab="two">Two</Sidebar.TabTrigger> | ||||
|                 </Sidebar.TabTriggers> | ||||
|               </Sidebar.Tabs> | ||||
|             </Sidebar> | ||||
|             <Sidebar.Trigger | ||||
|               name="custom" | ||||
|               tab="one" | ||||
|               style={{ | ||||
|                 position: "absolute", | ||||
|                 left: "50%", | ||||
|                 transform: "translateX(-50%)", | ||||
|                 bottom: "20px", | ||||
|                 zIndex: 9999999999999999, | ||||
|               }} | ||||
|             > | ||||
|               Toggle Custom Sidebar | ||||
|             </Sidebar.Trigger> | ||||
|             {renderMenu()} | ||||
|           </Excalidraw> | ||||
|           {renderExcalidraw(children)} | ||||
|           {Object.keys(commentIcons || []).length > 0 && renderCommentIcons()} | ||||
|           {comment && renderComment()} | ||||
|         </div> | ||||
| @@ -820,7 +881,7 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|                 files: excalidrawAPI.getFiles(), | ||||
|               }); | ||||
|               const ctx = canvas.getContext("2d")!; | ||||
|               ctx.font = "30px Virgil"; | ||||
|               ctx.font = "30px Excalifont"; | ||||
|               ctx.strokeText("My custom text", 50, 60); | ||||
|               setCanvasUrl(canvas.toDataURL()); | ||||
|             }} | ||||
| @@ -841,7 +902,7 @@ export default function App({ appTitle, useCustom, customArgs }: AppProps) { | ||||
|                 files: excalidrawAPI.getFiles(), | ||||
|               }); | ||||
|               const ctx = canvas.getContext("2d")!; | ||||
|               ctx.font = "30px Virgil"; | ||||
|               ctx.font = "30px Excalifont"; | ||||
|               ctx.strokeText("My custom text", 50, 60); | ||||
|               setCanvasUrl(canvas.toDataURL()); | ||||
|             }} | ||||
							
								
								
									
										30
									
								
								examples/with-script-in-browser/components/MobileFooter.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,30 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
| import type { ExcalidrawImperativeAPI } from "@excalidraw/excalidraw/types"; | ||||
|  | ||||
| import CustomFooter from "./CustomFooter"; | ||||
|  | ||||
| const MobileFooter = ({ | ||||
|   excalidrawAPI, | ||||
|   excalidrawLib, | ||||
| }: { | ||||
|   excalidrawAPI: ExcalidrawImperativeAPI; | ||||
|   excalidrawLib: typeof TExcalidraw; | ||||
| }) => { | ||||
|   const { useDevice, Footer } = excalidrawLib; | ||||
|  | ||||
|   const device = useDevice(); | ||||
|   if (device.editor.isMobile) { | ||||
|     return ( | ||||
|       <Footer> | ||||
|         <CustomFooter | ||||
|           excalidrawAPI={excalidrawAPI} | ||||
|           excalidrawLib={excalidrawLib} | ||||
|         /> | ||||
|       </Footer> | ||||
|     ); | ||||
|   } | ||||
|   return null; | ||||
| }; | ||||
| export default MobileFooter; | ||||
| @@ -1,5 +1,7 @@ | ||||
| import React, { useState } from "react"; | ||||
| 
 | ||||
| import "./ExampleSidebar.scss"; | ||||
| 
 | ||||
| export default function Sidebar({ children }: { children: React.ReactNode }) { | ||||
|   const [open, setOpen] = useState(false); | ||||
| 
 | ||||
| @@ -11,19 +11,21 @@ | ||||
|     <title>React App</title> | ||||
|     <script> | ||||
|       window.name = "codesandbox"; | ||||
|       window.EXCALIDRAW_ASSET_PATH = | ||||
|         "https://esm.sh/@excalidraw/excalidraw@0.18.0/dist/prod/"; | ||||
|     </script> | ||||
|   </head> | ||||
| 
 | ||||
|   <body> | ||||
|     <noscript> You need to enable JavaScript to run this app. </noscript> | ||||
|     <div id="root"></div> | ||||
|     <script src="https://unpkg.com/react@18.2.0/umd/react.development.js"></script> | ||||
|     <script src="https://unpkg.com/react-dom@18.2.0/umd/react-dom.development.js"></script> | ||||
| 
 | ||||
|     <!-- This is so that we use the bundled excalidraw.development.js file instead | ||||
|     of the actual source code --> | ||||
|     <script src="./excalidraw.development.js"></script> | ||||
|     <script type="module"> | ||||
|       import * as ExcalidrawLib from "@excalidraw/excalidraw"; | ||||
| 
 | ||||
|     <script src="./bundle.js"></script> | ||||
|       console.log(ExcalidrawLib); | ||||
|       window.ExcalidrawLib = ExcalidrawLib; | ||||
|     </script> | ||||
|     <script type="module" src="index.tsx"></script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										29
									
								
								examples/with-script-in-browser/index.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,29 @@ | ||||
| import React, { StrictMode } from "react"; | ||||
| import { createRoot } from "react-dom/client"; | ||||
|  | ||||
| import "@excalidraw/excalidraw/index.css"; | ||||
|  | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
|  | ||||
| import App from "./components/ExampleApp"; | ||||
|  | ||||
| declare global { | ||||
|   interface Window { | ||||
|     ExcalidrawLib: typeof TExcalidraw; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const rootElement = document.getElementById("root")!; | ||||
| const root = createRoot(rootElement); | ||||
| const { Excalidraw } = window.ExcalidrawLib; | ||||
| root.render( | ||||
|   <StrictMode> | ||||
|     <App | ||||
|       appTitle={"Excalidraw Example"} | ||||
|       useCustom={(api: any, args?: any[]) => {}} | ||||
|       excalidrawLib={window.ExcalidrawLib} | ||||
|     > | ||||
|       <Excalidraw /> | ||||
|     </App> | ||||
|   </StrictMode>, | ||||
| ); | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { ExcalidrawElementSkeleton } from "../../../data/transform"; | ||||
| import { FileId } from "../../../element/types"; | ||||
| import type { ExcalidrawElementSkeleton } from "@excalidraw/excalidraw/data/transform"; | ||||
| import type { FileId } from "@excalidraw/excalidraw/element/types"; | ||||
| 
 | ||||
| const elements: ExcalidrawElementSkeleton[] = [ | ||||
|   { | ||||
| @@ -46,7 +46,7 @@ const elements: ExcalidrawElementSkeleton[] = [ | ||||
| ]; | ||||
| export default { | ||||
|   elements, | ||||
|   appState: { viewBackgroundColor: "#AFEEEE", currentItemFontFamily: 1 }, | ||||
|   appState: { viewBackgroundColor: "#AFEEEE", currentItemFontFamily: 5 }, | ||||
|   scrollToContent: true, | ||||
|   libraryItems: [ | ||||
|     [ | ||||
							
								
								
									
										22
									
								
								examples/with-script-in-browser/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| { | ||||
|   "name": "with-script-in-browser", | ||||
|   "version": "1.0.0", | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "react": "19.0.0", | ||||
|     "react-dom": "19.0.0", | ||||
|     "@excalidraw/excalidraw": "*", | ||||
|     "browser-fs-access": "0.29.1" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "vite": "5.0.12", | ||||
|     "typescript": "^5" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "start": "vite", | ||||
|     "build": "vite build", | ||||
|     "preview": "vite preview --port 5002", | ||||
|     "build:preview": "yarn build && yarn preview", | ||||
|     "build:package": "yarn workspace @excalidraw/excalidraw run build:esm" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										
											BIN
										
									
								
								examples/with-script-in-browser/public/images/doremon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 197 KiB | 
							
								
								
									
										
											BIN
										
									
								
								examples/with-script-in-browser/public/images/excalibot.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 30 KiB | 
							
								
								
									
										
											BIN
										
									
								
								examples/with-script-in-browser/public/images/pika.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 6.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								examples/with-script-in-browser/public/images/rocket.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 39 KiB | 
							
								
								
									
										9
									
								
								examples/with-script-in-browser/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| { | ||||
|   "compilerOptions": { | ||||
|     "module": "ES2022", | ||||
|     "moduleResolution": "Bundler", | ||||
|     "lib": ["ESNext", "DOM", "DOM.Iterable"], | ||||
|     "jsx": "react-jsx", | ||||
|     "skipLibCheck": true | ||||
|   } | ||||
| } | ||||
							
								
								
									
										145
									
								
								examples/with-script-in-browser/utils.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,145 @@ | ||||
| import { MIME_TYPES } from "@excalidraw/excalidraw"; | ||||
| import { fileOpen as _fileOpen } from "browser-fs-access"; | ||||
| import { unstable_batchedUpdates } from "react-dom"; | ||||
|  | ||||
| type FILE_EXTENSION = Exclude<keyof typeof MIME_TYPES, "binary">; | ||||
|  | ||||
| const INPUT_CHANGE_INTERVAL_MS = 500; | ||||
|  | ||||
| export type ResolvablePromise<T> = Promise<T> & { | ||||
|   resolve: [T] extends [undefined] ? (value?: T) => void : (value: T) => void; | ||||
|   reject: (error: Error) => void; | ||||
| }; | ||||
| export const resolvablePromise = <T>() => { | ||||
|   let resolve!: any; | ||||
|   let reject!: any; | ||||
|   const promise = new Promise((_resolve, _reject) => { | ||||
|     resolve = _resolve; | ||||
|     reject = _reject; | ||||
|   }); | ||||
|   (promise as any).resolve = resolve; | ||||
|   (promise as any).reject = reject; | ||||
|   return promise as ResolvablePromise<T>; | ||||
| }; | ||||
|  | ||||
| export const distance2d = (x1: number, y1: number, x2: number, y2: number) => { | ||||
|   const xd = x2 - x1; | ||||
|   const yd = y2 - y1; | ||||
|   return Math.hypot(xd, yd); | ||||
| }; | ||||
|  | ||||
| export const fileOpen = <M extends boolean | undefined = false>(opts: { | ||||
|   extensions?: FILE_EXTENSION[]; | ||||
|   description: string; | ||||
|   multiple?: M; | ||||
| }): Promise<M extends false | undefined ? File : File[]> => { | ||||
|   // an unsafe TS hack, alas not much we can do AFAIK | ||||
|   type RetType = M extends false | undefined ? File : File[]; | ||||
|  | ||||
|   const mimeTypes = opts.extensions?.reduce((mimeTypes, type) => { | ||||
|     mimeTypes.push(MIME_TYPES[type]); | ||||
|  | ||||
|     return mimeTypes; | ||||
|   }, [] as string[]); | ||||
|  | ||||
|   const extensions = opts.extensions?.reduce((acc, ext) => { | ||||
|     if (ext === "jpg") { | ||||
|       return acc.concat(".jpg", ".jpeg"); | ||||
|     } | ||||
|     return acc.concat(`.${ext}`); | ||||
|   }, [] as string[]); | ||||
|  | ||||
|   return _fileOpen({ | ||||
|     description: opts.description, | ||||
|     extensions, | ||||
|     mimeTypes, | ||||
|     multiple: opts.multiple ?? false, | ||||
|     legacySetup: (resolve, reject, input) => { | ||||
|       const scheduleRejection = debounce(reject, INPUT_CHANGE_INTERVAL_MS); | ||||
|       const focusHandler = () => { | ||||
|         checkForFile(); | ||||
|         document.addEventListener("keyup", scheduleRejection); | ||||
|         document.addEventListener("pointerup", scheduleRejection); | ||||
|         scheduleRejection(); | ||||
|       }; | ||||
|       const checkForFile = () => { | ||||
|         // this hack might not work when expecting multiple files | ||||
|         if (input.files?.length) { | ||||
|           const ret = opts.multiple ? [...input.files] : input.files[0]; | ||||
|           resolve(ret as RetType); | ||||
|         } | ||||
|       }; | ||||
|       requestAnimationFrame(() => { | ||||
|         window.addEventListener("focus", focusHandler); | ||||
|       }); | ||||
|       const interval = window.setInterval(() => { | ||||
|         checkForFile(); | ||||
|       }, INPUT_CHANGE_INTERVAL_MS); | ||||
|       return (rejectPromise) => { | ||||
|         clearInterval(interval); | ||||
|         scheduleRejection.cancel(); | ||||
|         window.removeEventListener("focus", focusHandler); | ||||
|         document.removeEventListener("keyup", scheduleRejection); | ||||
|         document.removeEventListener("pointerup", scheduleRejection); | ||||
|         if (rejectPromise) { | ||||
|           // so that something is shown in console if we need to debug this | ||||
|           console.warn("Opening the file was canceled (legacy-fs)."); | ||||
|           rejectPromise(new Error("Request Aborted")); | ||||
|         } | ||||
|       }; | ||||
|     }, | ||||
|   }) as Promise<RetType>; | ||||
| }; | ||||
|  | ||||
| export const debounce = <T extends any[]>( | ||||
|   fn: (...args: T) => void, | ||||
|   timeout: number, | ||||
| ) => { | ||||
|   let handle = 0; | ||||
|   let lastArgs: T | null = null; | ||||
|   const ret = (...args: T) => { | ||||
|     lastArgs = args; | ||||
|     clearTimeout(handle); | ||||
|     handle = window.setTimeout(() => { | ||||
|       lastArgs = null; | ||||
|       fn(...args); | ||||
|     }, timeout); | ||||
|   }; | ||||
|   ret.flush = () => { | ||||
|     clearTimeout(handle); | ||||
|     if (lastArgs) { | ||||
|       const _lastArgs = lastArgs; | ||||
|       lastArgs = null; | ||||
|       fn(..._lastArgs); | ||||
|     } | ||||
|   }; | ||||
|   ret.cancel = () => { | ||||
|     lastArgs = null; | ||||
|     clearTimeout(handle); | ||||
|   }; | ||||
|   return ret; | ||||
| }; | ||||
|  | ||||
| export const withBatchedUpdates = < | ||||
|   TFunction extends ((event: any) => void) | (() => void), | ||||
| >( | ||||
|   func: Parameters<TFunction>["length"] extends 0 | 1 ? TFunction : never, | ||||
| ) => | ||||
|   ((event) => { | ||||
|     unstable_batchedUpdates(func as TFunction, event); | ||||
|   }) as TFunction; | ||||
|  | ||||
| /** | ||||
|  * barches React state updates and throttles the calls to a single call per | ||||
|  * animation frame | ||||
|  */ | ||||
| export const withBatchedUpdatesThrottled = < | ||||
|   TFunction extends ((event: any) => void) | (() => void), | ||||
| >( | ||||
|   func: Parameters<TFunction>["length"] extends 0 | 1 ? TFunction : never, | ||||
| ) => { | ||||
|   // @ts-ignore | ||||
|   return throttleRAF<Parameters<TFunction>>(((event) => { | ||||
|     unstable_batchedUpdates(func, event); | ||||
|   }) as TFunction); | ||||
| }; | ||||
							
								
								
									
										5
									
								
								examples/with-script-in-browser/vercel.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| { | ||||
|   "outputDirectory": "dist", | ||||
|   "installCommand": "yarn install", | ||||
|   "buildCommand": "yarn build:package && yarn build" | ||||
| } | ||||
							
								
								
									
										19
									
								
								examples/with-script-in-browser/vite.config.mts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| import { defineConfig } from "vite"; | ||||
|  | ||||
| // https://vitejs.dev/config/ | ||||
| export default defineConfig({ | ||||
|   server: { | ||||
|     port: 3001, | ||||
|     // open the browser | ||||
|     open: true, | ||||
|   }, | ||||
|   publicDir: "public", | ||||
|   optimizeDeps: { | ||||
|     esbuildOptions: { | ||||
|       // Bumping to 2022 due to "Arbitrary module namespace identifier names" not being | ||||
|       // supported in Vite's default browser target https://github.com/vitejs/vite/issues/13556 | ||||
|       target: "es2022", | ||||
|       treeShaking: true, | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
							
								
								
									
										1166
									
								
								excalidraw-app/App.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -1,14 +1,21 @@ | ||||
| import { Stats } from "@excalidraw/excalidraw"; | ||||
| import { copyTextToSystemClipboard } from "@excalidraw/excalidraw/clipboard"; | ||||
| import { | ||||
|   DEFAULT_VERSION, | ||||
|   debounce, | ||||
|   getVersion, | ||||
|   nFormatter, | ||||
| } from "@excalidraw/common"; | ||||
| import { t } from "@excalidraw/excalidraw/i18n"; | ||||
| import { useEffect, useState } from "react"; | ||||
| import { debounce, getVersion, nFormatter } from "../src/utils"; | ||||
|  | ||||
| import type { NonDeletedExcalidrawElement } from "@excalidraw/element/types"; | ||||
| import type { UIAppState } from "@excalidraw/excalidraw/types"; | ||||
|  | ||||
| import { | ||||
|   getElementsStorageSize, | ||||
|   getTotalStorageSize, | ||||
| } from "./data/localStorage"; | ||||
| import { DEFAULT_VERSION } from "../src/constants"; | ||||
| import { t } from "../src/i18n"; | ||||
| import { copyTextToSystemClipboard } from "../src/clipboard"; | ||||
| import { NonDeletedExcalidrawElement } from "../src/element/types"; | ||||
| import { UIAppState } from "../src/types"; | ||||
|  | ||||
| type StorageSizes = { scene: number; total: number }; | ||||
|  | ||||
| @@ -51,39 +58,33 @@ const CustomStats = (props: Props) => { | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <tr> | ||||
|         <th colSpan={2}>{t("stats.storage")}</th> | ||||
|       </tr> | ||||
|       <tr> | ||||
|         <td>{t("stats.scene")}</td> | ||||
|         <td>{nFormatter(storageSizes.scene, 1)}</td> | ||||
|       </tr> | ||||
|       <tr> | ||||
|         <td>{t("stats.total")}</td> | ||||
|         <td>{nFormatter(storageSizes.total, 1)}</td> | ||||
|       </tr> | ||||
|       <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.setToast(t("toast.copyToClipboard")); | ||||
|             } catch {} | ||||
|           }} | ||||
|           title={t("stats.versionCopy")} | ||||
|         > | ||||
|           {timestamp} | ||||
|           <br /> | ||||
|           {hash} | ||||
|         </td> | ||||
|       </tr> | ||||
|     </> | ||||
|     <Stats.StatsRows order={-1}> | ||||
|       <Stats.StatsRow heading>{t("stats.version")}</Stats.StatsRow> | ||||
|       <Stats.StatsRow | ||||
|         style={{ textAlign: "center", cursor: "pointer" }} | ||||
|         onClick={async () => { | ||||
|           try { | ||||
|             await copyTextToSystemClipboard(getVersion()); | ||||
|             props.setToast(t("toast.copyToClipboard")); | ||||
|           } catch {} | ||||
|         }} | ||||
|         title={t("stats.versionCopy")} | ||||
|       > | ||||
|         {timestamp} | ||||
|         <br /> | ||||
|         {hash} | ||||
|       </Stats.StatsRow> | ||||
|  | ||||
|       <Stats.StatsRow heading>{t("stats.storage")}</Stats.StatsRow> | ||||
|       <Stats.StatsRow columns={2}> | ||||
|         <div>{t("stats.scene")}</div> | ||||
|         <div>{nFormatter(storageSizes.scene, 1)}</div> | ||||
|       </Stats.StatsRow> | ||||
|       <Stats.StatsRow columns={2}> | ||||
|         <div>{t("stats.total")}</div> | ||||
|         <div>{nFormatter(storageSizes.total, 1)}</div> | ||||
|       </Stats.StatsRow> | ||||
|     </Stats.StatsRows> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										224
									
								
								excalidraw-app/ExcalidrawPlusIframeExport.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,224 @@ | ||||
| import { base64urlToString } from "@excalidraw/excalidraw/data/encode"; | ||||
| import { ExcalidrawError } from "@excalidraw/excalidraw/errors"; | ||||
| import { useLayoutEffect, useRef } from "react"; | ||||
|  | ||||
| import type { | ||||
|   FileId, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "@excalidraw/element/types"; | ||||
| import type { AppState, BinaryFileData } from "@excalidraw/excalidraw/types"; | ||||
|  | ||||
| import { STORAGE_KEYS } from "./app_constants"; | ||||
| import { LocalData } from "./data/LocalData"; | ||||
|  | ||||
| const EVENT_REQUEST_SCENE = "REQUEST_SCENE"; | ||||
|  | ||||
| const EXCALIDRAW_PLUS_ORIGIN = import.meta.env.VITE_APP_PLUS_APP; | ||||
|  | ||||
| // ----------------------------------------------------------------------------- | ||||
| // outgoing message | ||||
| // ----------------------------------------------------------------------------- | ||||
| type MESSAGE_REQUEST_SCENE = { | ||||
|   type: "REQUEST_SCENE"; | ||||
|   jwt: string; | ||||
| }; | ||||
|  | ||||
| type MESSAGE_FROM_PLUS = MESSAGE_REQUEST_SCENE; | ||||
|  | ||||
| // incoming messages | ||||
| // ----------------------------------------------------------------------------- | ||||
| type MESSAGE_READY = { type: "READY" }; | ||||
| type MESSAGE_ERROR = { type: "ERROR"; message: string }; | ||||
| type MESSAGE_SCENE_DATA = { | ||||
|   type: "SCENE_DATA"; | ||||
|   elements: OrderedExcalidrawElement[]; | ||||
|   appState: Pick<AppState, "viewBackgroundColor">; | ||||
|   files: { loadedFiles: BinaryFileData[]; erroredFiles: Map<FileId, true> }; | ||||
| }; | ||||
|  | ||||
| type MESSAGE_FROM_EDITOR = MESSAGE_ERROR | MESSAGE_SCENE_DATA | MESSAGE_READY; | ||||
| // ----------------------------------------------------------------------------- | ||||
|  | ||||
| const parseSceneData = async ({ | ||||
|   rawElementsString, | ||||
|   rawAppStateString, | ||||
| }: { | ||||
|   rawElementsString: string | null; | ||||
|   rawAppStateString: string | null; | ||||
| }): Promise<MESSAGE_SCENE_DATA> => { | ||||
|   if (!rawElementsString || !rawAppStateString) { | ||||
|     throw new ExcalidrawError("Elements or appstate is missing."); | ||||
|   } | ||||
|  | ||||
|   try { | ||||
|     const elements = JSON.parse( | ||||
|       rawElementsString, | ||||
|     ) as OrderedExcalidrawElement[]; | ||||
|  | ||||
|     if (!elements.length) { | ||||
|       throw new ExcalidrawError("Scene is empty, nothing to export."); | ||||
|     } | ||||
|  | ||||
|     const appState = JSON.parse(rawAppStateString) as Pick< | ||||
|       AppState, | ||||
|       "viewBackgroundColor" | ||||
|     >; | ||||
|  | ||||
|     const fileIds = elements.reduce((acc, el) => { | ||||
|       if ("fileId" in el && el.fileId) { | ||||
|         acc.push(el.fileId); | ||||
|       } | ||||
|       return acc; | ||||
|     }, [] as FileId[]); | ||||
|  | ||||
|     const files = await LocalData.fileStorage.getFiles(fileIds); | ||||
|  | ||||
|     return { | ||||
|       type: "SCENE_DATA", | ||||
|       elements, | ||||
|       appState, | ||||
|       files, | ||||
|     }; | ||||
|   } catch (error: any) { | ||||
|     throw error instanceof ExcalidrawError | ||||
|       ? error | ||||
|       : new ExcalidrawError("Failed to parse scene data."); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const verifyJWT = async ({ | ||||
|   token, | ||||
|   publicKey, | ||||
| }: { | ||||
|   token: string; | ||||
|   publicKey: string; | ||||
| }) => { | ||||
|   try { | ||||
|     if (!publicKey) { | ||||
|       throw new ExcalidrawError("Public key is undefined"); | ||||
|     } | ||||
|  | ||||
|     const [header, payload, signature] = token.split("."); | ||||
|  | ||||
|     if (!header || !payload || !signature) { | ||||
|       throw new ExcalidrawError("Invalid JWT format"); | ||||
|     } | ||||
|  | ||||
|     // JWT is using Base64URL encoding | ||||
|     const decodedPayload = base64urlToString(payload); | ||||
|     const decodedSignature = base64urlToString(signature); | ||||
|  | ||||
|     const data = `${header}.${payload}`; | ||||
|     const signatureArrayBuffer = Uint8Array.from(decodedSignature, (c) => | ||||
|       c.charCodeAt(0), | ||||
|     ); | ||||
|  | ||||
|     const keyData = publicKey.replace(/-----\w+ PUBLIC KEY-----/g, ""); | ||||
|     const keyArrayBuffer = Uint8Array.from(atob(keyData), (c) => | ||||
|       c.charCodeAt(0), | ||||
|     ); | ||||
|  | ||||
|     const key = await crypto.subtle.importKey( | ||||
|       "spki", | ||||
|       keyArrayBuffer, | ||||
|       { name: "RSASSA-PKCS1-v1_5", hash: "SHA-256" }, | ||||
|       true, | ||||
|       ["verify"], | ||||
|     ); | ||||
|  | ||||
|     const isValid = await crypto.subtle.verify( | ||||
|       "RSASSA-PKCS1-v1_5", | ||||
|       key, | ||||
|       signatureArrayBuffer, | ||||
|       new TextEncoder().encode(data), | ||||
|     ); | ||||
|  | ||||
|     if (!isValid) { | ||||
|       throw new Error("Invalid JWT"); | ||||
|     } | ||||
|  | ||||
|     const parsedPayload = JSON.parse(decodedPayload); | ||||
|  | ||||
|     // Check for expiration | ||||
|     const currentTime = Math.floor(Date.now() / 1000); | ||||
|     if (parsedPayload.exp && parsedPayload.exp < currentTime) { | ||||
|       throw new Error("JWT has expired"); | ||||
|     } | ||||
|   } catch (error) { | ||||
|     console.error("Failed to verify JWT:", error); | ||||
|     throw new Error(error instanceof Error ? error.message : "Invalid JWT"); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export const ExcalidrawPlusIframeExport = () => { | ||||
|   const readyRef = useRef(false); | ||||
|  | ||||
|   useLayoutEffect(() => { | ||||
|     const handleMessage = async (event: MessageEvent<MESSAGE_FROM_PLUS>) => { | ||||
|       if (event.origin !== EXCALIDRAW_PLUS_ORIGIN) { | ||||
|         throw new ExcalidrawError("Invalid origin"); | ||||
|       } | ||||
|  | ||||
|       if (event.data.type === EVENT_REQUEST_SCENE) { | ||||
|         if (!event.data.jwt) { | ||||
|           throw new ExcalidrawError("JWT is missing"); | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|           try { | ||||
|             await verifyJWT({ | ||||
|               token: event.data.jwt, | ||||
|               publicKey: import.meta.env.VITE_APP_PLUS_EXPORT_PUBLIC_KEY, | ||||
|             }); | ||||
|           } catch (error: any) { | ||||
|             console.error(`Failed to verify JWT: ${error.message}`); | ||||
|             throw new ExcalidrawError("Failed to verify JWT"); | ||||
|           } | ||||
|  | ||||
|           const parsedSceneData: MESSAGE_SCENE_DATA = await parseSceneData({ | ||||
|             rawAppStateString: localStorage.getItem( | ||||
|               STORAGE_KEYS.LOCAL_STORAGE_APP_STATE, | ||||
|             ), | ||||
|             rawElementsString: localStorage.getItem( | ||||
|               STORAGE_KEYS.LOCAL_STORAGE_ELEMENTS, | ||||
|             ), | ||||
|           }); | ||||
|  | ||||
|           event.source!.postMessage(parsedSceneData, { | ||||
|             targetOrigin: EXCALIDRAW_PLUS_ORIGIN, | ||||
|           }); | ||||
|         } catch (error) { | ||||
|           const responseData: MESSAGE_ERROR = { | ||||
|             type: "ERROR", | ||||
|             message: | ||||
|               error instanceof ExcalidrawError | ||||
|                 ? error.message | ||||
|                 : "Failed to export scene data", | ||||
|           }; | ||||
|           event.source!.postMessage(responseData, { | ||||
|             targetOrigin: EXCALIDRAW_PLUS_ORIGIN, | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
|  | ||||
|     window.addEventListener("message", handleMessage); | ||||
|  | ||||
|     // so we don't send twice in StrictMode | ||||
|     if (!readyRef.current) { | ||||
|       readyRef.current = true; | ||||
|       const message: MESSAGE_FROM_EDITOR = { type: "READY" }; | ||||
|       window.parent.postMessage(message, EXCALIDRAW_PLUS_ORIGIN); | ||||
|     } | ||||
|  | ||||
|     return () => { | ||||
|       window.removeEventListener("message", handleMessage); | ||||
|     }; | ||||
|   }, []); | ||||
|  | ||||
|   // Since this component is expected to run in a hidden iframe on Excaildraw+, | ||||
|   // it doesn't need to render anything. All the data we need is available in | ||||
|   // LocalStorage and IndexedDB. It only needs to handle the messaging between | ||||
|   // the parent window and the iframe with the relevant data. | ||||
|   return null; | ||||
| }; | ||||
| @@ -1,3 +1,37 @@ | ||||
| import { unstable_createStore } from "jotai"; | ||||
| // eslint-disable-next-line no-restricted-imports | ||||
| import { | ||||
|   atom, | ||||
|   Provider, | ||||
|   useAtom, | ||||
|   useAtomValue, | ||||
|   useSetAtom, | ||||
|   createStore, | ||||
|   type PrimitiveAtom, | ||||
| } from "jotai"; | ||||
| import { useLayoutEffect } from "react"; | ||||
|  | ||||
| export const appJotaiStore = unstable_createStore(); | ||||
| export const appJotaiStore = createStore(); | ||||
|  | ||||
| export { atom, Provider, useAtom, useAtomValue, useSetAtom }; | ||||
|  | ||||
| export const useAtomWithInitialValue = < | ||||
|   T extends unknown, | ||||
|   A extends PrimitiveAtom<T>, | ||||
| >( | ||||
|   atom: A, | ||||
|   initialValue: T | (() => T), | ||||
| ) => { | ||||
|   const [value, setValue] = useAtom(atom); | ||||
|  | ||||
|   useLayoutEffect(() => { | ||||
|     if (typeof initialValue === "function") { | ||||
|       // @ts-ignore | ||||
|       setValue(initialValue()); | ||||
|     } else { | ||||
|       setValue(initialValue); | ||||
|     } | ||||
|     // eslint-disable-next-line react-hooks/exhaustive-deps | ||||
|   }, []); | ||||
|  | ||||
|   return [value, setValue] as const; | ||||
| }; | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| import { useSetAtom } from "jotai"; | ||||
| import { useI18n, languages } from "@excalidraw/excalidraw/i18n"; | ||||
| import React from "react"; | ||||
| import { appLangCodeAtom } from ".."; | ||||
| import { useI18n } from "../../src/i18n"; | ||||
| import { languages } from "../../src/i18n"; | ||||
| 
 | ||||
| import { useSetAtom } from "../app-jotai"; | ||||
| 
 | ||||
| import { appLangCodeAtom } from "./language-state"; | ||||
| 
 | ||||
| export const LanguageList = ({ style }: { style?: React.CSSProperties }) => { | ||||
|   const { t, langCode } = useI18n(); | ||||
							
								
								
									
										25
									
								
								excalidraw-app/app-language/language-detector.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | ||||
| import { defaultLang, languages } from "@excalidraw/excalidraw"; | ||||
| import LanguageDetector from "i18next-browser-languagedetector"; | ||||
|  | ||||
| export const languageDetector = new LanguageDetector(); | ||||
|  | ||||
| languageDetector.init({ | ||||
|   languageUtils: {}, | ||||
| }); | ||||
|  | ||||
| export const getPreferredLanguage = () => { | ||||
|   const detectedLanguages = languageDetector.detect(); | ||||
|  | ||||
|   const detectedLanguage = Array.isArray(detectedLanguages) | ||||
|     ? detectedLanguages[0] | ||||
|     : detectedLanguages; | ||||
|  | ||||
|   const initialLanguage = | ||||
|     (detectedLanguage | ||||
|       ? // region code may not be defined if user uses generic preferred language | ||||
|         // (e.g. chinese vs instead of chinese-simplified) | ||||
|         languages.find((lang) => lang.code.startsWith(detectedLanguage))?.code | ||||
|       : null) || defaultLang.code; | ||||
|  | ||||
|   return initialLanguage; | ||||
| }; | ||||
							
								
								
									
										17
									
								
								excalidraw-app/app-language/language-state.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | ||||
| import { useEffect } from "react"; | ||||
|  | ||||
| import { atom, useAtom } from "../app-jotai"; | ||||
|  | ||||
| import { getPreferredLanguage, languageDetector } from "./language-detector"; | ||||
|  | ||||
| export const appLangCodeAtom = atom(getPreferredLanguage()); | ||||
|  | ||||
| export const useAppLangCode = () => { | ||||
|   const [langCode, setLangCode] = useAtom(appLangCodeAtom); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     languageDetector.cacheUserLanguage(langCode); | ||||
|   }, [langCode]); | ||||
|  | ||||
|   return [langCode, setLangCode] as const; | ||||
| }; | ||||
| @@ -15,11 +15,17 @@ export const FILE_CACHE_MAX_AGE_SEC = 31536000; | ||||
| export const WS_EVENTS = { | ||||
|   SERVER_VOLATILE: "server-volatile-broadcast", | ||||
|   SERVER: "server-broadcast", | ||||
| }; | ||||
|   USER_FOLLOW_CHANGE: "user-follow", | ||||
|   USER_FOLLOW_ROOM_CHANGE: "user-follow-room-change", | ||||
| } as const; | ||||
|  | ||||
| export enum WS_SCENE_EVENT_TYPES { | ||||
| export enum WS_SUBTYPES { | ||||
|   INVALID_RESPONSE = "INVALID_RESPONSE", | ||||
|   INIT = "SCENE_INIT", | ||||
|   UPDATE = "SCENE_UPDATE", | ||||
|   MOUSE_LOCATION = "MOUSE_LOCATION", | ||||
|   IDLE_STATUS = "IDLE_STATUS", | ||||
|   USER_VISIBLE_SCENE_BOUNDS = "USER_VISIBLE_SCENE_BOUNDS", | ||||
| } | ||||
|  | ||||
| export const FIREBASE_STORAGE_PREFIXES = { | ||||
| @@ -33,10 +39,15 @@ export const STORAGE_KEYS = { | ||||
|   LOCAL_STORAGE_ELEMENTS: "excalidraw", | ||||
|   LOCAL_STORAGE_APP_STATE: "excalidraw-state", | ||||
|   LOCAL_STORAGE_COLLAB: "excalidraw-collab", | ||||
|   LOCAL_STORAGE_LIBRARY: "excalidraw-library", | ||||
|   LOCAL_STORAGE_THEME: "excalidraw-theme", | ||||
|   LOCAL_STORAGE_DEBUG: "excalidraw-debug", | ||||
|   VERSION_DATA_STATE: "version-dataState", | ||||
|   VERSION_FILES: "version-files", | ||||
|  | ||||
|   IDB_LIBRARY: "excalidraw-library", | ||||
|  | ||||
|   // do not use apart from migrations | ||||
|   __LEGACY_LOCAL_STORAGE_LIBRARY: "excalidraw-library", | ||||
| } as const; | ||||
|  | ||||
| export const COOKIES = { | ||||
|   | ||||
| @@ -1,40 +1,76 @@ | ||||
| import throttle from "lodash.throttle"; | ||||
| import { PureComponent } from "react"; | ||||
| import { ExcalidrawImperativeAPI } from "../../src/types"; | ||||
| import { ErrorDialog } from "../../src/components/ErrorDialog"; | ||||
| import { APP_NAME, ENV, EVENT } from "../../src/constants"; | ||||
| import { ImportedDataState } from "../../src/data/types"; | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   InitializedExcalidrawImageElement, | ||||
| } from "../../src/element/types"; | ||||
| import { | ||||
|   CaptureUpdateAction, | ||||
|   getSceneVersion, | ||||
|   restoreElements, | ||||
| } from "../../src/packages/excalidraw/index"; | ||||
| import { Collaborator, Gesture } from "../../src/types"; | ||||
|   zoomToFitBounds, | ||||
|   reconcileElements, | ||||
| } from "@excalidraw/excalidraw"; | ||||
| import { ErrorDialog } from "@excalidraw/excalidraw/components/ErrorDialog"; | ||||
| import { APP_NAME, EVENT } from "@excalidraw/common"; | ||||
| import { | ||||
|   IDLE_THRESHOLD, | ||||
|   ACTIVE_THRESHOLD, | ||||
|   UserIdleState, | ||||
|   assertNever, | ||||
|   isDevEnv, | ||||
|   isTestEnv, | ||||
|   preventUnload, | ||||
|   resolvablePromise, | ||||
|   withBatchedUpdates, | ||||
| } from "../../src/utils"; | ||||
|   throttleRAF, | ||||
| } from "@excalidraw/common"; | ||||
| import { decryptData } from "@excalidraw/excalidraw/data/encryption"; | ||||
| import { getVisibleSceneBounds } from "@excalidraw/element"; | ||||
| import { newElementWith } from "@excalidraw/element"; | ||||
| import { isImageElement, isInitializedImageElement } from "@excalidraw/element"; | ||||
| import { AbortError } from "@excalidraw/excalidraw/errors"; | ||||
| import { t } from "@excalidraw/excalidraw/i18n"; | ||||
| import { withBatchedUpdates } from "@excalidraw/excalidraw/reactUtils"; | ||||
|  | ||||
| import throttle from "lodash.throttle"; | ||||
| import { PureComponent } from "react"; | ||||
|  | ||||
| import type { | ||||
|   ReconciledExcalidrawElement, | ||||
|   RemoteExcalidrawElement, | ||||
| } from "@excalidraw/excalidraw/data/reconcile"; | ||||
| import type { ImportedDataState } from "@excalidraw/excalidraw/data/types"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
|   InitializedExcalidrawImageElement, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "@excalidraw/element/types"; | ||||
| import type { | ||||
|   BinaryFileData, | ||||
|   ExcalidrawImperativeAPI, | ||||
|   SocketId, | ||||
|   Collaborator, | ||||
|   Gesture, | ||||
| } from "@excalidraw/excalidraw/types"; | ||||
| import type { Mutable, ValueOf } from "@excalidraw/common/utility-types"; | ||||
|  | ||||
| import { appJotaiStore, atom } from "../app-jotai"; | ||||
| import { | ||||
|   CURSOR_SYNC_TIMEOUT, | ||||
|   FILE_UPLOAD_MAX_BYTES, | ||||
|   FIREBASE_STORAGE_PREFIXES, | ||||
|   INITIAL_SCENE_UPDATE_TIMEOUT, | ||||
|   LOAD_IMAGES_TIMEOUT, | ||||
|   WS_SCENE_EVENT_TYPES, | ||||
|   WS_SUBTYPES, | ||||
|   SYNC_FULL_SCENE_INTERVAL_MS, | ||||
|   WS_EVENTS, | ||||
| } from "../app_constants"; | ||||
| import { | ||||
|   generateCollaborationLinkData, | ||||
|   getCollaborationLink, | ||||
|   getCollabServer, | ||||
|   getSyncableElements, | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
| import { | ||||
|   encodeFilesForUpload, | ||||
|   FileManager, | ||||
|   updateStaleImageStatuses, | ||||
| } from "../data/FileManager"; | ||||
| import { LocalData } from "../data/LocalData"; | ||||
| import { | ||||
|   isSavedToFirebase, | ||||
|   loadFilesFromFirebase, | ||||
| @@ -46,43 +82,30 @@ import { | ||||
|   importUsernameFromLocalStorage, | ||||
|   saveUsernameToLocalStorage, | ||||
| } from "../data/localStorage"; | ||||
| import Portal from "./Portal"; | ||||
| import RoomDialog from "./RoomDialog"; | ||||
| import { t } from "../../src/i18n"; | ||||
| import { UserIdleState } from "../../src/types"; | ||||
| import { IDLE_THRESHOLD, ACTIVE_THRESHOLD } from "../../src/constants"; | ||||
| import { | ||||
|   encodeFilesForUpload, | ||||
|   FileManager, | ||||
|   updateStaleImageStatuses, | ||||
| } from "../data/FileManager"; | ||||
| import { AbortError } from "../../src/errors"; | ||||
| import { | ||||
|   isImageElement, | ||||
|   isInitializedImageElement, | ||||
| } from "../../src/element/typeChecks"; | ||||
| import { newElementWith } from "../../src/element/mutateElement"; | ||||
| import { | ||||
|   ReconciledElements, | ||||
|   reconcileElements as _reconcileElements, | ||||
| } from "./reconciliation"; | ||||
| import { decryptData } from "../../src/data/encryption"; | ||||
| import { resetBrowserStateVersions } from "../data/tabSync"; | ||||
| import { LocalData } from "../data/LocalData"; | ||||
| import { atom, useAtom } from "jotai"; | ||||
| import { appJotaiStore } from "../app-jotai"; | ||||
|  | ||||
| import { collabErrorIndicatorAtom } from "./CollabError"; | ||||
| import Portal from "./Portal"; | ||||
|  | ||||
| import type { | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
|  | ||||
| export const collabAPIAtom = atom<CollabAPI | null>(null); | ||||
| export const collabDialogShownAtom = atom(false); | ||||
| export const isCollaboratingAtom = atom(false); | ||||
| export const isOfflineAtom = atom(false); | ||||
|  | ||||
| interface CollabState { | ||||
|   errorMessage: string; | ||||
|   errorMessage: string | null; | ||||
|   /** errors related to saving */ | ||||
|   dialogNotifiedErrors: Record<string, boolean>; | ||||
|   username: string; | ||||
|   activeRoomLink: string; | ||||
|   activeRoomLink: string | null; | ||||
| } | ||||
|  | ||||
| export const activeRoomLinkAtom = atom<string | null>(null); | ||||
|  | ||||
| type CollabInstance = InstanceType<typeof Collab>; | ||||
|  | ||||
| export interface CollabAPI { | ||||
| @@ -93,32 +116,34 @@ export interface CollabAPI { | ||||
|   stopCollaboration: CollabInstance["stopCollaboration"]; | ||||
|   syncElements: CollabInstance["syncElements"]; | ||||
|   fetchImageFilesFromFirebase: CollabInstance["fetchImageFilesFromFirebase"]; | ||||
|   setUsername: (username: string) => void; | ||||
|   setUsername: CollabInstance["setUsername"]; | ||||
|   getUsername: CollabInstance["getUsername"]; | ||||
|   getActiveRoomLink: CollabInstance["getActiveRoomLink"]; | ||||
|   setCollabError: CollabInstance["setErrorDialog"]; | ||||
| } | ||||
|  | ||||
| interface PublicProps { | ||||
| interface CollabProps { | ||||
|   excalidrawAPI: ExcalidrawImperativeAPI; | ||||
| } | ||||
|  | ||||
| type Props = PublicProps & { modalIsShown: boolean }; | ||||
|  | ||||
| class Collab extends PureComponent<Props, CollabState> { | ||||
| class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|   portal: Portal; | ||||
|   fileManager: FileManager; | ||||
|   excalidrawAPI: Props["excalidrawAPI"]; | ||||
|   excalidrawAPI: CollabProps["excalidrawAPI"]; | ||||
|   activeIntervalId: number | null; | ||||
|   idleTimeoutId: number | null; | ||||
|  | ||||
|   private socketInitializationTimer?: number; | ||||
|   private lastBroadcastedOrReceivedSceneVersion: number = -1; | ||||
|   private collaborators = new Map<string, Collaborator>(); | ||||
|   private collaborators = new Map<SocketId, Collaborator>(); | ||||
|  | ||||
|   constructor(props: Props) { | ||||
|   constructor(props: CollabProps) { | ||||
|     super(props); | ||||
|     this.state = { | ||||
|       errorMessage: "", | ||||
|       errorMessage: null, | ||||
|       dialogNotifiedErrors: {}, | ||||
|       username: importUsernameFromLocalStorage() || "", | ||||
|       activeRoomLink: "", | ||||
|       activeRoomLink: null, | ||||
|     }; | ||||
|     this.portal = new Portal(this); | ||||
|     this.fileManager = new FileManager({ | ||||
| @@ -136,7 +161,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|           throw new AbortError(); | ||||
|         } | ||||
|  | ||||
|         return saveFilesToFirebase({ | ||||
|         const { savedFiles, erroredFiles } = await saveFilesToFirebase({ | ||||
|           prefix: `${FIREBASE_STORAGE_PREFIXES.collabFiles}/${roomId}`, | ||||
|           files: await encodeFilesForUpload({ | ||||
|             files: addedFiles, | ||||
| @@ -144,6 +169,29 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|             maxBytes: FILE_UPLOAD_MAX_BYTES, | ||||
|           }), | ||||
|         }); | ||||
|  | ||||
|         return { | ||||
|           savedFiles: savedFiles.reduce( | ||||
|             (acc: Map<FileId, BinaryFileData>, id) => { | ||||
|               const fileData = addedFiles.get(id); | ||||
|               if (fileData) { | ||||
|                 acc.set(id, fileData); | ||||
|               } | ||||
|               return acc; | ||||
|             }, | ||||
|             new Map(), | ||||
|           ), | ||||
|           erroredFiles: erroredFiles.reduce( | ||||
|             (acc: Map<FileId, BinaryFileData>, id) => { | ||||
|               const fileData = addedFiles.get(id); | ||||
|               if (fileData) { | ||||
|                 acc.set(id, fileData); | ||||
|               } | ||||
|               return acc; | ||||
|             }, | ||||
|             new Map(), | ||||
|           ), | ||||
|         }; | ||||
|       }, | ||||
|     }); | ||||
|     this.excalidrawAPI = props.excalidrawAPI; | ||||
| @@ -151,12 +199,28 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     this.idleTimeoutId = null; | ||||
|   } | ||||
|  | ||||
|   private onUmmount: (() => void) | null = null; | ||||
|  | ||||
|   componentDidMount() { | ||||
|     window.addEventListener(EVENT.BEFORE_UNLOAD, this.beforeUnload); | ||||
|     window.addEventListener("online", this.onOfflineStatusToggle); | ||||
|     window.addEventListener("offline", this.onOfflineStatusToggle); | ||||
|     window.addEventListener(EVENT.UNLOAD, this.onUnload); | ||||
|  | ||||
|     const unsubOnUserFollow = this.excalidrawAPI.onUserFollow((payload) => { | ||||
|       this.portal.socket && this.portal.broadcastUserFollowed(payload); | ||||
|     }); | ||||
|     const throttledRelayUserViewportBounds = throttleRAF( | ||||
|       this.relayVisibleSceneBounds, | ||||
|     ); | ||||
|     const unsubOnScrollChange = this.excalidrawAPI.onScrollChange(() => | ||||
|       throttledRelayUserViewportBounds(), | ||||
|     ); | ||||
|     this.onUmmount = () => { | ||||
|       unsubOnUserFollow(); | ||||
|       unsubOnScrollChange(); | ||||
|     }; | ||||
|  | ||||
|     this.onOfflineStatusToggle(); | ||||
|  | ||||
|     const collabAPI: CollabAPI = { | ||||
| @@ -167,11 +231,14 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       fetchImageFilesFromFirebase: this.fetchImageFilesFromFirebase, | ||||
|       stopCollaboration: this.stopCollaboration, | ||||
|       setUsername: this.setUsername, | ||||
|       getUsername: this.getUsername, | ||||
|       getActiveRoomLink: this.getActiveRoomLink, | ||||
|       setCollabError: this.setErrorDialog, | ||||
|     }; | ||||
|  | ||||
|     appJotaiStore.set(collabAPIAtom, collabAPI); | ||||
|  | ||||
|     if (import.meta.env.MODE === ENV.TEST || import.meta.env.DEV) { | ||||
|     if (isTestEnv() || isDevEnv()) { | ||||
|       window.collab = window.collab || ({} as Window["collab"]); | ||||
|       Object.defineProperties(window, { | ||||
|         collab: { | ||||
| @@ -204,6 +271,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       window.clearTimeout(this.idleTimeoutId); | ||||
|       this.idleTimeoutId = null; | ||||
|     } | ||||
|     this.onUmmount?.(); | ||||
|   } | ||||
|  | ||||
|   isCollaborating = () => appJotaiStore.get(isCollaboratingAtom)!; | ||||
| @@ -230,7 +298,13 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       //  the purpose is to run in immediately after user decides to stay | ||||
|       this.saveCollabRoomToFirebase(syncableElements); | ||||
|  | ||||
|       preventUnload(event); | ||||
|       if (import.meta.env.VITE_APP_DISABLE_PREVENT_UNLOAD !== "true") { | ||||
|         preventUnload(event); | ||||
|       } else { | ||||
|         console.warn( | ||||
|           "preventing unload disabled (VITE_APP_DISABLE_PREVENT_UNLOAD)", | ||||
|         ); | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
|  | ||||
| @@ -238,24 +312,39 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     syncableElements: readonly SyncableExcalidrawElement[], | ||||
|   ) => { | ||||
|     try { | ||||
|       const savedData = await saveToFirebase( | ||||
|       const storedElements = await saveToFirebase( | ||||
|         this.portal, | ||||
|         syncableElements, | ||||
|         this.excalidrawAPI.getAppState(), | ||||
|       ); | ||||
|  | ||||
|       if (this.isCollaborating() && savedData && savedData.reconciledElements) { | ||||
|         this.handleRemoteSceneUpdate( | ||||
|           this.reconcileElements(savedData.reconciledElements), | ||||
|         ); | ||||
|       this.resetErrorIndicator(); | ||||
|  | ||||
|       if (this.isCollaborating() && storedElements) { | ||||
|         this.handleRemoteSceneUpdate(this._reconcileElements(storedElements)); | ||||
|       } | ||||
|     } catch (error: any) { | ||||
|       this.setState({ | ||||
|         // firestore doesn't return a specific error code when size exceeded | ||||
|         errorMessage: /is longer than.*?bytes/.test(error.message) | ||||
|           ? t("errors.collabSaveFailed_sizeExceeded") | ||||
|           : t("errors.collabSaveFailed"), | ||||
|       }); | ||||
|       const errorMessage = /is longer than.*?bytes/.test(error.message) | ||||
|         ? t("errors.collabSaveFailed_sizeExceeded") | ||||
|         : t("errors.collabSaveFailed"); | ||||
|  | ||||
|       if ( | ||||
|         !this.state.dialogNotifiedErrors[errorMessage] || | ||||
|         !this.isCollaborating() | ||||
|       ) { | ||||
|         this.setErrorDialog(errorMessage); | ||||
|         this.setState({ | ||||
|           dialogNotifiedErrors: { | ||||
|             ...this.state.dialogNotifiedErrors, | ||||
|             [errorMessage]: true, | ||||
|           }, | ||||
|         }); | ||||
|       } | ||||
|  | ||||
|       if (this.isCollaborating()) { | ||||
|         this.setErrorIndicator(errorMessage); | ||||
|       } | ||||
|  | ||||
|       console.error(error); | ||||
|     } | ||||
|   }; | ||||
| @@ -264,6 +353,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     this.queueBroadcastAllElements.cancel(); | ||||
|     this.queueSaveToFirebase.cancel(); | ||||
|     this.loadImageFiles.cancel(); | ||||
|     this.resetErrorIndicator(true); | ||||
|  | ||||
|     this.saveCollabRoomToFirebase( | ||||
|       getSyncableElements( | ||||
| @@ -302,7 +392,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|  | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         elements, | ||||
|         commitToHistory: false, | ||||
|         captureUpdate: CaptureUpdateAction.NEVER, | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
| @@ -313,9 +403,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     this.fileManager.reset(); | ||||
|     if (!opts?.isUnload) { | ||||
|       this.setIsCollaborating(false); | ||||
|       this.setState({ | ||||
|         activeRoomLink: "", | ||||
|       }); | ||||
|       this.setActiveRoomLink(null); | ||||
|       this.collaborators = new Map(); | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         collaborators: this.collaborators, | ||||
| @@ -339,7 +427,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       .filter((element) => { | ||||
|         return ( | ||||
|           isInitializedImageElement(element) && | ||||
|           !this.fileManager.isFileHandled(element.fileId) && | ||||
|           !this.fileManager.isFileTracked(element.fileId) && | ||||
|           !element.isDeleted && | ||||
|           (opts.forceFetchFiles | ||||
|             ? element.status !== "pending" || | ||||
| @@ -356,7 +444,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     iv: Uint8Array, | ||||
|     encryptedData: ArrayBuffer, | ||||
|     decryptionKey: string, | ||||
|   ) => { | ||||
|   ): Promise<ValueOf<SocketUpdateDataSource>> => { | ||||
|     try { | ||||
|       const decrypted = await decryptData(iv, encryptedData, decryptionKey); | ||||
|  | ||||
| @@ -368,7 +456,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       window.alert(t("alerts.decryptFailed")); | ||||
|       console.error(error); | ||||
|       return { | ||||
|         type: "INVALID_RESPONSE", | ||||
|         type: WS_SUBTYPES.INVALID_RESPONSE, | ||||
|       }; | ||||
|     } | ||||
|   }; | ||||
| @@ -377,11 +465,11 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|  | ||||
|   startCollaboration = async ( | ||||
|     existingRoomLinkData: null | { roomId: string; roomKey: string }, | ||||
|   ): Promise<ImportedDataState | null> => { | ||||
|   ) => { | ||||
|     if (!this.state.username) { | ||||
|       import("@excalidraw/random-username").then(({ getRandomUsername }) => { | ||||
|         const username = getRandomUsername(); | ||||
|         this.onUsernameChange(username); | ||||
|         this.setUsername(username); | ||||
|       }); | ||||
|     } | ||||
|  | ||||
| @@ -403,7 +491,11 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     const scenePromise = resolvablePromise<ImportedDataState | null>(); | ||||
|     // TODO: `ImportedDataState` type here seems abused | ||||
|     const scenePromise = resolvablePromise< | ||||
|       | (ImportedDataState & { elements: readonly OrderedExcalidrawElement[] }) | ||||
|       | null | ||||
|     >(); | ||||
|  | ||||
|     this.setIsCollaborating(true); | ||||
|     LocalData.pauseSave("collaboration"); | ||||
| @@ -423,13 +515,9 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     this.fallbackInitializationHandler = fallbackInitializationHandler; | ||||
|  | ||||
|     try { | ||||
|       const socketServerData = await getCollabServer(); | ||||
|  | ||||
|       this.portal.socket = this.portal.open( | ||||
|         socketIOClient(socketServerData.url, { | ||||
|           transports: socketServerData.polling | ||||
|             ? ["websocket", "polling"] | ||||
|             : ["websocket"], | ||||
|         socketIOClient(import.meta.env.VITE_APP_WS_SERVER_URL, { | ||||
|           transports: ["websocket", "polling"], | ||||
|         }), | ||||
|         roomId, | ||||
|         roomKey, | ||||
| @@ -438,7 +526,7 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       this.portal.socket.once("connect_error", fallbackInitializationHandler); | ||||
|     } catch (error: any) { | ||||
|       console.error(error); | ||||
|       this.setState({ errorMessage: error.message }); | ||||
|       this.setErrorDialog(error.message); | ||||
|       return null; | ||||
|     } | ||||
|  | ||||
| @@ -449,14 +537,13 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|         } | ||||
|         return element; | ||||
|       }); | ||||
|       // remove deleted elements from elements array & history to ensure we don't | ||||
|       // remove deleted elements from elements array to ensure we don't | ||||
|       // expose potentially sensitive user data in case user manually deletes | ||||
|       // existing elements (or clears scene), which would otherwise be persisted | ||||
|       // to database even if deleted before creating the room. | ||||
|       this.excalidrawAPI.history.clear(); | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         elements, | ||||
|         commitToHistory: true, | ||||
|         captureUpdate: CaptureUpdateAction.NEVER, | ||||
|       }); | ||||
|  | ||||
|       this.saveCollabRoomToFirebase(getSyncableElements(elements)); | ||||
| @@ -484,16 +571,15 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|         ); | ||||
|  | ||||
|         switch (decryptedData.type) { | ||||
|           case "INVALID_RESPONSE": | ||||
|           case WS_SUBTYPES.INVALID_RESPONSE: | ||||
|             return; | ||||
|           case WS_SCENE_EVENT_TYPES.INIT: { | ||||
|           case WS_SUBTYPES.INIT: { | ||||
|             if (!this.portal.socketInitialized) { | ||||
|               this.initializeRoom({ fetchScene: false }); | ||||
|               const remoteElements = decryptedData.payload.elements; | ||||
|               const reconciledElements = this.reconcileElements(remoteElements); | ||||
|               this.handleRemoteSceneUpdate(reconciledElements, { | ||||
|                 init: true, | ||||
|               }); | ||||
|               const reconciledElements = | ||||
|                 this._reconcileElements(remoteElements); | ||||
|               this.handleRemoteSceneUpdate(reconciledElements); | ||||
|               // noop if already resolved via init from firebase | ||||
|               scenePromise.resolve({ | ||||
|                 elements: reconciledElements, | ||||
| @@ -502,41 +588,75 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|             } | ||||
|             break; | ||||
|           } | ||||
|           case WS_SCENE_EVENT_TYPES.UPDATE: | ||||
|           case WS_SUBTYPES.UPDATE: | ||||
|             this.handleRemoteSceneUpdate( | ||||
|               this.reconcileElements(decryptedData.payload.elements), | ||||
|               this._reconcileElements(decryptedData.payload.elements), | ||||
|             ); | ||||
|             break; | ||||
|           case "MOUSE_LOCATION": { | ||||
|           case WS_SUBTYPES.MOUSE_LOCATION: { | ||||
|             const { pointer, button, username, selectedElementIds } = | ||||
|               decryptedData.payload; | ||||
|  | ||||
|             const socketId: SocketUpdateDataSource["MOUSE_LOCATION"]["payload"]["socketId"] = | ||||
|               decryptedData.payload.socketId || | ||||
|               // @ts-ignore legacy, see #2094 (#2097) | ||||
|               decryptedData.payload.socketID; | ||||
|  | ||||
|             const collaborators = new Map(this.collaborators); | ||||
|             const user = collaborators.get(socketId) || {}!; | ||||
|             user.pointer = pointer; | ||||
|             user.button = button; | ||||
|             user.selectedElementIds = selectedElementIds; | ||||
|             user.username = username; | ||||
|             collaborators.set(socketId, user); | ||||
|             this.updateCollaborator(socketId, { | ||||
|               pointer, | ||||
|               button, | ||||
|               selectedElementIds, | ||||
|               username, | ||||
|             }); | ||||
|  | ||||
|             break; | ||||
|           } | ||||
|  | ||||
|           case WS_SUBTYPES.USER_VISIBLE_SCENE_BOUNDS: { | ||||
|             const { sceneBounds, socketId } = decryptedData.payload; | ||||
|  | ||||
|             const appState = this.excalidrawAPI.getAppState(); | ||||
|  | ||||
|             // we're not following the user | ||||
|             // (shouldn't happen, but could be late message or bug upstream) | ||||
|             if (appState.userToFollow?.socketId !== socketId) { | ||||
|               console.warn( | ||||
|                 `receiving remote client's (from ${socketId}) viewport bounds even though we're not subscribed to it!`, | ||||
|               ); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             // cross-follow case, ignore updates in this case | ||||
|             if ( | ||||
|               appState.userToFollow && | ||||
|               appState.followedBy.has(appState.userToFollow.socketId) | ||||
|             ) { | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             this.excalidrawAPI.updateScene({ | ||||
|               collaborators, | ||||
|               appState: zoomToFitBounds({ | ||||
|                 appState, | ||||
|                 bounds: sceneBounds, | ||||
|                 fitToViewport: true, | ||||
|                 viewportZoomFactor: 1, | ||||
|               }).appState, | ||||
|             }); | ||||
|  | ||||
|             break; | ||||
|           } | ||||
|  | ||||
|           case WS_SUBTYPES.IDLE_STATUS: { | ||||
|             const { userState, socketId, username } = decryptedData.payload; | ||||
|             this.updateCollaborator(socketId, { | ||||
|               userState, | ||||
|               username, | ||||
|             }); | ||||
|             break; | ||||
|           } | ||||
|           case "IDLE_STATUS": { | ||||
|             const { userState, socketId, username } = decryptedData.payload; | ||||
|             const collaborators = new Map(this.collaborators); | ||||
|             const user = collaborators.get(socketId) || {}!; | ||||
|             user.userState = userState; | ||||
|             user.username = username; | ||||
|             this.excalidrawAPI.updateScene({ | ||||
|               collaborators, | ||||
|             }); | ||||
|             break; | ||||
|  | ||||
|           default: { | ||||
|             assertNever(decryptedData, null); | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
| @@ -553,11 +673,20 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|       scenePromise.resolve(sceneData); | ||||
|     }); | ||||
|  | ||||
|     this.portal.socket.on( | ||||
|       WS_EVENTS.USER_FOLLOW_ROOM_CHANGE, | ||||
|       (followedBy: SocketId[]) => { | ||||
|         this.excalidrawAPI.updateScene({ | ||||
|           appState: { followedBy: new Set(followedBy) }, | ||||
|         }); | ||||
|  | ||||
|         this.relayVisibleSceneBounds({ force: true }); | ||||
|       }, | ||||
|     ); | ||||
|  | ||||
|     this.initializeIdleDetector(); | ||||
|  | ||||
|     this.setState({ | ||||
|       activeRoomLink: window.location.href, | ||||
|     }); | ||||
|     this.setActiveRoomLink(window.location.href); | ||||
|  | ||||
|     return scenePromise; | ||||
|   }; | ||||
| @@ -609,17 +738,15 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     return null; | ||||
|   }; | ||||
|  | ||||
|   private reconcileElements = ( | ||||
|   private _reconcileElements = ( | ||||
|     remoteElements: readonly ExcalidrawElement[], | ||||
|   ): ReconciledElements => { | ||||
|   ): ReconciledExcalidrawElement[] => { | ||||
|     const localElements = this.getSceneElementsIncludingDeleted(); | ||||
|     const appState = this.excalidrawAPI.getAppState(); | ||||
|  | ||||
|     remoteElements = restoreElements(remoteElements, null); | ||||
|  | ||||
|     const reconciledElements = _reconcileElements( | ||||
|     const restoredRemoteElements = restoreElements(remoteElements, null); | ||||
|     const reconciledElements = reconcileElements( | ||||
|       localElements, | ||||
|       remoteElements, | ||||
|       restoredRemoteElements as RemoteExcalidrawElement[], | ||||
|       appState, | ||||
|     ); | ||||
|  | ||||
| @@ -650,20 +777,13 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|   }, LOAD_IMAGES_TIMEOUT); | ||||
|  | ||||
|   private handleRemoteSceneUpdate = ( | ||||
|     elements: ReconciledElements, | ||||
|     { init = false }: { init?: boolean } = {}, | ||||
|     elements: ReconciledExcalidrawElement[], | ||||
|   ) => { | ||||
|     this.excalidrawAPI.updateScene({ | ||||
|       elements, | ||||
|       commitToHistory: !!init, | ||||
|       captureUpdate: CaptureUpdateAction.NEVER, | ||||
|     }); | ||||
|  | ||||
|     // We haven't yet implemented multiplayer undo functionality, so we clear the undo stack | ||||
|     // when we receive any messages from another peer. This UX can be pretty rough -- if you | ||||
|     // undo, a user makes a change, and then try to redo, your element(s) will be lost. However, | ||||
|     // right now we think this is the right tradeoff. | ||||
|     this.excalidrawAPI.history.clear(); | ||||
|  | ||||
|     this.loadImageFiles(); | ||||
|   }; | ||||
|  | ||||
| @@ -721,20 +841,39 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     document.addEventListener(EVENT.VISIBILITY_CHANGE, this.onVisibilityChange); | ||||
|   }; | ||||
|  | ||||
|   setCollaborators(sockets: string[]) { | ||||
|   setCollaborators(sockets: SocketId[]) { | ||||
|     const collaborators: InstanceType<typeof Collab>["collaborators"] = | ||||
|       new Map(); | ||||
|     for (const socketId of sockets) { | ||||
|       if (this.collaborators.has(socketId)) { | ||||
|         collaborators.set(socketId, this.collaborators.get(socketId)!); | ||||
|       } else { | ||||
|         collaborators.set(socketId, {}); | ||||
|       } | ||||
|       collaborators.set( | ||||
|         socketId, | ||||
|         Object.assign({}, this.collaborators.get(socketId), { | ||||
|           isCurrentUser: socketId === this.portal.socket?.id, | ||||
|         }), | ||||
|       ); | ||||
|     } | ||||
|     this.collaborators = collaborators; | ||||
|     this.excalidrawAPI.updateScene({ collaborators }); | ||||
|   } | ||||
|  | ||||
|   updateCollaborator = (socketId: SocketId, updates: Partial<Collaborator>) => { | ||||
|     const collaborators = new Map(this.collaborators); | ||||
|     const user: Mutable<Collaborator> = Object.assign( | ||||
|       {}, | ||||
|       collaborators.get(socketId), | ||||
|       updates, | ||||
|       { | ||||
|         isCurrentUser: socketId === this.portal.socket?.id, | ||||
|       }, | ||||
|     ); | ||||
|     collaborators.set(socketId, user); | ||||
|     this.collaborators = collaborators; | ||||
|  | ||||
|     this.excalidrawAPI.updateScene({ | ||||
|       collaborators, | ||||
|     }); | ||||
|   }; | ||||
|  | ||||
|   public setLastBroadcastedOrReceivedSceneVersion = (version: number) => { | ||||
|     this.lastBroadcastedOrReceivedSceneVersion = version; | ||||
|   }; | ||||
| @@ -760,29 +899,42 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     CURSOR_SYNC_TIMEOUT, | ||||
|   ); | ||||
|  | ||||
|   relayVisibleSceneBounds = (props?: { force: boolean }) => { | ||||
|     const appState = this.excalidrawAPI.getAppState(); | ||||
|  | ||||
|     if (this.portal.socket && (appState.followedBy.size > 0 || props?.force)) { | ||||
|       this.portal.broadcastVisibleSceneBounds( | ||||
|         { | ||||
|           sceneBounds: getVisibleSceneBounds(appState), | ||||
|         }, | ||||
|         `follow@${this.portal.socket.id}`, | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   onIdleStateChange = (userState: UserIdleState) => { | ||||
|     this.portal.broadcastIdleChange(userState); | ||||
|   }; | ||||
|  | ||||
|   broadcastElements = (elements: readonly ExcalidrawElement[]) => { | ||||
|   broadcastElements = (elements: readonly OrderedExcalidrawElement[]) => { | ||||
|     if ( | ||||
|       getSceneVersion(elements) > | ||||
|       this.getLastBroadcastedOrReceivedSceneVersion() | ||||
|     ) { | ||||
|       this.portal.broadcastScene(WS_SCENE_EVENT_TYPES.UPDATE, elements, false); | ||||
|       this.portal.broadcastScene(WS_SUBTYPES.UPDATE, elements, false); | ||||
|       this.lastBroadcastedOrReceivedSceneVersion = getSceneVersion(elements); | ||||
|       this.queueBroadcastAllElements(); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   syncElements = (elements: readonly ExcalidrawElement[]) => { | ||||
|   syncElements = (elements: readonly OrderedExcalidrawElement[]) => { | ||||
|     this.broadcastElements(elements); | ||||
|     this.queueSaveToFirebase(); | ||||
|   }; | ||||
|  | ||||
|   queueBroadcastAllElements = throttle(() => { | ||||
|     this.portal.broadcastScene( | ||||
|       WS_SCENE_EVENT_TYPES.UPDATE, | ||||
|       WS_SUBTYPES.UPDATE, | ||||
|       this.excalidrawAPI.getSceneElementsIncludingDeleted(), | ||||
|       true, | ||||
|     ); | ||||
| @@ -808,41 +960,49 @@ class Collab extends PureComponent<Props, CollabState> { | ||||
|     { leading: false }, | ||||
|   ); | ||||
|  | ||||
|   handleClose = () => { | ||||
|     appJotaiStore.set(collabDialogShownAtom, false); | ||||
|   }; | ||||
|  | ||||
|   setUsername = (username: string) => { | ||||
|     this.setState({ username }); | ||||
|   }; | ||||
|  | ||||
|   onUsernameChange = (username: string) => { | ||||
|     this.setUsername(username); | ||||
|     saveUsernameToLocalStorage(username); | ||||
|   }; | ||||
|  | ||||
|   render() { | ||||
|     const { username, errorMessage, activeRoomLink } = this.state; | ||||
|   getUsername = () => this.state.username; | ||||
|  | ||||
|     const { modalIsShown } = this.props; | ||||
|   setActiveRoomLink = (activeRoomLink: string | null) => { | ||||
|     this.setState({ activeRoomLink }); | ||||
|     appJotaiStore.set(activeRoomLinkAtom, activeRoomLink); | ||||
|   }; | ||||
|  | ||||
|   getActiveRoomLink = () => this.state.activeRoomLink; | ||||
|  | ||||
|   setErrorIndicator = (errorMessage: string | null) => { | ||||
|     appJotaiStore.set(collabErrorIndicatorAtom, { | ||||
|       message: errorMessage, | ||||
|       nonce: Date.now(), | ||||
|     }); | ||||
|   }; | ||||
|  | ||||
|   resetErrorIndicator = (resetDialogNotifiedErrors = false) => { | ||||
|     appJotaiStore.set(collabErrorIndicatorAtom, { message: null, nonce: 0 }); | ||||
|     if (resetDialogNotifiedErrors) { | ||||
|       this.setState({ | ||||
|         dialogNotifiedErrors: {}, | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   setErrorDialog = (errorMessage: string | null) => { | ||||
|     this.setState({ | ||||
|       errorMessage, | ||||
|     }); | ||||
|   }; | ||||
|  | ||||
|   render() { | ||||
|     const { errorMessage } = this.state; | ||||
|  | ||||
|     return ( | ||||
|       <> | ||||
|         {modalIsShown && ( | ||||
|           <RoomDialog | ||||
|             handleClose={this.handleClose} | ||||
|             activeRoomLink={activeRoomLink} | ||||
|             username={username} | ||||
|             onUsernameChange={this.onUsernameChange} | ||||
|             onRoomCreate={() => this.startCollaboration(null)} | ||||
|             onRoomDestroy={this.stopCollaboration} | ||||
|             setErrorMessage={(errorMessage) => { | ||||
|               this.setState({ errorMessage }); | ||||
|             }} | ||||
|           /> | ||||
|         )} | ||||
|         {errorMessage && ( | ||||
|           <ErrorDialog onClose={() => this.setState({ errorMessage: "" })}> | ||||
|         {errorMessage != null && ( | ||||
|           <ErrorDialog onClose={() => this.setErrorDialog(null)}> | ||||
|             {errorMessage} | ||||
|           </ErrorDialog> | ||||
|         )} | ||||
| @@ -857,15 +1017,10 @@ declare global { | ||||
|   } | ||||
| } | ||||
|  | ||||
| if (import.meta.env.MODE === ENV.TEST || import.meta.env.DEV) { | ||||
| if (isTestEnv() || isDevEnv()) { | ||||
|   window.collab = window.collab || ({} as Window["collab"]); | ||||
| } | ||||
|  | ||||
| const _Collab: React.FC<PublicProps> = (props) => { | ||||
|   const [collabDialogShown] = useAtom(collabDialogShownAtom); | ||||
|   return <Collab {...props} modalIsShown={collabDialogShown} />; | ||||
| }; | ||||
|  | ||||
| export default _Collab; | ||||
| export default Collab; | ||||
|  | ||||
| export type TCollabClass = Collab; | ||||
|   | ||||
							
								
								
									
										35
									
								
								excalidraw-app/collab/CollabError.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | ||||
| @import "../../packages/excalidraw/css/variables.module.scss"; | ||||
|  | ||||
| .excalidraw { | ||||
|   .collab-errors-button { | ||||
|     width: 26px; | ||||
|     height: 26px; | ||||
|     margin-inline-end: 1rem; | ||||
|  | ||||
|     color: var(--color-danger); | ||||
|  | ||||
|     flex-shrink: 0; | ||||
|   } | ||||
|  | ||||
|   .collab-errors-button-shake { | ||||
|     animation: strong-shake 0.15s 6; | ||||
|   } | ||||
|  | ||||
|   @keyframes strong-shake { | ||||
|     0% { | ||||
|       transform: rotate(0deg); | ||||
|     } | ||||
|     25% { | ||||
|       transform: rotate(10deg); | ||||
|     } | ||||
|     50% { | ||||
|       transform: rotate(0deg); | ||||
|     } | ||||
|     75% { | ||||
|       transform: rotate(-10deg); | ||||
|     } | ||||
|     100% { | ||||
|       transform: rotate(0deg); | ||||
|     } | ||||
|   } | ||||
| } | ||||