mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-26 16:34:22 +01:00 
			
		
		
		
	Compare commits
	
		
			4 Commits
		
	
	
		
			dwelle/fix
			...
			frame-resi
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ae88ea555c | ||
|   | 14eecf651f | ||
|   | c951a001f7 | ||
|   | 25ab75cb9b | 
| @@ -4,16 +4,8 @@ | ||||
| !.eslintrc.json | ||||
| !.npmrc | ||||
| !.prettierrc | ||||
| !excalidraw-app/ | ||||
| !package.json | ||||
| !public/ | ||||
| !packages/ | ||||
| !scripts/ | ||||
| !src/ | ||||
| !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 | ||||
|   | ||||
| @@ -7,20 +7,23 @@ 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 | ||||
|  | ||||
| VITE_APP_PLUS_LP=https://plus.excalidraw.com | ||||
| VITE_APP_PLUS_APP=http://localhost:3000 | ||||
| # set this only if using the collaboration workflow we use on excalidraw.com | ||||
| VITE_APP_PORTAL_URL= | ||||
|  | ||||
| VITE_APP_AI_BACKEND=http://localhost:3015 | ||||
| VITE_APP_PLUS_LP=https://plus.excalidraw.com | ||||
| VITE_APP_PLUS_APP=https://app.excalidraw.com | ||||
|  | ||||
| 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_ENABLE_TRACKING=true | ||||
| VITE_APP_DISABLE_TRACKING=true | ||||
|  | ||||
| FAST_REFRESH=false | ||||
|  | ||||
| @@ -37,14 +40,3 @@ 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' | ||||
|   | ||||
| @@ -4,29 +4,15 @@ 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 | ||||
|  | ||||
| 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 | ||||
| # 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_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_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 | ||||
|  | ||||
| VITE_APP_DISABLE_TRACKING= | ||||
|   | ||||
| @@ -5,7 +5,4 @@ package-lock.json | ||||
| firebase/ | ||||
| dist/ | ||||
| public/workbox | ||||
| packages/excalidraw/types | ||||
| examples/**/public | ||||
| dev-dist | ||||
| coverage | ||||
| src/packages/excalidraw/types | ||||
|   | ||||
| @@ -2,7 +2,6 @@ | ||||
|   "extends": ["@excalidraw/eslint-config", "react-app"], | ||||
|   "rules": { | ||||
|     "import/no-anonymous-default-export": "off", | ||||
|     "no-restricted-globals": "off", | ||||
|     "@typescript-eslint/consistent-type-imports": ["error", { "prefer": "type-imports", "disallowTypeAnnotations": false, "fixStyle": "separate-type-imports" }] | ||||
|     "no-restricted-globals": "off" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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 -W | ||||
|           yarn add @actions/core | ||||
|           yarn autorelease | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							| @@ -44,7 +44,7 @@ jobs: | ||||
|       - name: Auto release preview | ||||
|         id: "autorelease" | ||||
|         run: | | ||||
|           yarn add @actions/core -W | ||||
|           yarn add @actions/core | ||||
|           yarn autorelease preview ${{ github.event.issue.number }} | ||||
|       - name: Post comment post release | ||||
|         if: always() | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,7 @@ jobs: | ||||
|  | ||||
|       - name: Install and lint | ||||
|         run: | | ||||
|           yarn install | ||||
|           yarn --frozen-lockfile | ||||
|           yarn test:other | ||||
|           yarn test:code | ||||
|           yarn test:typecheck | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/locales-coverage.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/locales-coverage.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v2 | ||||
|         with: | ||||
|           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} | ||||
|  | ||||
| @@ -22,11 +22,11 @@ jobs: | ||||
|       - name: Create report file | ||||
|         run: | | ||||
|           yarn locales-coverage | ||||
|           FILE_CHANGED=$(git diff packages/excalidraw/locales/percentages.json) | ||||
|           FILE_CHANGED=$(git diff src/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 packages/excalidraw/locales/percentages.json | ||||
|             git add src/locales/percentages.json | ||||
|             git commit -am "Auto commit: Calculate translation coverage" | ||||
|             git push | ||||
|           fi | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/semantic-pr-title.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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@v5 | ||||
|       - uses: amannn/action-semantic-pull-request@v3.0.0 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/workflows/size-limit.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/size-limit.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,14 +15,16 @@ jobs: | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: 18.x | ||||
|       - name: Install in packages/excalidraw | ||||
|         run: yarn | ||||
|         working-directory: packages/excalidraw | ||||
|       - name: Install | ||||
|         run: yarn --frozen-lockfile | ||||
|       - name: Install in src/packages/excalidraw | ||||
|         run: yarn --frozen-lockfile | ||||
|         working-directory: src/packages/excalidraw | ||||
|         env: | ||||
|           CI: true | ||||
|       - uses: andresz1/size-limit-action@v1 | ||||
|         with: | ||||
|           github_token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           build_script: build:esm | ||||
|           build_script: build:umd | ||||
|           skip_step: install | ||||
|           directory: packages/excalidraw | ||||
|           directory: src/packages/excalidraw | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/test-coverage-pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-coverage-pr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,7 @@ jobs: | ||||
|         with: | ||||
|           node-version: "18.x" | ||||
|       - name: "Install Deps" | ||||
|         run: yarn install | ||||
|         run: yarn --frozen-lockfile | ||||
|       - name: "Test Coverage" | ||||
|         run: yarn test:coverage | ||||
|       - name: "Report Coverage" | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,19 +1,17 @@ | ||||
| name: Tests | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: master | ||||
| on: pull_request | ||||
|  | ||||
| jobs: | ||||
|   test: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v2 | ||||
|       - name: Setup Node.js 18.x | ||||
|         uses: actions/setup-node@v4 | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 18.x | ||||
|       - name: Install and test | ||||
|         run: | | ||||
|           yarn install | ||||
|           yarn --frozen-lockfile | ||||
|           yarn test:app | ||||
|   | ||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -21,9 +21,10 @@ npm-debug.log* | ||||
| package-lock.json | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| packages/excalidraw/types | ||||
| 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 | ||||
| coverage | ||||
| dev-dist | ||||
| html | ||||
| examples/**/bundle.* | ||||
| meta*.json | ||||
							
								
								
									
										12
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -2,18 +2,16 @@ FROM node:18 AS build | ||||
|  | ||||
| WORKDIR /opt/node_app | ||||
|  | ||||
| COPY . . | ||||
|  | ||||
| # do not ignore optional dependencies: | ||||
| # Error: Cannot find module @rollup/rollup-linux-x64-gnu | ||||
| RUN yarn --network-timeout 600000 | ||||
| COPY package.json yarn.lock ./ | ||||
| RUN yarn --ignore-optional --network-timeout 600000 | ||||
|  | ||||
| ARG NODE_ENV=production | ||||
|  | ||||
| COPY . . | ||||
| RUN yarn build:app:docker | ||||
|  | ||||
| FROM nginx:1.27-alpine | ||||
| FROM nginx:1.21-alpine | ||||
|  | ||||
| COPY --from=build /opt/node_app/excalidraw-app/build /usr/share/nginx/html | ||||
| COPY --from=build /opt/node_app/build /usr/share/nginx/html | ||||
|  | ||||
| HEALTHCHECK CMD wget -q -O /dev/null http://localhost || exit 1 | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
|  | ||||
| <h4 align="center"> | ||||
|   <a href="https://excalidraw.com">Excalidraw Editor</a> | | ||||
|   <a href="https://plus.excalidraw.com/blog">Blog</a> | | ||||
|   <a href="https://blog.excalidraw.com">Blog</a> | | ||||
|   <a href="https://docs.excalidraw.com">Documentation</a> | | ||||
|   <a href="https://plus.excalidraw.com">Excalidraw+</a> | ||||
| </h4> | ||||
| @@ -85,7 +85,7 @@ We'll be adding these features as drop-in plugins for the npm package in the fut | ||||
|  | ||||
| ## Quick start | ||||
|  | ||||
| **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). | ||||
| Install the [Excalidraw npm package](https://www.npmjs.com/package/@excalidraw/excalidraw): | ||||
|  | ||||
| ``` | ||||
| npm install react react-dom @excalidraw/excalidraw | ||||
| @@ -97,7 +97,7 @@ or via yarn | ||||
| yarn add react react-dom @excalidraw/excalidraw | ||||
| ``` | ||||
|  | ||||
| Check out our [documentation](https://docs.excalidraw.com/docs/@excalidraw/excalidraw/installation) for more details! | ||||
| Don't forget to check out our [Documentation](https://docs.excalidraw.com)! | ||||
|  | ||||
| ## Contributing | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,3 @@ | ||||
| files: | ||||
|   - source: /packages/excalidraw/locales/en.json | ||||
|     translation: /packages/excalidraw/locales/%locale%.json | ||||
|   - source: /src/locales/en.json | ||||
|     translation: /src/locales/%locale%.json | ||||
|   | ||||
| @@ -133,7 +133,7 @@ function App() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Here is a [complete list](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/components/main-menu/DefaultItems.tsx) of the default items. | ||||
| Here is a [complete list](https://github.com/excalidraw/excalidraw/blob/master/src/components/mainMenu/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 hand-drawn like diagrams." | ||||
|   alt="Excalidraw logo: Sketch handrawn like diagrams." | ||||
| /> | ||||
|  | ||||
| ### Center | ||||
|   | ||||
| @@ -12,7 +12,7 @@ import { FONT_FAMILY } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| | Font Family | Description            | | ||||
| | ----------- | ---------------------- | | ||||
| | `Virgil`    | The `Hand-drawn` font | | ||||
| | `Virgil`    | The `handwritten` font | | ||||
| | `Helvetica` | The `Normal` Font      | | ||||
| | `Cascadia`  | The `Code` Font        | | ||||
|  | ||||
| @@ -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/packages/excalidraw/constants.ts#L101) contains all the mime types supported by `Excalidraw`. | ||||
| [`MIME_TYPES`](https://github.com/excalidraw/excalidraw/blob/master/src/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/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). | ||||
| 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). | ||||
|  | ||||
| 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). | ||||
| 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). | ||||
|  | ||||
| ## convertToExcalidrawElements | ||||
|  | ||||
| @@ -19,7 +19,7 @@ convertToExcalidrawElements( | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | `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. | | ||||
| | `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. | | ||||
| | `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/packages/excalidraw/element/types.ts#L27) as well to decorate the shapes. | ||||
| You can pass additional [`properties`](https://github.com/excalidraw/excalidraw/blob/master/src/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/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. | ||||
| 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. | ||||
|  | ||||
| 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/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 | ||||
| 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 | ||||
|  | ||||
| ```js | ||||
| convertToExcalidrawElements([ | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| <pre> | ||||
|   (api:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L616"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/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/src/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 library | | ||||
| | [updateLibrary](#updatelibrary) | `function` | updates the scene with the sceneData | | ||||
| | [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 | | ||||
| | [toggleSidebar](#toggleSidebar) | `function` | Toggles specific sidebar on/off | | ||||
| | [toggleMenu](#togglemenu) | `function` | Toggles specific menus 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/packages/excalidraw/types.ts#L339"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/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. | | ||||
| | `commitToStore` | `boolean` | Implies if the change should be captured and commited to the `store`. Commited changes are emmitted and listened to by other components, such as `History` for undo / redo purposes. Defaults to `false`. | | ||||
| | `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`. | | ||||
|  | ||||
| ```jsx live | ||||
| function App() { | ||||
| @@ -115,7 +115,7 @@ function App() { | ||||
|       <button className="custom-button" onClick={updateScene}> | ||||
|         Update Scene | ||||
|       </button> | ||||
|       <Excalidraw excalidrawAPI={(api) => setExcalidrawAPI(api)} /> | ||||
|       <Excalidraw ref={(api) => setExcalidrawAPI(api)} /> | ||||
|     </div> | ||||
|   ); | ||||
| } | ||||
| @@ -125,13 +125,13 @@ function App() { | ||||
|  | ||||
| <pre> | ||||
|   (opts: { <br /> libraryItems:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L249"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/types.ts#L246"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L246"> | ||||
|     LibraryItems | ||||
|   </a> | ||||
|   > | ||||
| @@ -141,7 +141,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/packages/excalidraw/types.ts#L249) | \_ | The `libraryItems` to be replaced/merged with current library | | ||||
| | `libraryItems` | [LibraryItemsSource](https://github.com/excalidraw/excalidraw/blob/master/src/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 +188,8 @@ function App() { | ||||
|         Update Library | ||||
|       </button> | ||||
|       <Excalidraw | ||||
|         excalidrawAPI={(api) => setExcalidrawAPI(api)} | ||||
|         // initial data retrieved from https://github.com/excalidraw/excalidraw/blob/master/dev-docs/packages/excalidraw/initialData.js | ||||
|         ref={(api) => setExcalidrawAPI(api)} | ||||
|         // initial data retrieved from https://github.com/excalidraw/excalidraw/blob/master/dev-docs/src/initialData.js | ||||
|         initialData={{ | ||||
|           libraryItems: initialData.libraryItems, | ||||
|           appState: { openSidebar: "library" }, | ||||
| @@ -204,7 +204,7 @@ function App() { | ||||
|  | ||||
| <pre> | ||||
|   (files:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L59"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L59"> | ||||
|     BinaryFileData | ||||
|   </a> | ||||
|   ) => void | ||||
| @@ -224,7 +224,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/packages/excalidraw/element/types.ts#L115"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L115"> | ||||
|     ExcalidrawElement[] | ||||
|   </a> | ||||
| </pre> | ||||
| @@ -235,7 +235,7 @@ Returns all the elements including the deleted in the scene. | ||||
|  | ||||
| <pre> | ||||
|   () => NonDeleted< | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L115"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L115"> | ||||
|     ExcalidrawElement | ||||
|   </a> | ||||
|   []> | ||||
| @@ -247,7 +247,7 @@ Returns all the elements excluding the deleted in the scene | ||||
|  | ||||
| <pre> | ||||
|   () =>{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95"> | ||||
|     AppState | ||||
|   </a> | ||||
| </pre> | ||||
| @@ -288,7 +288,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/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. | | ||||
| | 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. | | ||||
| | 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 +336,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/packages/excalidraw/types.ts#L82"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L82"> | ||||
|     files | ||||
|   </a> | ||||
| </pre> | ||||
| @@ -364,7 +364,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/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` | | ||||
| | `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` | | ||||
| | `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/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> } | ||||
| { 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> } | ||||
| </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/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. | | ||||
| | `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. | | ||||
| | `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/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. | | ||||
| | `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. | | ||||
|  | ||||
| You might want to use this when you want to load excalidraw with some initial elements and app state. | ||||
|  | ||||
|   | ||||
| @@ -9,9 +9,9 @@ All `props` are _optional_. | ||||
| | [`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 | | ||||
| | [`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 something is pasted into the scene | | ||||
| | [`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. | | ||||
| | [`langCode`](#langcode) | `string` | `en` | Language code string to be used in Excalidraw | | ||||
| @@ -23,17 +23,17 @@ All `props` are _optional_. | ||||
| | [`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/packages/excalidraw/constants.ts#L151) | To customise UI options. Currently we support customising [`canvas actions`](/docs/@excalidraw/excalidraw/api/props/ui-options#canvasactions) | | ||||
| | [`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) | | ||||
| | [`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 | | ||||
| | [`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 | | ||||
| | [`renderEmbeddable`](/docs/@excalidraw/excalidraw/api/props/render-props#renderEmbeddable) | `function` | \_ | Render function that can override the built-in `<iframe>` | | ||||
|  | ||||
| ### 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/packages/excalidraw/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/src/element/types.ts#L66) and is optional. | ||||
|  | ||||
| You can use this to add any extra information you need to keep track of. | ||||
|  | ||||
| @@ -59,11 +59,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/packages/excalidraw/element/types.ts#L114) in the scene. | ||||
| 1. `excalidrawElements`: Array of [excalidrawElements](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114) in the scene. | ||||
|  | ||||
| 2. `appState`: [AppState](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95) of the scene. | ||||
| 2. `appState`: [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95) of the scene. | ||||
|  | ||||
| 3. `files`: The [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L64) which are added to the scene. | ||||
| 3. `files`: The [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/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,14 +79,14 @@ 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/packages/excalidraw/types.ts#L131) map of the scene | ||||
| 3.`pointersMap`: [`pointers`](https://github.com/excalidraw/excalidraw/blob/master/src/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/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. | ||||
| 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. | ||||
| 3. `canvas`: The `HTMLCanvasElement` of the scene. | ||||
|  | ||||
| ### onPointerDown | ||||
| @@ -96,11 +96,11 @@ This prop if passed will be triggered on pointer down events and has the below s | ||||
|  | ||||
| <pre> | ||||
| (activeTool:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L115"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L115"> | ||||
|     {" "} | ||||
|     AppState["activeTool"] | ||||
|   </a> | ||||
|   , pointerDownState: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L424"> | ||||
|   , pointerDownState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L424"> | ||||
|     PointerDownState | ||||
|   </a>) => void | ||||
| </pre> | ||||
| @@ -119,7 +119,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/packages/excalidraw/clipboard.ts#L18"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/clipboard.ts#L18"> | ||||
|     ClipboardData | ||||
|   </a> | ||||
|   , event: ClipboardEvent | null) => boolean | ||||
| @@ -135,7 +135,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/packages/excalidraw/types.ts#L200"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200"> | ||||
|     LibraryItems | ||||
|   </a> | ||||
|   ) => void | Promise<any> | ||||
| @@ -149,7 +149,7 @@ This prop if passed will be triggered when clicked on `link`. To handle the redi | ||||
|  | ||||
| <pre> | ||||
|   (element:{" "} | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L114"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L114"> | ||||
|     ExcalidrawElement | ||||
|   </a> | ||||
|   , event: CustomEvent<{ nativeEvent: MouseEvent }>) => void | ||||
| @@ -182,7 +182,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/packages/excalidraw/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/src/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 +191,7 @@ import { defaultLang, languages } from "@excalidraw/excalidraw"; | ||||
| | name | type | | ||||
| | --- | --- | | ||||
| | `defaultLang` | `string` | | ||||
| | `languages` | [`Language[]`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/i18n.ts#L15) | | ||||
| | `languages` | [`Language[]`](https://github.com/excalidraw/excalidraw/blob/master/src/i18n.ts#L15) | | ||||
|  | ||||
| ### viewModeEnabled | ||||
|  | ||||
|   | ||||
| @@ -4,7 +4,7 @@ | ||||
|  | ||||
| <pre> | ||||
|   (isMobile: boolean, appState: | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L95"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/types.ts#L95"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/types.ts#L372"> | ||||
|   <br /> canvasActions?: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/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/src/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/packages/utils/export.ts#L24">ExportOpts</a> | ||||
| }: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L21">ExportOpts</a> | ||||
| </pre> | ||||
|  | ||||
| | Name | Type | Default | Description | | ||||
| | --- | --- | --- | --- | | ||||
| | `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. | | ||||
| | `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. | | ||||
| | [`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/packages/excalidraw/types.ts#L59) | _ | The files added to the scene. | | ||||
| | `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/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 excalidrawAPI={(api) => setExcalidrawAPI(api)} | ||||
|         <Excalidraw ref={(api) => setExcalidrawAPI(api)} | ||||
| /> | ||||
|       </div> | ||||
|     </> | ||||
| @@ -105,7 +105,7 @@ function App() { | ||||
|  | ||||
| <pre> | ||||
| exportToBlob(<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/packages/utils.ts#L14">ExportOpts</a> & {<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/element/types.ts#L114"> | ||||
|     <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/packages/excalidraw/types.ts#L95"> AppState | ||||
|     <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L95"> AppState | ||||
|     </a>,<br/>  | ||||
|   exportPadding: number,<br/>  | ||||
|   metadata: string,<br/>  | ||||
|   files:  | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L59"> | ||||
|   <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/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 | | ||||
| | 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 | | ||||
| | exportPadding | number | 10 | The `padding` to be added on canvas | | ||||
| | files | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/types.ts#L64) | undefined | The `files` added to the scene. | | ||||
| | files | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/packages/utils.ts#L21">ExportOpts</a> & {<br/>  | ||||
|   opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/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> | ||||
| 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> | ||||
| </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/packages/excalidraw/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/src/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/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/>  | ||||
|   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/> | ||||
| ) | ||||
| </pre> | ||||
|  | ||||
| | Prop | Type | Description | | ||||
| | ---- | ---- | ---- | | ||||
| | `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`. | | ||||
| | `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`. | | ||||
| | [`opts`](#opts) | `Object` | The extra optional parameter to configure restored elements | ||||
|  | ||||
| #### localElements | ||||
| @@ -70,15 +70,15 @@ Parameter `refreshDimensions` indicates whether we should also `recalculate` tex | ||||
|  | ||||
| <pre> | ||||
| restore( | ||||
|   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/> | ||||
|   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/> | ||||
|  | ||||
| ) | ||||
| </pre> | ||||
|  | ||||
| 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`. | ||||
| 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`. | ||||
|  | ||||
| **_How to use_** | ||||
|  | ||||
| @@ -93,7 +93,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/packages/excalidraw/data/types.ts#L34">ImportedDataState["libraryItems"]</a>,<br/>  | ||||
| restoreLibraryItems(libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/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/src/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/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/> | ||||
|   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/> | ||||
| }): 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/packages/excalidraw/types.ts#L200">LibraryItems[]</a>) | ||||
|   libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/element/types.ts#L114">ExcalidrawElement</a>): boolean | ||||
| isInvisiblySmallElement(element:  <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/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/>  | ||||
|   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/>  | ||||
|   fileHandle?: FileSystemHandle | null <br/> | ||||
| ) => Promise<<a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/data/restore.ts#L61">RestoredDataState</a>> | ||||
| ) => Promise<<a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/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/>  | ||||
|   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/>  | ||||
|   fileHandle?: FileSystemHandle | null<br/> | ||||
| ) => 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>}> | ||||
| ) => 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>}> | ||||
| </pre> | ||||
|  | ||||
| ### getFreeDrawSvgPath | ||||
| @@ -149,7 +149,7 @@ import { getFreeDrawSvgPath } from "@excalidraw/excalidraw"; | ||||
| **Signature** | ||||
|  | ||||
| <pre> | ||||
| getFreeDrawSvgPath(element: <a href="https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/element/types.ts#L182">ExcalidrawFreeDrawElement</a>) | ||||
| getFreeDrawSvgPath(element: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/element/types.ts#L80">ExcalidrawElement</a>): boolean | ||||
| isLinearElement(elementType?: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/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> | ||||
| 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> | ||||
| </pre> | ||||
|  | ||||
| ### mergeLibraryItems | ||||
| @@ -196,9 +196,9 @@ import { mergeLibraryItems } from "@excalidraw/excalidraw"; | ||||
|  | ||||
| <pre> | ||||
| mergeLibraryItems(<br/>  | ||||
|   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> | ||||
|   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> | ||||
| </pre> | ||||
|  | ||||
| ### parseLibraryTokensFromUrl | ||||
| @@ -239,8 +239,8 @@ export const App = () => { | ||||
|  | ||||
| <pre> | ||||
| useHandleLibrary(opts: {<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/> | ||||
|   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/> | ||||
| }); | ||||
| </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/packages/excalidraw/element/types.ts#L114">ExcalidrawElement[]</a>) | ||||
| getSceneVersion(elements:  <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/types.ts#L95">AppState</a><br/>): { x: number, y: number } | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/types.ts#L95">AppState</a><br/>): {x: number, y: number} | ||||
|   appState: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/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/packages/excalidraw/i18n.ts#L15) | | ||||
| | `useI18n` | [`() => { langCode, t }`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/i18n.ts#L15) | | ||||
| | `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) | | ||||
|  | ||||
| ```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/packages/excalidraw/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/src/css/theme.scss), though most of them will not make sense to override. | ||||
|  | ||||
| ```css showLineNumbers | ||||
| .custom-styles .excalidraw { | ||||
|   | ||||
| @@ -13,7 +13,7 @@ To start the example app using the `@excalidraw/excalidraw` package, follow the | ||||
| 1. Install the dependencies | ||||
|  | ||||
|    ```bash | ||||
|    cd packages/excalidraw && yarn | ||||
|    cd src/packages/excalidraw && yarn | ||||
|    ``` | ||||
|  | ||||
| 2. Start the example app | ||||
|   | ||||
| @@ -39,7 +39,7 @@ Since Vite removes env variables by default, you can update the vite config to e | ||||
|  | ||||
| ``` | ||||
|  define: { | ||||
|     "process.env.IS_PREACT": JSON.stringify("true"), | ||||
|     "process.env.IS_PREACT": process.env.IS_PREACT, | ||||
|   }, | ||||
| ``` | ||||
|  | ||||
|   | ||||
| @@ -32,9 +32,15 @@ function App() { | ||||
|  | ||||
| ### Next.js | ||||
|  | ||||
| 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`. | ||||
| Since _Excalidraw_ doesn't support server side rendering, you should render the component once the host is `mounted`. | ||||
|  | ||||
| If you want to only import `Excalidraw` component you can do :point_down: | ||||
| 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`. | ||||
|  | ||||
| ```jsx showLineNumbers | ||||
| import dynamic from "next/dynamic"; | ||||
| @@ -49,88 +55,25 @@ export default function App() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| 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. | ||||
| Here is a working [demo](https://codesandbox.io/p/sandbox/excalidraw-with-next-dynamic-k8yjq2). | ||||
|  | ||||
| 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: | ||||
|  | ||||
| <Tabs> | ||||
|   <TabItem value="Excalidraw Wrapper" label="Excalidraw Wrapper" > | ||||
| 2. Importing Excalidraw once **client** is rendered. | ||||
|  | ||||
|   ```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>  | ||||
| ```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), | ||||
|     ); | ||||
|   }; | ||||
|   export default ExcalidrawWrapper; | ||||
|   ``` | ||||
|  | ||||
|   </TabItem> | ||||
|  | ||||
|   <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> | ||||
|  | ||||
|  | ||||
| Here is a [source code](https://github.com/excalidraw/excalidraw/tree/master/examples/excalidraw/with-nextjs) for the example with app and pages router. You you can try it out [here](https://excalidraw-package-example-with-nextjs-gh6smrdnq-excalidraw.vercel.app/). | ||||
|   }, []); | ||||
|   return <>{Excalidraw && <Excalidraw />}</>; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Here is a working [demo](https://codesandbox.io/p/sandbox/excalidraw-with-next-5xb3d) | ||||
|  | ||||
| The `types` are available at `@excalidraw/excalidraw/types`, you can view [example for typescript](https://codesandbox.io/s/excalidraw-types-9h2dm) | ||||
|  | ||||
| @@ -150,7 +93,7 @@ Since Vite removes env variables by default, you can update the vite config to e | ||||
|  | ||||
| ``` | ||||
|  define: { | ||||
|     "process.env.IS_PREACT": JSON.stringify("true"), | ||||
|     "process.env.IS_PREACT": process.env.IS_PREACT, | ||||
|   }, | ||||
| ``` | ||||
| :::  | ||||
| @@ -205,7 +148,7 @@ import TabItem from "@theme/TabItem"; | ||||
|       <h1>Excalidraw Embed Example</h1> | ||||
|       <div id="app"></div> | ||||
|     </div> | ||||
|     <script type="text/javascript" src="packages/excalidraw/index.js"></script> | ||||
|     <script type="text/javascript" src="src/index.js"></script> | ||||
|   </body> | ||||
| </html> | ||||
| ``` | ||||
|   | ||||
| @@ -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 parseMermaidToExcalidraw(mermaidSyntax: string, { | ||||
|   const { elements, files } = await parseMermaid(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/packages/excalidraw/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/src/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/packages/excalidraw/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/src/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/packages/excalidraw/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/src/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/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 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 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/packages/excalidraw/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/src/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/packages/excalidraw/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/src/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": 20, | ||||
|     "gridSize": null, | ||||
|     "viewBackgroundColor": "#ffffff" | ||||
|   }, | ||||
|  | ||||
|   | ||||
| @@ -52,6 +52,15 @@ 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,7 +41,10 @@ const config = { | ||||
|           showLastUpdateTime: true, | ||||
|         }, | ||||
|         theme: { | ||||
|           customCss: [require.resolve("./src/css/custom.scss")], | ||||
|           customCss: [ | ||||
|             require.resolve("./src/css/custom.scss"), | ||||
|             require.resolve("../src/packages/excalidraw/example/App.scss"), | ||||
|           ], | ||||
|         }, | ||||
|       }), | ||||
|     ], | ||||
| @@ -66,7 +69,7 @@ const config = { | ||||
|             label: "Docs", | ||||
|           }, | ||||
|           { | ||||
|             to: "https://plus.excalidraw.com/blog", | ||||
|             to: "https://blog.excalidraw.com", | ||||
|             label: "Blog", | ||||
|             position: "left", | ||||
|           }, | ||||
| @@ -111,7 +114,7 @@ const config = { | ||||
|             items: [ | ||||
|               { | ||||
|                 label: "Blog", | ||||
|                 to: "https://plus.excalidraw.com/blog", | ||||
|                 to: "https://blog.excalidraw.com", | ||||
|               }, | ||||
|               { | ||||
|                 label: "GitHub", | ||||
|   | ||||
| @@ -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.6", | ||||
|     "@excalidraw/excalidraw": "0.17.0", | ||||
|     "@mdx-js/react": "^1.6.22", | ||||
|     "clsx": "^1.2.1", | ||||
|     "docusaurus-plugin-sass": "0.2.3", | ||||
|     "prism-react-renderer": "^1.3.5", | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0", | ||||
|     "react": "^17.0.2", | ||||
|     "react-dom": "^17.0.2", | ||||
|     "sass": "1.57.1" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|   | ||||
| @@ -59,7 +59,7 @@ pre a { | ||||
|   padding: 5px; | ||||
|   background: #70b1ec; | ||||
|   color: white; | ||||
|   font-weight: 700; | ||||
|   font-weight: bold; | ||||
|   border: none; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,4 +0,0 @@ | ||||
| { | ||||
|   "outputDirectory": "build", | ||||
|   "installCommand": "yarn install" | ||||
| } | ||||
| @@ -1547,7 +1547,7 @@ | ||||
|     "@docusaurus/theme-search-algolia" "2.2.0" | ||||
|     "@docusaurus/types" "2.2.0" | ||||
|  | ||||
| "@docusaurus/react-loadable@5.5.2": | ||||
| "@docusaurus/react-loadable@5.5.2", "react-loadable@npm:@docusaurus/react-loadable@5.5.2": | ||||
|   version "5.5.2" | ||||
|   resolved "https://registry.yarnpkg.com/@docusaurus/react-loadable/-/react-loadable-5.5.2.tgz#81aae0db81ecafbdaee3651f12804580868fa6ce" | ||||
|   integrity sha512-A3dYjdBGuy0IGT+wyLIGIKLRE+sAk1iNk0f1HjNDysO7u8lhL4N3VEm+FAubmJbAztn94F7MxBTPmnixbiyFdQ== | ||||
| @@ -1718,10 +1718,10 @@ | ||||
|     url-loader "^4.1.1" | ||||
|     webpack "^5.73.0" | ||||
|  | ||||
| "@excalidraw/excalidraw@0.17.6": | ||||
|   version "0.17.6" | ||||
|   resolved "https://registry.yarnpkg.com/@excalidraw/excalidraw/-/excalidraw-0.17.6.tgz#5fd208ce69d33ca712d1804b50d7d06d5c46ac4d" | ||||
|   integrity sha512-fyCl+zG/Z5yhHDh5Fq2ZGmphcrALmuOdtITm8gN4d8w4ntnaopTXcTfnAAaU3VleDC6LhTkoLOTG6P5kgREiIg== | ||||
| "@excalidraw/excalidraw@0.17.0": | ||||
|   version "0.17.0" | ||||
|   resolved "https://registry.yarnpkg.com/@excalidraw/excalidraw/-/excalidraw-0.17.0.tgz#3c64aa8e36406ac171b008cfecbdce5bb0755725" | ||||
|   integrity sha512-NzP22v5xMqxYW27ZtTHhiGFe7kE8NeBk45aoeM/mDSkXiOXPDH+PcvwzHRN/Ei+Vj/0sTPHxejn8bZyRWKGjXg== | ||||
|  | ||||
| "@hapi/hoek@^9.0.0": | ||||
|   version "9.3.0" | ||||
| @@ -2789,14 +2789,7 @@ brace-expansion@^1.1.7: | ||||
|     balanced-match "^1.0.0" | ||||
|     concat-map "0.0.1" | ||||
|  | ||||
| braces@^3.0.3: | ||||
|   version "3.0.3" | ||||
|   resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.3.tgz#490332f40919452272d55a8480adc0c441358789" | ||||
|   integrity sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA== | ||||
|   dependencies: | ||||
|     fill-range "^7.1.1" | ||||
|  | ||||
| braces@~3.0.2: | ||||
| braces@^3.0.2, braces@~3.0.2: | ||||
|   version "3.0.2" | ||||
|   resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" | ||||
|   integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== | ||||
| @@ -4018,13 +4011,6 @@ fill-range@^7.0.1: | ||||
|   dependencies: | ||||
|     to-regex-range "^5.0.1" | ||||
|  | ||||
| fill-range@^7.1.1: | ||||
|   version "7.1.1" | ||||
|   resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.1.1.tgz#44265d3cac07e3ea7dc247516380643754a05292" | ||||
|   integrity sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg== | ||||
|   dependencies: | ||||
|     to-regex-range "^5.0.1" | ||||
|  | ||||
| finalhandler@1.2.0: | ||||
|   version "1.2.0" | ||||
|   resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.2.0.tgz#7d23fe5731b207b4640e4fcd00aec1f9207a7b32" | ||||
| @@ -5221,11 +5207,11 @@ methods@~1.1.2: | ||||
|   integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w== | ||||
|  | ||||
| micromatch@^4.0.2, micromatch@^4.0.4, micromatch@^4.0.5: | ||||
|   version "4.0.8" | ||||
|   resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.8.tgz#d66fa18f3a47076789320b9b1af32bd86d9fa202" | ||||
|   integrity sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA== | ||||
|   version "4.0.5" | ||||
|   resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6" | ||||
|   integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA== | ||||
|   dependencies: | ||||
|     braces "^3.0.3" | ||||
|     braces "^3.0.2" | ||||
|     picomatch "^2.3.1" | ||||
|  | ||||
| mime-db@1.52.0, "mime-db@>= 1.43.0 < 2": | ||||
| @@ -6204,13 +6190,14 @@ react-dev-utils@^12.0.1: | ||||
|     strip-ansi "^6.0.1" | ||||
|     text-table "^0.2.0" | ||||
|  | ||||
| react-dom@18.2.0: | ||||
|   version "18.2.0" | ||||
|   resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-18.2.0.tgz#22aaf38708db2674ed9ada224ca4aa708d821e3d" | ||||
|   integrity sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g== | ||||
| react-dom@^17.0.2: | ||||
|   version "17.0.2" | ||||
|   resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-17.0.2.tgz#ecffb6845e3ad8dbfcdc498f0d0a939736502c23" | ||||
|   integrity sha512-s4h96KtLDUQlsENhMn1ar8t2bEa+q/YAtj8pPPdIjPDGBDIVNsrD9aXNWqspUe6AzKCIG0C1HZZLqLV7qpOBGA== | ||||
|   dependencies: | ||||
|     loose-envify "^1.1.0" | ||||
|     scheduler "^0.23.0" | ||||
|     object-assign "^4.1.1" | ||||
|     scheduler "^0.20.2" | ||||
|  | ||||
| react-error-overlay@^6.0.11: | ||||
|   version "6.0.11" | ||||
| @@ -6273,14 +6260,6 @@ react-loadable-ssr-addon-v5-slorber@^1.0.1: | ||||
|   dependencies: | ||||
|     "@babel/runtime" "^7.10.3" | ||||
|  | ||||
| "react-loadable@npm:@docusaurus/react-loadable@5.5.2": | ||||
|   version "5.5.2" | ||||
|   resolved "https://registry.yarnpkg.com/@docusaurus/react-loadable/-/react-loadable-5.5.2.tgz#81aae0db81ecafbdaee3651f12804580868fa6ce" | ||||
|   integrity sha512-A3dYjdBGuy0IGT+wyLIGIKLRE+sAk1iNk0f1HjNDysO7u8lhL4N3VEm+FAubmJbAztn94F7MxBTPmnixbiyFdQ== | ||||
|   dependencies: | ||||
|     "@types/react" "*" | ||||
|     prop-types "^15.6.2" | ||||
|  | ||||
| react-router-config@^5.1.1: | ||||
|   version "5.1.1" | ||||
|   resolved "https://registry.yarnpkg.com/react-router-config/-/react-router-config-5.1.1.tgz#0f4263d1a80c6b2dc7b9c1902c9526478194a988" | ||||
| @@ -6331,12 +6310,13 @@ react-textarea-autosize@^8.3.2: | ||||
|     use-composed-ref "^1.3.0" | ||||
|     use-latest "^1.2.1" | ||||
|  | ||||
| react@18.2.0: | ||||
|   version "18.2.0" | ||||
|   resolved "https://registry.yarnpkg.com/react/-/react-18.2.0.tgz#555bd98592883255fa00de14f1151a917b5d77d5" | ||||
|   integrity sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ== | ||||
| react@^17.0.2: | ||||
|   version "17.0.2" | ||||
|   resolved "https://registry.yarnpkg.com/react/-/react-17.0.2.tgz#d0b5cc516d29eb3eee383f75b62864cfb6800037" | ||||
|   integrity sha512-gnhPt75i/dq/z3/6q/0asP78D0u592D5L1pd7M8P+dck6Fu/jJeL6iVVK23fptSUZj8Vjf++7wXA8UNclGQcbA== | ||||
|   dependencies: | ||||
|     loose-envify "^1.1.0" | ||||
|     object-assign "^4.1.1" | ||||
|  | ||||
| readable-stream@^2.0.1: | ||||
|   version "2.3.7" | ||||
| @@ -6684,12 +6664,13 @@ sax@^1.2.4: | ||||
|   resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9" | ||||
|   integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw== | ||||
|  | ||||
| scheduler@^0.23.0: | ||||
|   version "0.23.2" | ||||
|   resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.23.2.tgz#414ba64a3b282892e944cf2108ecc078d115cdc3" | ||||
|   integrity sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ== | ||||
| scheduler@^0.20.2: | ||||
|   version "0.20.2" | ||||
|   resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.20.2.tgz#4baee39436e34aa93b4874bddcbf0fe8b8b50e91" | ||||
|   integrity sha512-2eWfGgAqqWFGqtdMmcL5zCMK1U8KlXv8SQFGglL3CEtd0aDVDWgeF/YoCmvln55m5zSk3J/20hTaSBeSObsQDQ== | ||||
|   dependencies: | ||||
|     loose-envify "^1.1.0" | ||||
|     object-assign "^4.1.1" | ||||
|  | ||||
| schema-utils@2.7.0: | ||||
|   version "2.7.0" | ||||
|   | ||||
| @@ -1,27 +0,0 @@ | ||||
| import type { ExcalidrawImperativeAPI } from "@excalidraw/excalidraw/dist/excalidraw/types"; | ||||
| import CustomFooter from "./CustomFooter"; | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
|  | ||||
| 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,13 +0,0 @@ | ||||
| { | ||||
|   "name": "examples", | ||||
|   "version": "1.0.0", | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0", | ||||
|     "@excalidraw/excalidraw": "*" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "typescript": "^5" | ||||
|   } | ||||
| } | ||||
| @@ -1,3 +0,0 @@ | ||||
| { | ||||
|   "extends": "../../tsconfig" | ||||
| } | ||||
| @@ -1,146 +0,0 @@ | ||||
| import { unstable_batchedUpdates } from "react-dom"; | ||||
| import { fileOpen as _fileOpen } from "browser-fs-access"; | ||||
| import { MIME_TYPES } from "@excalidraw/excalidraw"; | ||||
| import { AbortError } from "../../packages/excalidraw/errors"; | ||||
|  | ||||
| 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 AbortError()); | ||||
|         } | ||||
|       }; | ||||
|     }, | ||||
|   }) 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); | ||||
| }; | ||||
							
								
								
									
										39
									
								
								examples/excalidraw/with-nextjs/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								examples/excalidraw/with-nextjs/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,39 +0,0 @@ | ||||
| # 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 | ||||
| @@ -1,36 +0,0 @@ | ||||
| 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. | ||||
| @@ -1,12 +0,0 @@ | ||||
| /** @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; | ||||
| @@ -1,26 +0,0 @@ | ||||
| { | ||||
|   "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": { | ||||
|     "@excalidraw/excalidraw": "*", | ||||
|     "next": "14.1", | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/node": "^20", | ||||
|     "@types/react": "18.2.0", | ||||
|     "@types/react-dom": "18.2.0", | ||||
|     "path2d-polyfill": "2.0.1", | ||||
|     "typescript": "^5" | ||||
|   } | ||||
| } | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 25 KiB | 
| @@ -1,11 +0,0 @@ | ||||
| export default function RootLayout({ | ||||
|   children, | ||||
| }: { | ||||
|   children: React.ReactNode; | ||||
| }) { | ||||
|   return ( | ||||
|     <html lang="en"> | ||||
|       <body>{children}</body> | ||||
|     </html> | ||||
|   ); | ||||
| } | ||||
| @@ -1,26 +0,0 @@ | ||||
| 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 /> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
| @@ -1,15 +0,0 @@ | ||||
| * { | ||||
|   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; | ||||
| } | ||||
| @@ -1,22 +0,0 @@ | ||||
| "use client"; | ||||
| import * as excalidrawLib from "@excalidraw/excalidraw"; | ||||
| import { Excalidraw } from "@excalidraw/excalidraw"; | ||||
| import App from "../../components/ExampleApp"; | ||||
|  | ||||
| import "@excalidraw/excalidraw/index.css"; | ||||
|  | ||||
| const ExcalidrawWrapper: React.FC = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <App | ||||
|         appTitle={"Excalidraw with Nextjs Example"} | ||||
|         useCustom={(api: any, args?: any[]) => {}} | ||||
|         excalidrawLib={excalidrawLib} | ||||
|       > | ||||
|         <Excalidraw /> | ||||
|       </App> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default ExcalidrawWrapper; | ||||
| @@ -1,22 +0,0 @@ | ||||
| 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 /> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
| @@ -1,28 +0,0 @@ | ||||
| { | ||||
|   "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"] | ||||
| } | ||||
| @@ -1,3 +0,0 @@ | ||||
| { | ||||
|   "outputDirectory": "build" | ||||
| } | ||||
| @@ -1,252 +0,0 @@ | ||||
| # 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" | ||||
| @@ -1,2 +0,0 @@ | ||||
| # copied assets | ||||
| public/**/*.woff2 | ||||
| @@ -1,28 +0,0 @@ | ||||
| import App from "../components/ExampleApp"; | ||||
| import React, { StrictMode } from "react"; | ||||
| import { createRoot } from "react-dom/client"; | ||||
|  | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
|  | ||||
| import "@excalidraw/excalidraw/index.css"; | ||||
|  | ||||
| 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,21 +0,0 @@ | ||||
| { | ||||
|   "name": "with-script-in-browser", | ||||
|   "version": "1.0.0", | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0", | ||||
|     "@excalidraw/excalidraw": "*" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "vite": "5.0.12", | ||||
|     "typescript": "^5" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "build:workspace": "yarn workspace @excalidraw/excalidraw run build:esm && yarn copy:assets", | ||||
|     "copy:assets": "cp -r ../../../packages/excalidraw/dist/prod/fonts ./public", | ||||
|     "start": "yarn build:workspace && vite", | ||||
|     "build": "yarn build:workspace && vite build", | ||||
|     "build:preview": "yarn build && vite preview --port 5002" | ||||
|   } | ||||
| } | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 197 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 30 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 6.1 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 39 KiB | 
| @@ -1,4 +0,0 @@ | ||||
| { | ||||
|   "outputDirectory": "dist", | ||||
|   "installCommand": "yarn install" | ||||
| } | ||||
| @@ -1,11 +0,0 @@ | ||||
| import { defineConfig } from "vite"; | ||||
|  | ||||
| // https://vitejs.dev/config/ | ||||
| export default defineConfig({ | ||||
|   server: { | ||||
|     port: 3001, | ||||
|     // open the browser | ||||
|     open: true, | ||||
|   }, | ||||
|   publicDir: "public", | ||||
| }); | ||||
| @@ -1,313 +0,0 @@ | ||||
| # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. | ||||
| # yarn lockfile v1 | ||||
|  | ||||
|  | ||||
| "@esbuild/aix-ppc64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.19.11.tgz#2acd20be6d4f0458bc8c784103495ff24f13b1d3" | ||||
|   integrity sha512-FnzU0LyE3ySQk7UntJO4+qIiQgI7KoODnZg5xzXIrFJlKd2P2gwHsHY4927xj9y5PJmJSzULiUCWmv7iWnNa7g== | ||||
|  | ||||
| "@esbuild/android-arm64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.19.11.tgz#b45d000017385c9051a4f03e17078abb935be220" | ||||
|   integrity sha512-aiu7K/5JnLj//KOnOfEZ0D90obUkRzDMyqd/wNAUQ34m4YUPVhRZpnqKV9uqDGxT7cToSDnIHsGooyIczu9T+Q== | ||||
|  | ||||
| "@esbuild/android-arm@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.19.11.tgz#f46f55414e1c3614ac682b29977792131238164c" | ||||
|   integrity sha512-5OVapq0ClabvKvQ58Bws8+wkLCV+Rxg7tUVbo9xu034Nm536QTII4YzhaFriQ7rMrorfnFKUsArD2lqKbFY4vw== | ||||
|  | ||||
| "@esbuild/android-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.19.11.tgz#bfc01e91740b82011ef503c48f548950824922b2" | ||||
|   integrity sha512-eccxjlfGw43WYoY9QgB82SgGgDbibcqyDTlk3l3C0jOVHKxrjdc9CTwDUQd0vkvYg5um0OH+GpxYvp39r+IPOg== | ||||
|  | ||||
| "@esbuild/darwin-arm64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.19.11.tgz#533fb7f5a08c37121d82c66198263dcc1bed29bf" | ||||
|   integrity sha512-ETp87DRWuSt9KdDVkqSoKoLFHYTrkyz2+65fj9nfXsaV3bMhTCjtQfw3y+um88vGRKRiF7erPrh/ZuIdLUIVxQ== | ||||
|  | ||||
| "@esbuild/darwin-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.19.11.tgz#62f3819eff7e4ddc656b7c6815a31cf9a1e7d98e" | ||||
|   integrity sha512-fkFUiS6IUK9WYUO/+22omwetaSNl5/A8giXvQlcinLIjVkxwTLSktbF5f/kJMftM2MJp9+fXqZ5ezS7+SALp4g== | ||||
|  | ||||
| "@esbuild/freebsd-arm64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.19.11.tgz#d478b4195aa3ca44160272dab85ef8baf4175b4a" | ||||
|   integrity sha512-lhoSp5K6bxKRNdXUtHoNc5HhbXVCS8V0iZmDvyWvYq9S5WSfTIHU2UGjcGt7UeS6iEYp9eeymIl5mJBn0yiuxA== | ||||
|  | ||||
| "@esbuild/freebsd-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.19.11.tgz#7bdcc1917409178257ca6a1a27fe06e797ec18a2" | ||||
|   integrity sha512-JkUqn44AffGXitVI6/AbQdoYAq0TEullFdqcMY/PCUZ36xJ9ZJRtQabzMA+Vi7r78+25ZIBosLTOKnUXBSi1Kw== | ||||
|  | ||||
| "@esbuild/linux-arm64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.19.11.tgz#58ad4ff11685fcc735d7ff4ca759ab18fcfe4545" | ||||
|   integrity sha512-LneLg3ypEeveBSMuoa0kwMpCGmpu8XQUh+mL8XXwoYZ6Be2qBnVtcDI5azSvh7vioMDhoJFZzp9GWp9IWpYoUg== | ||||
|  | ||||
| "@esbuild/linux-arm@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.19.11.tgz#ce82246d873b5534d34de1e5c1b33026f35e60e3" | ||||
|   integrity sha512-3CRkr9+vCV2XJbjwgzjPtO8T0SZUmRZla+UL1jw+XqHZPkPgZiyWvbDvl9rqAN8Zl7qJF0O/9ycMtjU67HN9/Q== | ||||
|  | ||||
| "@esbuild/linux-ia32@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.19.11.tgz#cbae1f313209affc74b80f4390c4c35c6ab83fa4" | ||||
|   integrity sha512-caHy++CsD8Bgq2V5CodbJjFPEiDPq8JJmBdeyZ8GWVQMjRD0sU548nNdwPNvKjVpamYYVL40AORekgfIubwHoA== | ||||
|  | ||||
| "@esbuild/linux-loong64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.19.11.tgz#5f32aead1c3ec8f4cccdb7ed08b166224d4e9121" | ||||
|   integrity sha512-ppZSSLVpPrwHccvC6nQVZaSHlFsvCQyjnvirnVjbKSHuE5N24Yl8F3UwYUUR1UEPaFObGD2tSvVKbvR+uT1Nrg== | ||||
|  | ||||
| "@esbuild/linux-mips64el@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.19.11.tgz#38eecf1cbb8c36a616261de858b3c10d03419af9" | ||||
|   integrity sha512-B5x9j0OgjG+v1dF2DkH34lr+7Gmv0kzX6/V0afF41FkPMMqaQ77pH7CrhWeR22aEeHKaeZVtZ6yFwlxOKPVFyg== | ||||
|  | ||||
| "@esbuild/linux-ppc64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.19.11.tgz#9c5725a94e6ec15b93195e5a6afb821628afd912" | ||||
|   integrity sha512-MHrZYLeCG8vXblMetWyttkdVRjQlQUb/oMgBNurVEnhj4YWOr4G5lmBfZjHYQHHN0g6yDmCAQRR8MUHldvvRDA== | ||||
|  | ||||
| "@esbuild/linux-riscv64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.19.11.tgz#2dc4486d474a2a62bbe5870522a9a600e2acb916" | ||||
|   integrity sha512-f3DY++t94uVg141dozDu4CCUkYW+09rWtaWfnb3bqe4w5NqmZd6nPVBm+qbz7WaHZCoqXqHz5p6CM6qv3qnSSQ== | ||||
|  | ||||
| "@esbuild/linux-s390x@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.19.11.tgz#4ad8567df48f7dd4c71ec5b1753b6f37561a65a8" | ||||
|   integrity sha512-A5xdUoyWJHMMlcSMcPGVLzYzpcY8QP1RtYzX5/bS4dvjBGVxdhuiYyFwp7z74ocV7WDc0n1harxmpq2ePOjI0Q== | ||||
|  | ||||
| "@esbuild/linux-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.19.11.tgz#b7390c4d5184f203ebe7ddaedf073df82a658766" | ||||
|   integrity sha512-grbyMlVCvJSfxFQUndw5mCtWs5LO1gUlwP4CDi4iJBbVpZcqLVT29FxgGuBJGSzyOxotFG4LoO5X+M1350zmPA== | ||||
|  | ||||
| "@esbuild/netbsd-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.19.11.tgz#d633c09492a1721377f3bccedb2d821b911e813d" | ||||
|   integrity sha512-13jvrQZJc3P230OhU8xgwUnDeuC/9egsjTkXN49b3GcS5BKvJqZn86aGM8W9pd14Kd+u7HuFBMVtrNGhh6fHEQ== | ||||
|  | ||||
| "@esbuild/openbsd-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.19.11.tgz#17388c76e2f01125bf831a68c03a7ffccb65d1a2" | ||||
|   integrity sha512-ysyOGZuTp6SNKPE11INDUeFVVQFrhcNDVUgSQVDzqsqX38DjhPEPATpid04LCoUr2WXhQTEZ8ct/EgJCUDpyNw== | ||||
|  | ||||
| "@esbuild/sunos-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.19.11.tgz#e320636f00bb9f4fdf3a80e548cb743370d41767" | ||||
|   integrity sha512-Hf+Sad9nVwvtxy4DXCZQqLpgmRTQqyFyhT3bZ4F2XlJCjxGmRFF0Shwn9rzhOYRB61w9VMXUkxlBy56dk9JJiQ== | ||||
|  | ||||
| "@esbuild/win32-arm64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.19.11.tgz#c778b45a496e90b6fc373e2a2bb072f1441fe0ee" | ||||
|   integrity sha512-0P58Sbi0LctOMOQbpEOvOL44Ne0sqbS0XWHMvvrg6NE5jQ1xguCSSw9jQeUk2lfrXYsKDdOe6K+oZiwKPilYPQ== | ||||
|  | ||||
| "@esbuild/win32-ia32@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.19.11.tgz#481a65fee2e5cce74ec44823e6b09ecedcc5194c" | ||||
|   integrity sha512-6YOrWS+sDJDmshdBIQU+Uoyh7pQKrdykdefC1avn76ss5c+RN6gut3LZA4E2cH5xUEp5/cA0+YxRaVtRAb0xBg== | ||||
|  | ||||
| "@esbuild/win32-x64@0.19.11": | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.19.11.tgz#a5d300008960bb39677c46bf16f53ec70d8dee04" | ||||
|   integrity sha512-vfkhltrjCAb603XaFhqhAF4LGDi2M4OrCRrFusyQ+iTLQ/o60QQXxc9cZC/FFpihBI9N1Grn6SMKVJ4KP7Fuiw== | ||||
|  | ||||
| "@rollup/rollup-android-arm-eabi@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.9.5.tgz#b752b6c88a14ccfcbdf3f48c577ccc3a7f0e66b9" | ||||
|   integrity sha512-idWaG8xeSRCfRq9KpRysDHJ/rEHBEXcHuJ82XY0yYFIWnLMjZv9vF/7DOq8djQ2n3Lk6+3qfSH8AqlmHlmi1MA== | ||||
|  | ||||
| "@rollup/rollup-android-arm64@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.9.5.tgz#33757c3a448b9ef77b6f6292d8b0ec45c87e9c1a" | ||||
|   integrity sha512-f14d7uhAMtsCGjAYwZGv6TwuS3IFaM4ZnGMUn3aCBgkcHAYErhV1Ad97WzBvS2o0aaDv4mVz+syiN0ElMyfBPg== | ||||
|  | ||||
| "@rollup/rollup-darwin-arm64@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.9.5.tgz#5234ba62665a3f443143bc8bcea9df2cc58f55fb" | ||||
|   integrity sha512-ndoXeLx455FffL68OIUrVr89Xu1WLzAG4n65R8roDlCoYiQcGGg6MALvs2Ap9zs7AHg8mpHtMpwC8jBBjZrT/w== | ||||
|  | ||||
| "@rollup/rollup-darwin-x64@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.9.5.tgz#981256c054d3247b83313724938d606798a919d1" | ||||
|   integrity sha512-UmElV1OY2m/1KEEqTlIjieKfVwRg0Zwg4PLgNf0s3glAHXBN99KLpw5A5lrSYCa1Kp63czTpVll2MAqbZYIHoA== | ||||
|  | ||||
| "@rollup/rollup-linux-arm-gnueabihf@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.9.5.tgz#120678a5a2b3a283a548dbb4d337f9187a793560" | ||||
|   integrity sha512-Q0LcU61v92tQB6ae+udZvOyZ0wfpGojtAKrrpAaIqmJ7+psq4cMIhT/9lfV6UQIpeItnq/2QDROhNLo00lOD1g== | ||||
|  | ||||
| "@rollup/rollup-linux-arm64-gnu@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.9.5.tgz#c99d857e2372ece544b6f60b85058ad259f64114" | ||||
|   integrity sha512-dkRscpM+RrR2Ee3eOQmRWFjmV/payHEOrjyq1VZegRUa5OrZJ2MAxBNs05bZuY0YCtpqETDy1Ix4i/hRqX98cA== | ||||
|  | ||||
| "@rollup/rollup-linux-arm64-musl@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.9.5.tgz#3064060f568a5718c2a06858cd6e6d24f2ff8632" | ||||
|   integrity sha512-QaKFVOzzST2xzY4MAmiDmURagWLFh+zZtttuEnuNn19AiZ0T3fhPyjPPGwLNdiDT82ZE91hnfJsUiDwF9DClIQ== | ||||
|  | ||||
| "@rollup/rollup-linux-riscv64-gnu@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.9.5.tgz#987d30b5d2b992fff07d055015991a57ff55fbad" | ||||
|   integrity sha512-HeGqmRJuyVg6/X6MpE2ur7GbymBPS8Np0S/vQFHDmocfORT+Zt76qu+69NUoxXzGqVP1pzaY6QIi0FJWLC3OPA== | ||||
|  | ||||
| "@rollup/rollup-linux-x64-gnu@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.9.5.tgz#85946ee4d068bd12197aeeec2c6f679c94978a49" | ||||
|   integrity sha512-Dq1bqBdLaZ1Gb/l2e5/+o3B18+8TI9ANlA1SkejZqDgdU/jK/ThYaMPMJpVMMXy2uRHvGKbkz9vheVGdq3cJfA== | ||||
|  | ||||
| "@rollup/rollup-linux-x64-musl@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.9.5.tgz#fe0b20f9749a60eb1df43d20effa96c756ddcbd4" | ||||
|   integrity sha512-ezyFUOwldYpj7AbkwyW9AJ203peub81CaAIVvckdkyH8EvhEIoKzaMFJj0G4qYJ5sw3BpqhFrsCc30t54HV8vg== | ||||
|  | ||||
| "@rollup/rollup-win32-arm64-msvc@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.9.5.tgz#422661ef0e16699a234465d15b2c1089ef963b2a" | ||||
|   integrity sha512-aHSsMnUw+0UETB0Hlv7B/ZHOGY5bQdwMKJSzGfDfvyhnpmVxLMGnQPGNE9wgqkLUs3+gbG1Qx02S2LLfJ5GaRQ== | ||||
|  | ||||
| "@rollup/rollup-win32-ia32-msvc@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.9.5.tgz#7b73a145891c202fbcc08759248983667a035d85" | ||||
|   integrity sha512-AiqiLkb9KSf7Lj/o1U3SEP9Zn+5NuVKgFdRIZkvd4N0+bYrTOovVd0+LmYCPQGbocT4kvFyK+LXCDiXPBF3fyA== | ||||
|  | ||||
| "@rollup/rollup-win32-x64-msvc@4.9.5": | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.9.5.tgz#10491ccf4f63c814d4149e0316541476ea603602" | ||||
|   integrity sha512-1q+mykKE3Vot1kaFJIDoUFv5TuW+QQVaf2FmTT9krg86pQrGStOSJJ0Zil7CFagyxDuouTepzt5Y5TVzyajOdQ== | ||||
|  | ||||
| "@types/estree@1.0.5": | ||||
|   version "1.0.5" | ||||
|   resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" | ||||
|   integrity sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw== | ||||
|  | ||||
| esbuild@^0.19.3: | ||||
|   version "0.19.11" | ||||
|   resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.19.11.tgz#4a02dca031e768b5556606e1b468fe72e3325d96" | ||||
|   integrity sha512-HJ96Hev2hX/6i5cDVwcqiJBBtuo9+FeIJOtZ9W1kA5M6AMJRHUZlpYZ1/SbEwtO0ioNAW8rUooVpC/WehY2SfA== | ||||
|   optionalDependencies: | ||||
|     "@esbuild/aix-ppc64" "0.19.11" | ||||
|     "@esbuild/android-arm" "0.19.11" | ||||
|     "@esbuild/android-arm64" "0.19.11" | ||||
|     "@esbuild/android-x64" "0.19.11" | ||||
|     "@esbuild/darwin-arm64" "0.19.11" | ||||
|     "@esbuild/darwin-x64" "0.19.11" | ||||
|     "@esbuild/freebsd-arm64" "0.19.11" | ||||
|     "@esbuild/freebsd-x64" "0.19.11" | ||||
|     "@esbuild/linux-arm" "0.19.11" | ||||
|     "@esbuild/linux-arm64" "0.19.11" | ||||
|     "@esbuild/linux-ia32" "0.19.11" | ||||
|     "@esbuild/linux-loong64" "0.19.11" | ||||
|     "@esbuild/linux-mips64el" "0.19.11" | ||||
|     "@esbuild/linux-ppc64" "0.19.11" | ||||
|     "@esbuild/linux-riscv64" "0.19.11" | ||||
|     "@esbuild/linux-s390x" "0.19.11" | ||||
|     "@esbuild/linux-x64" "0.19.11" | ||||
|     "@esbuild/netbsd-x64" "0.19.11" | ||||
|     "@esbuild/openbsd-x64" "0.19.11" | ||||
|     "@esbuild/sunos-x64" "0.19.11" | ||||
|     "@esbuild/win32-arm64" "0.19.11" | ||||
|     "@esbuild/win32-ia32" "0.19.11" | ||||
|     "@esbuild/win32-x64" "0.19.11" | ||||
|  | ||||
| fsevents@~2.3.2, fsevents@~2.3.3: | ||||
|   version "2.3.3" | ||||
|   resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" | ||||
|   integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== | ||||
|  | ||||
| "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.7: | ||||
|   version "3.3.7" | ||||
|   resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.7.tgz#d0c301a691bc8d54efa0a2226ccf3fe2fd656bd8" | ||||
|   integrity sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g== | ||||
|  | ||||
| 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.32: | ||||
|   version "8.4.33" | ||||
|   resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.33.tgz#1378e859c9f69bf6f638b990a0212f43e2aaa742" | ||||
|   integrity sha512-Kkpbhhdjw2qQs2O2DGX+8m5OVqEcbB9HRBvuYM9pgrjEFUg30A9LmXNlTAUj4S9kgtGyrMbTzVjH7E+s5Re2yg== | ||||
|   dependencies: | ||||
|     nanoid "^3.3.7" | ||||
|     picocolors "^1.0.0" | ||||
|     source-map-js "^1.0.2" | ||||
|  | ||||
| react-dom@18.2.0: | ||||
|   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.2.0: | ||||
|   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" | ||||
|  | ||||
| rollup@^4.2.0: | ||||
|   version "4.9.5" | ||||
|   resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.9.5.tgz#62999462c90f4c8b5d7c38fc7161e63b29101b05" | ||||
|   integrity sha512-E4vQW0H/mbNMw2yLSqJyjtkHY9dslf/p0zuT1xehNRqUTBOFMqEjguDvqhXr7N7r/4ttb2jr4T41d3dncmIgbQ== | ||||
|   dependencies: | ||||
|     "@types/estree" "1.0.5" | ||||
|   optionalDependencies: | ||||
|     "@rollup/rollup-android-arm-eabi" "4.9.5" | ||||
|     "@rollup/rollup-android-arm64" "4.9.5" | ||||
|     "@rollup/rollup-darwin-arm64" "4.9.5" | ||||
|     "@rollup/rollup-darwin-x64" "4.9.5" | ||||
|     "@rollup/rollup-linux-arm-gnueabihf" "4.9.5" | ||||
|     "@rollup/rollup-linux-arm64-gnu" "4.9.5" | ||||
|     "@rollup/rollup-linux-arm64-musl" "4.9.5" | ||||
|     "@rollup/rollup-linux-riscv64-gnu" "4.9.5" | ||||
|     "@rollup/rollup-linux-x64-gnu" "4.9.5" | ||||
|     "@rollup/rollup-linux-x64-musl" "4.9.5" | ||||
|     "@rollup/rollup-win32-arm64-msvc" "4.9.5" | ||||
|     "@rollup/rollup-win32-ia32-msvc" "4.9.5" | ||||
|     "@rollup/rollup-win32-x64-msvc" "4.9.5" | ||||
|     fsevents "~2.3.2" | ||||
|  | ||||
| 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== | ||||
|  | ||||
| vite@5.0.6: | ||||
|   version "5.0.6" | ||||
|   resolved "https://registry.yarnpkg.com/vite/-/vite-5.0.6.tgz#f9e13503a4c5ccd67312c67803dec921f3bdea7c" | ||||
|   integrity sha512-MD3joyAEBtV7QZPl2JVVUai6zHms3YOmLR+BpMzLlX2Yzjfcc4gTgNi09d/Rua3F4EtC8zdwPU8eQYyib4vVMQ== | ||||
|   dependencies: | ||||
|     esbuild "^0.19.3" | ||||
|     postcss "^8.4.32" | ||||
|     rollup "^4.2.0" | ||||
|   optionalDependencies: | ||||
|     fsevents "~2.3.3" | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,15 +1,14 @@ | ||||
| import { useEffect, useState } from "react"; | ||||
| import { debounce, getVersion, nFormatter } from "../packages/excalidraw/utils"; | ||||
| import { debounce, getVersion, nFormatter } from "../src/utils"; | ||||
| import { | ||||
|   getElementsStorageSize, | ||||
|   getTotalStorageSize, | ||||
| } from "./data/localStorage"; | ||||
| import { DEFAULT_VERSION } from "../packages/excalidraw/constants"; | ||||
| import { t } from "../packages/excalidraw/i18n"; | ||||
| import { copyTextToSystemClipboard } from "../packages/excalidraw/clipboard"; | ||||
| import type { NonDeletedExcalidrawElement } from "../packages/excalidraw/element/types"; | ||||
| import type { UIAppState } from "../packages/excalidraw/types"; | ||||
| import { Stats } from "../packages/excalidraw"; | ||||
| 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 }; | ||||
|  | ||||
| @@ -52,33 +51,39 @@ const CustomStats = (props: Props) => { | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <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> | ||||
|     <> | ||||
|       <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> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,222 +0,0 @@ | ||||
| import { useLayoutEffect, useRef } from "react"; | ||||
| import { STORAGE_KEYS } from "./app_constants"; | ||||
| import { LocalData } from "./data/LocalData"; | ||||
| import type { | ||||
|   FileId, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../packages/excalidraw/element/types"; | ||||
| import type { AppState, BinaryFileData } from "../packages/excalidraw/types"; | ||||
| import { ExcalidrawError } from "../packages/excalidraw/errors"; | ||||
| import { base64urlToString } from "../packages/excalidraw/data/encode"; | ||||
|  | ||||
| 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,25 +0,0 @@ | ||||
| import LanguageDetector from "i18next-browser-languagedetector"; | ||||
| import { defaultLang, languages } from "../../packages/excalidraw"; | ||||
|  | ||||
| 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; | ||||
| }; | ||||
| @@ -1,15 +0,0 @@ | ||||
| import { atom, useAtom } from "jotai"; | ||||
| import { useEffect } from "react"; | ||||
| 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,17 +15,11 @@ 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_SUBTYPES { | ||||
|   INVALID_RESPONSE = "INVALID_RESPONSE", | ||||
| export enum WS_SCENE_EVENT_TYPES { | ||||
|   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 = { | ||||
| @@ -39,15 +33,10 @@ 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,51 +1,39 @@ | ||||
| import throttle from "lodash.throttle"; | ||||
| import { PureComponent } from "react"; | ||||
| import type { | ||||
|   BinaryFileData, | ||||
|   ExcalidrawImperativeAPI, | ||||
|   SocketId, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import { ErrorDialog } from "../../packages/excalidraw/components/ErrorDialog"; | ||||
| import { APP_NAME, ENV, EVENT } from "../../packages/excalidraw/constants"; | ||||
| import type { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
|   InitializedExcalidrawImageElement, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| 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 { | ||||
|   StoreAction, | ||||
|   getSceneVersion, | ||||
|   restoreElements, | ||||
|   zoomToFitBounds, | ||||
|   reconcileElements, | ||||
| } from "../../packages/excalidraw"; | ||||
| import type { Collaborator, Gesture } from "../../packages/excalidraw/types"; | ||||
| } from "../../src/packages/excalidraw/index"; | ||||
| import { Collaborator, Gesture } from "../../src/types"; | ||||
| import { | ||||
|   assertNever, | ||||
|   preventUnload, | ||||
|   resolvablePromise, | ||||
|   throttleRAF, | ||||
| } from "../../packages/excalidraw/utils"; | ||||
|   withBatchedUpdates, | ||||
| } from "../../src/utils"; | ||||
| import { | ||||
|   CURSOR_SYNC_TIMEOUT, | ||||
|   FILE_UPLOAD_MAX_BYTES, | ||||
|   FIREBASE_STORAGE_PREFIXES, | ||||
|   INITIAL_SCENE_UPDATE_TIMEOUT, | ||||
|   LOAD_IMAGES_TIMEOUT, | ||||
|   WS_SUBTYPES, | ||||
|   WS_SCENE_EVENT_TYPES, | ||||
|   SYNC_FULL_SCENE_INTERVAL_MS, | ||||
|   WS_EVENTS, | ||||
| } from "../app_constants"; | ||||
| import type { | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
| import { | ||||
|   generateCollaborationLinkData, | ||||
|   getCollaborationLink, | ||||
|   getCollabServer, | ||||
|   getSyncableElements, | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
| import { | ||||
|   isSavedToFirebase, | ||||
| @@ -59,51 +47,42 @@ import { | ||||
|   saveUsernameToLocalStorage, | ||||
| } from "../data/localStorage"; | ||||
| import Portal from "./Portal"; | ||||
| import { t } from "../../packages/excalidraw/i18n"; | ||||
| import { UserIdleState } from "../../packages/excalidraw/types"; | ||||
| import { | ||||
|   IDLE_THRESHOLD, | ||||
|   ACTIVE_THRESHOLD, | ||||
| } from "../../packages/excalidraw/constants"; | ||||
| 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 "../../packages/excalidraw/errors"; | ||||
| import { AbortError } from "../../src/errors"; | ||||
| import { | ||||
|   isImageElement, | ||||
|   isInitializedImageElement, | ||||
| } from "../../packages/excalidraw/element/typeChecks"; | ||||
| import { newElementWith } from "../../packages/excalidraw/element/mutateElement"; | ||||
| import { decryptData } from "../../packages/excalidraw/data/encryption"; | ||||
| } 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 } from "jotai"; | ||||
| import { atom, useAtom } from "jotai"; | ||||
| import { appJotaiStore } from "../app-jotai"; | ||||
| import type { Mutable, ValueOf } from "../../packages/excalidraw/utility-types"; | ||||
| import { getVisibleSceneBounds } from "../../packages/excalidraw/element/bounds"; | ||||
| import { withBatchedUpdates } from "../../packages/excalidraw/reactUtils"; | ||||
| import { collabErrorIndicatorAtom } from "./CollabError"; | ||||
| import type { | ||||
|   ReconciledExcalidrawElement, | ||||
|   RemoteExcalidrawElement, | ||||
| } from "../../packages/excalidraw/data/reconcile"; | ||||
|  | ||||
| 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 | null; | ||||
|   /** errors related to saving */ | ||||
|   dialogNotifiedErrors: Record<string, boolean>; | ||||
|   errorMessage: string; | ||||
|   username: string; | ||||
|   activeRoomLink: string | null; | ||||
|   activeRoomLink: string; | ||||
| } | ||||
|  | ||||
| export const activeRoomLinkAtom = atom<string | null>(null); | ||||
|  | ||||
| type CollabInstance = InstanceType<typeof Collab>; | ||||
|  | ||||
| export interface CollabAPI { | ||||
| @@ -114,34 +93,32 @@ export interface CollabAPI { | ||||
|   stopCollaboration: CollabInstance["stopCollaboration"]; | ||||
|   syncElements: CollabInstance["syncElements"]; | ||||
|   fetchImageFilesFromFirebase: CollabInstance["fetchImageFilesFromFirebase"]; | ||||
|   setUsername: CollabInstance["setUsername"]; | ||||
|   getUsername: CollabInstance["getUsername"]; | ||||
|   getActiveRoomLink: CollabInstance["getActiveRoomLink"]; | ||||
|   setCollabError: CollabInstance["setErrorDialog"]; | ||||
|   setUsername: (username: string) => void; | ||||
| } | ||||
|  | ||||
| interface CollabProps { | ||||
| interface PublicProps { | ||||
|   excalidrawAPI: ExcalidrawImperativeAPI; | ||||
| } | ||||
|  | ||||
| class Collab extends PureComponent<CollabProps, CollabState> { | ||||
| type Props = PublicProps & { modalIsShown: boolean }; | ||||
|  | ||||
| class Collab extends PureComponent<Props, CollabState> { | ||||
|   portal: Portal; | ||||
|   fileManager: FileManager; | ||||
|   excalidrawAPI: CollabProps["excalidrawAPI"]; | ||||
|   excalidrawAPI: Props["excalidrawAPI"]; | ||||
|   activeIntervalId: number | null; | ||||
|   idleTimeoutId: number | null; | ||||
|  | ||||
|   private socketInitializationTimer?: number; | ||||
|   private lastBroadcastedOrReceivedSceneVersion: number = -1; | ||||
|   private collaborators = new Map<SocketId, Collaborator>(); | ||||
|   private collaborators = new Map<string, Collaborator>(); | ||||
|  | ||||
|   constructor(props: CollabProps) { | ||||
|   constructor(props: Props) { | ||||
|     super(props); | ||||
|     this.state = { | ||||
|       errorMessage: null, | ||||
|       dialogNotifiedErrors: {}, | ||||
|       errorMessage: "", | ||||
|       username: importUsernameFromLocalStorage() || "", | ||||
|       activeRoomLink: null, | ||||
|       activeRoomLink: "", | ||||
|     }; | ||||
|     this.portal = new Portal(this); | ||||
|     this.fileManager = new FileManager({ | ||||
| @@ -159,7 +136,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|           throw new AbortError(); | ||||
|         } | ||||
|  | ||||
|         const { savedFiles, erroredFiles } = await saveFilesToFirebase({ | ||||
|         return saveFilesToFirebase({ | ||||
|           prefix: `${FIREBASE_STORAGE_PREFIXES.collabFiles}/${roomId}`, | ||||
|           files: await encodeFilesForUpload({ | ||||
|             files: addedFiles, | ||||
| @@ -167,29 +144,6 @@ class Collab extends PureComponent<CollabProps, 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; | ||||
| @@ -197,28 +151,12 @@ class Collab extends PureComponent<CollabProps, 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 = { | ||||
| @@ -229,9 +167,6 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       fetchImageFilesFromFirebase: this.fetchImageFilesFromFirebase, | ||||
|       stopCollaboration: this.stopCollaboration, | ||||
|       setUsername: this.setUsername, | ||||
|       getUsername: this.getUsername, | ||||
|       getActiveRoomLink: this.getActiveRoomLink, | ||||
|       setCollabError: this.setErrorDialog, | ||||
|     }; | ||||
|  | ||||
|     appJotaiStore.set(collabAPIAtom, collabAPI); | ||||
| @@ -269,7 +204,6 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       window.clearTimeout(this.idleTimeoutId); | ||||
|       this.idleTimeoutId = null; | ||||
|     } | ||||
|     this.onUmmount?.(); | ||||
|   } | ||||
|  | ||||
|   isCollaborating = () => appJotaiStore.get(isCollaboratingAtom)!; | ||||
| @@ -304,39 +238,24 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     syncableElements: readonly SyncableExcalidrawElement[], | ||||
|   ) => { | ||||
|     try { | ||||
|       const storedElements = await saveToFirebase( | ||||
|       const savedData = await saveToFirebase( | ||||
|         this.portal, | ||||
|         syncableElements, | ||||
|         this.excalidrawAPI.getAppState(), | ||||
|       ); | ||||
|  | ||||
|       this.resetErrorIndicator(); | ||||
|  | ||||
|       if (this.isCollaborating() && storedElements) { | ||||
|         this.handleRemoteSceneUpdate(this._reconcileElements(storedElements)); | ||||
|       if (this.isCollaborating() && savedData && savedData.reconciledElements) { | ||||
|         this.handleRemoteSceneUpdate( | ||||
|           this.reconcileElements(savedData.reconciledElements), | ||||
|         ); | ||||
|       } | ||||
|     } catch (error: any) { | ||||
|       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); | ||||
|       } | ||||
|  | ||||
|       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"), | ||||
|       }); | ||||
|       console.error(error); | ||||
|     } | ||||
|   }; | ||||
| @@ -345,7 +264,6 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     this.queueBroadcastAllElements.cancel(); | ||||
|     this.queueSaveToFirebase.cancel(); | ||||
|     this.loadImageFiles.cancel(); | ||||
|     this.resetErrorIndicator(true); | ||||
|  | ||||
|     this.saveCollabRoomToFirebase( | ||||
|       getSyncableElements( | ||||
| @@ -384,7 +302,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|  | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         elements, | ||||
|         storeAction: StoreAction.UPDATE, | ||||
|         commitToHistory: false, | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
| @@ -395,7 +313,9 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     this.fileManager.reset(); | ||||
|     if (!opts?.isUnload) { | ||||
|       this.setIsCollaborating(false); | ||||
|       this.setActiveRoomLink(null); | ||||
|       this.setState({ | ||||
|         activeRoomLink: "", | ||||
|       }); | ||||
|       this.collaborators = new Map(); | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         collaborators: this.collaborators, | ||||
| @@ -419,7 +339,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       .filter((element) => { | ||||
|         return ( | ||||
|           isInitializedImageElement(element) && | ||||
|           !this.fileManager.isFileTracked(element.fileId) && | ||||
|           !this.fileManager.isFileHandled(element.fileId) && | ||||
|           !element.isDeleted && | ||||
|           (opts.forceFetchFiles | ||||
|             ? element.status !== "pending" || | ||||
| @@ -436,7 +356,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     iv: Uint8Array, | ||||
|     encryptedData: ArrayBuffer, | ||||
|     decryptionKey: string, | ||||
|   ): Promise<ValueOf<SocketUpdateDataSource>> => { | ||||
|   ) => { | ||||
|     try { | ||||
|       const decrypted = await decryptData(iv, encryptedData, decryptionKey); | ||||
|  | ||||
| @@ -448,7 +368,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       window.alert(t("alerts.decryptFailed")); | ||||
|       console.error(error); | ||||
|       return { | ||||
|         type: WS_SUBTYPES.INVALID_RESPONSE, | ||||
|         type: "INVALID_RESPONSE", | ||||
|       }; | ||||
|     } | ||||
|   }; | ||||
| @@ -457,11 +377,11 @@ class Collab extends PureComponent<CollabProps, 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.setUsername(username); | ||||
|         this.onUsernameChange(username); | ||||
|       }); | ||||
|     } | ||||
|  | ||||
| @@ -483,11 +403,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     // TODO: `ImportedDataState` type here seems abused | ||||
|     const scenePromise = resolvablePromise< | ||||
|       | (ImportedDataState & { elements: readonly OrderedExcalidrawElement[] }) | ||||
|       | null | ||||
|     >(); | ||||
|     const scenePromise = resolvablePromise<ImportedDataState | null>(); | ||||
|  | ||||
|     this.setIsCollaborating(true); | ||||
|     LocalData.pauseSave("collaboration"); | ||||
| @@ -507,9 +423,13 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     this.fallbackInitializationHandler = fallbackInitializationHandler; | ||||
|  | ||||
|     try { | ||||
|       const socketServerData = await getCollabServer(); | ||||
|  | ||||
|       this.portal.socket = this.portal.open( | ||||
|         socketIOClient(import.meta.env.VITE_APP_WS_SERVER_URL, { | ||||
|           transports: ["websocket", "polling"], | ||||
|         socketIOClient(socketServerData.url, { | ||||
|           transports: socketServerData.polling | ||||
|             ? ["websocket", "polling"] | ||||
|             : ["websocket"], | ||||
|         }), | ||||
|         roomId, | ||||
|         roomKey, | ||||
| @@ -518,7 +438,7 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       this.portal.socket.once("connect_error", fallbackInitializationHandler); | ||||
|     } catch (error: any) { | ||||
|       console.error(error); | ||||
|       this.setErrorDialog(error.message); | ||||
|       this.setState({ errorMessage: error.message }); | ||||
|       return null; | ||||
|     } | ||||
|  | ||||
| @@ -529,13 +449,14 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|         } | ||||
|         return element; | ||||
|       }); | ||||
|       // remove deleted elements from elements array to ensure we don't | ||||
|       // remove deleted elements from elements array & history 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, | ||||
|         storeAction: StoreAction.UPDATE, | ||||
|         commitToHistory: true, | ||||
|       }); | ||||
|  | ||||
|       this.saveCollabRoomToFirebase(getSyncableElements(elements)); | ||||
| @@ -563,15 +484,16 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|         ); | ||||
|  | ||||
|         switch (decryptedData.type) { | ||||
|           case WS_SUBTYPES.INVALID_RESPONSE: | ||||
|           case "INVALID_RESPONSE": | ||||
|             return; | ||||
|           case WS_SUBTYPES.INIT: { | ||||
|           case WS_SCENE_EVENT_TYPES.INIT: { | ||||
|             if (!this.portal.socketInitialized) { | ||||
|               this.initializeRoom({ fetchScene: false }); | ||||
|               const remoteElements = decryptedData.payload.elements; | ||||
|               const reconciledElements = | ||||
|                 this._reconcileElements(remoteElements); | ||||
|               this.handleRemoteSceneUpdate(reconciledElements); | ||||
|               const reconciledElements = this.reconcileElements(remoteElements); | ||||
|               this.handleRemoteSceneUpdate(reconciledElements, { | ||||
|                 init: true, | ||||
|               }); | ||||
|               // noop if already resolved via init from firebase | ||||
|               scenePromise.resolve({ | ||||
|                 elements: reconciledElements, | ||||
| @@ -580,76 +502,42 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|             } | ||||
|             break; | ||||
|           } | ||||
|           case WS_SUBTYPES.UPDATE: | ||||
|           case WS_SCENE_EVENT_TYPES.UPDATE: | ||||
|             this.handleRemoteSceneUpdate( | ||||
|               this._reconcileElements(decryptedData.payload.elements), | ||||
|               this.reconcileElements(decryptedData.payload.elements), | ||||
|             ); | ||||
|             break; | ||||
|           case WS_SUBTYPES.MOUSE_LOCATION: { | ||||
|           case "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; | ||||
|  | ||||
|             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; | ||||
|             } | ||||
|  | ||||
|             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.excalidrawAPI.updateScene({ | ||||
|               appState: zoomToFitBounds({ | ||||
|                 appState, | ||||
|                 bounds: sceneBounds, | ||||
|                 fitToViewport: true, | ||||
|                 viewportZoomFactor: 1, | ||||
|               }).appState, | ||||
|               collaborators, | ||||
|             }); | ||||
|  | ||||
|             break; | ||||
|           } | ||||
|  | ||||
|           case WS_SUBTYPES.IDLE_STATUS: { | ||||
|           case "IDLE_STATUS": { | ||||
|             const { userState, socketId, username } = decryptedData.payload; | ||||
|             this.updateCollaborator(socketId, { | ||||
|               userState, | ||||
|               username, | ||||
|             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); | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|     ); | ||||
| @@ -665,20 +553,11 @@ class Collab extends PureComponent<CollabProps, 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.setActiveRoomLink(window.location.href); | ||||
|     this.setState({ | ||||
|       activeRoomLink: window.location.href, | ||||
|     }); | ||||
|  | ||||
|     return scenePromise; | ||||
|   }; | ||||
| @@ -730,15 +609,17 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     return null; | ||||
|   }; | ||||
|  | ||||
|   private _reconcileElements = ( | ||||
|   private reconcileElements = ( | ||||
|     remoteElements: readonly ExcalidrawElement[], | ||||
|   ): ReconciledExcalidrawElement[] => { | ||||
|   ): ReconciledElements => { | ||||
|     const localElements = this.getSceneElementsIncludingDeleted(); | ||||
|     const appState = this.excalidrawAPI.getAppState(); | ||||
|     const restoredRemoteElements = restoreElements(remoteElements, null); | ||||
|     const reconciledElements = reconcileElements( | ||||
|  | ||||
|     remoteElements = restoreElements(remoteElements, null); | ||||
|  | ||||
|     const reconciledElements = _reconcileElements( | ||||
|       localElements, | ||||
|       restoredRemoteElements as RemoteExcalidrawElement[], | ||||
|       remoteElements, | ||||
|       appState, | ||||
|     ); | ||||
|  | ||||
| @@ -769,13 +650,20 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|   }, LOAD_IMAGES_TIMEOUT); | ||||
|  | ||||
|   private handleRemoteSceneUpdate = ( | ||||
|     elements: ReconciledExcalidrawElement[], | ||||
|     elements: ReconciledElements, | ||||
|     { init = false }: { init?: boolean } = {}, | ||||
|   ) => { | ||||
|     this.excalidrawAPI.updateScene({ | ||||
|       elements, | ||||
|       storeAction: StoreAction.UPDATE, | ||||
|       commitToHistory: !!init, | ||||
|     }); | ||||
|  | ||||
|     // 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(); | ||||
|   }; | ||||
|  | ||||
| @@ -833,39 +721,20 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     document.addEventListener(EVENT.VISIBILITY_CHANGE, this.onVisibilityChange); | ||||
|   }; | ||||
|  | ||||
|   setCollaborators(sockets: SocketId[]) { | ||||
|   setCollaborators(sockets: string[]) { | ||||
|     const collaborators: InstanceType<typeof Collab>["collaborators"] = | ||||
|       new Map(); | ||||
|     for (const socketId of sockets) { | ||||
|       collaborators.set( | ||||
|         socketId, | ||||
|         Object.assign({}, this.collaborators.get(socketId), { | ||||
|           isCurrentUser: socketId === this.portal.socket?.id, | ||||
|         }), | ||||
|       ); | ||||
|       if (this.collaborators.has(socketId)) { | ||||
|         collaborators.set(socketId, this.collaborators.get(socketId)!); | ||||
|       } else { | ||||
|         collaborators.set(socketId, {}); | ||||
|       } | ||||
|     } | ||||
|     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; | ||||
|   }; | ||||
| @@ -891,42 +760,29 @@ class Collab extends PureComponent<CollabProps, 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 OrderedExcalidrawElement[]) => { | ||||
|   broadcastElements = (elements: readonly ExcalidrawElement[]) => { | ||||
|     if ( | ||||
|       getSceneVersion(elements) > | ||||
|       this.getLastBroadcastedOrReceivedSceneVersion() | ||||
|     ) { | ||||
|       this.portal.broadcastScene(WS_SUBTYPES.UPDATE, elements, false); | ||||
|       this.portal.broadcastScene(WS_SCENE_EVENT_TYPES.UPDATE, elements, false); | ||||
|       this.lastBroadcastedOrReceivedSceneVersion = getSceneVersion(elements); | ||||
|       this.queueBroadcastAllElements(); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   syncElements = (elements: readonly OrderedExcalidrawElement[]) => { | ||||
|   syncElements = (elements: readonly ExcalidrawElement[]) => { | ||||
|     this.broadcastElements(elements); | ||||
|     this.queueSaveToFirebase(); | ||||
|   }; | ||||
|  | ||||
|   queueBroadcastAllElements = throttle(() => { | ||||
|     this.portal.broadcastScene( | ||||
|       WS_SUBTYPES.UPDATE, | ||||
|       WS_SCENE_EVENT_TYPES.UPDATE, | ||||
|       this.excalidrawAPI.getSceneElementsIncludingDeleted(), | ||||
|       true, | ||||
|     ); | ||||
| @@ -952,49 +808,41 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|     { leading: false }, | ||||
|   ); | ||||
|  | ||||
|   handleClose = () => { | ||||
|     appJotaiStore.set(collabDialogShownAtom, false); | ||||
|   }; | ||||
|  | ||||
|   setUsername = (username: string) => { | ||||
|     this.setState({ username }); | ||||
|   }; | ||||
|  | ||||
|   onUsernameChange = (username: string) => { | ||||
|     this.setUsername(username); | ||||
|     saveUsernameToLocalStorage(username); | ||||
|   }; | ||||
|  | ||||
|   getUsername = () => this.state.username; | ||||
|  | ||||
|   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; | ||||
|     const { username, errorMessage, activeRoomLink } = this.state; | ||||
|  | ||||
|     const { modalIsShown } = this.props; | ||||
|  | ||||
|     return ( | ||||
|       <> | ||||
|         {errorMessage != null && ( | ||||
|           <ErrorDialog onClose={() => this.setErrorDialog(null)}> | ||||
|         {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} | ||||
|           </ErrorDialog> | ||||
|         )} | ||||
| @@ -1013,6 +861,11 @@ if (import.meta.env.MODE === ENV.TEST || import.meta.env.DEV) { | ||||
|   window.collab = window.collab || ({} as Window["collab"]); | ||||
| } | ||||
|  | ||||
| export default Collab; | ||||
| const _Collab: React.FC<PublicProps> = (props) => { | ||||
|   const [collabDialogShown] = useAtom(collabDialogShownAtom); | ||||
|   return <Collab {...props} modalIsShown={collabDialogShown} />; | ||||
| }; | ||||
|  | ||||
| export default _Collab; | ||||
|  | ||||
| export type TCollabClass = Collab; | ||||
|   | ||||
| @@ -1,35 +0,0 @@ | ||||
| @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); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,54 +0,0 @@ | ||||
| import { Tooltip } from "../../packages/excalidraw/components/Tooltip"; | ||||
| import { warning } from "../../packages/excalidraw/components/icons"; | ||||
| import clsx from "clsx"; | ||||
| import { useEffect, useRef, useState } from "react"; | ||||
|  | ||||
| import "./CollabError.scss"; | ||||
| import { atom } from "jotai"; | ||||
|  | ||||
| type ErrorIndicator = { | ||||
|   message: string | null; | ||||
|   /** used to rerun the useEffect responsible for animation */ | ||||
|   nonce: number; | ||||
| }; | ||||
|  | ||||
| export const collabErrorIndicatorAtom = atom<ErrorIndicator>({ | ||||
|   message: null, | ||||
|   nonce: 0, | ||||
| }); | ||||
|  | ||||
| const CollabError = ({ collabError }: { collabError: ErrorIndicator }) => { | ||||
|   const [isAnimating, setIsAnimating] = useState(false); | ||||
|   const clearAnimationRef = useRef<string | number | NodeJS.Timeout>(); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     setIsAnimating(true); | ||||
|     clearAnimationRef.current = setTimeout(() => { | ||||
|       setIsAnimating(false); | ||||
|     }, 1000); | ||||
|  | ||||
|     return () => { | ||||
|       clearTimeout(clearAnimationRef.current); | ||||
|     }; | ||||
|   }, [collabError.message, collabError.nonce]); | ||||
|  | ||||
|   if (!collabError.message) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <Tooltip label={collabError.message} long={true}> | ||||
|       <div | ||||
|         className={clsx("collab-errors-button", { | ||||
|           "collab-errors-button-shake": isAnimating, | ||||
|         })} | ||||
|       > | ||||
|         {warning} | ||||
|       </div> | ||||
|     </Tooltip> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| CollabError.displayName = "CollabError"; | ||||
|  | ||||
| export default CollabError; | ||||
| @@ -1,29 +1,28 @@ | ||||
| import type { | ||||
| import { | ||||
|   isSyncableElement, | ||||
|   SocketUpdateData, | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
| import { isSyncableElement } from "../data"; | ||||
|  | ||||
| import type { TCollabClass } from "./Collab"; | ||||
| import { TCollabClass } from "./Collab"; | ||||
|  | ||||
| import type { OrderedExcalidrawElement } from "../../packages/excalidraw/element/types"; | ||||
| import { WS_EVENTS, FILE_UPLOAD_TIMEOUT, WS_SUBTYPES } from "../app_constants"; | ||||
| import type { | ||||
|   OnUserFollowedPayload, | ||||
|   SocketId, | ||||
|   UserIdleState, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import { trackEvent } from "../../packages/excalidraw/analytics"; | ||||
| import { ExcalidrawElement } from "../../src/element/types"; | ||||
| import { | ||||
|   WS_EVENTS, | ||||
|   FILE_UPLOAD_TIMEOUT, | ||||
|   WS_SCENE_EVENT_TYPES, | ||||
| } from "../app_constants"; | ||||
| import { UserIdleState } from "../../src/types"; | ||||
| import { trackEvent } from "../../src/analytics"; | ||||
| import throttle from "lodash.throttle"; | ||||
| import { newElementWith } from "../../packages/excalidraw/element/mutateElement"; | ||||
| import { encryptData } from "../../packages/excalidraw/data/encryption"; | ||||
| import type { Socket } from "socket.io-client"; | ||||
| import { StoreAction } from "../../packages/excalidraw"; | ||||
| import { newElementWith } from "../../src/element/mutateElement"; | ||||
| import { BroadcastedExcalidrawElement } from "./reconciliation"; | ||||
| import { encryptData } from "../../src/data/encryption"; | ||||
| import { PRECEDING_ELEMENT_KEY } from "../../src/constants"; | ||||
|  | ||||
| class Portal { | ||||
|   collab: TCollabClass; | ||||
|   socket: Socket | null = null; | ||||
|   socket: SocketIOClient.Socket | null = null; | ||||
|   socketInitialized: boolean = false; // we don't want the socket to emit any updates until it is fully initialized | ||||
|   roomId: string | null = null; | ||||
|   roomKey: string | null = null; | ||||
| @@ -33,7 +32,7 @@ class Portal { | ||||
|     this.collab = collab; | ||||
|   } | ||||
|  | ||||
|   open(socket: Socket, id: string, key: string) { | ||||
|   open(socket: SocketIOClient.Socket, id: string, key: string) { | ||||
|     this.socket = socket; | ||||
|     this.roomId = id; | ||||
|     this.roomKey = key; | ||||
| @@ -47,12 +46,12 @@ class Portal { | ||||
|     }); | ||||
|     this.socket.on("new-user", async (_socketId: string) => { | ||||
|       this.broadcastScene( | ||||
|         WS_SUBTYPES.INIT, | ||||
|         WS_SCENE_EVENT_TYPES.INIT, | ||||
|         this.collab.getSceneElementsIncludingDeleted(), | ||||
|         /* syncAll */ true, | ||||
|       ); | ||||
|     }); | ||||
|     this.socket.on("room-user-change", (clients: SocketId[]) => { | ||||
|     this.socket.on("room-user-change", (clients: string[]) => { | ||||
|       this.collab.setCollaborators(clients); | ||||
|     }); | ||||
|  | ||||
| @@ -84,7 +83,6 @@ class Portal { | ||||
|   async _broadcastSocketData( | ||||
|     data: SocketUpdateData, | ||||
|     volatile: boolean = false, | ||||
|     roomId?: string, | ||||
|   ) { | ||||
|     if (this.isOpen()) { | ||||
|       const json = JSON.stringify(data); | ||||
| @@ -93,7 +91,7 @@ class Portal { | ||||
|  | ||||
|       this.socket?.emit( | ||||
|         volatile ? WS_EVENTS.SERVER_VOLATILE : WS_EVENTS.SERVER, | ||||
|         roomId ?? this.roomId, | ||||
|         this.roomId, | ||||
|         encryptedBuffer, | ||||
|         iv, | ||||
|       ); | ||||
| @@ -116,51 +114,52 @@ class Portal { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     let isChanged = false; | ||||
|     const newElements = this.collab.excalidrawAPI | ||||
|       .getSceneElementsIncludingDeleted() | ||||
|       .map((element) => { | ||||
|         if (this.collab.fileManager.shouldUpdateImageElementStatus(element)) { | ||||
|           isChanged = true; | ||||
|           // this will signal collaborators to pull image data from server | ||||
|           // (using mutation instead of newElementWith otherwise it'd break | ||||
|           // in-progress dragging) | ||||
|           return newElementWith(element, { status: "saved" }); | ||||
|         } | ||||
|         return element; | ||||
|       }); | ||||
|  | ||||
|     if (isChanged) { | ||||
|       this.collab.excalidrawAPI.updateScene({ | ||||
|         elements: newElements, | ||||
|         storeAction: StoreAction.UPDATE, | ||||
|       }); | ||||
|     } | ||||
|     this.collab.excalidrawAPI.updateScene({ | ||||
|       elements: this.collab.excalidrawAPI | ||||
|         .getSceneElementsIncludingDeleted() | ||||
|         .map((element) => { | ||||
|           if (this.collab.fileManager.shouldUpdateImageElementStatus(element)) { | ||||
|             // this will signal collaborators to pull image data from server | ||||
|             // (using mutation instead of newElementWith otherwise it'd break | ||||
|             // in-progress dragging) | ||||
|             return newElementWith(element, { status: "saved" }); | ||||
|           } | ||||
|           return element; | ||||
|         }), | ||||
|     }); | ||||
|   }, FILE_UPLOAD_TIMEOUT); | ||||
|  | ||||
|   broadcastScene = async ( | ||||
|     updateType: WS_SUBTYPES.INIT | WS_SUBTYPES.UPDATE, | ||||
|     elements: readonly OrderedExcalidrawElement[], | ||||
|     updateType: WS_SCENE_EVENT_TYPES.INIT | WS_SCENE_EVENT_TYPES.UPDATE, | ||||
|     allElements: readonly ExcalidrawElement[], | ||||
|     syncAll: boolean, | ||||
|   ) => { | ||||
|     if (updateType === WS_SUBTYPES.INIT && !syncAll) { | ||||
|     if (updateType === WS_SCENE_EVENT_TYPES.INIT && !syncAll) { | ||||
|       throw new Error("syncAll must be true when sending SCENE.INIT"); | ||||
|     } | ||||
|  | ||||
|     // sync out only the elements we think we need to to save bandwidth. | ||||
|     // periodically we'll resync the whole thing to make sure no one diverges | ||||
|     // due to a dropped message (server goes down etc). | ||||
|     const syncableElements = elements.reduce((acc, element) => { | ||||
|       if ( | ||||
|         (syncAll || | ||||
|           !this.broadcastedElementVersions.has(element.id) || | ||||
|           element.version > this.broadcastedElementVersions.get(element.id)!) && | ||||
|         isSyncableElement(element) | ||||
|       ) { | ||||
|         acc.push(element); | ||||
|       } | ||||
|       return acc; | ||||
|     }, [] as SyncableExcalidrawElement[]); | ||||
|     const syncableElements = allElements.reduce( | ||||
|       (acc, element: BroadcastedExcalidrawElement, idx, elements) => { | ||||
|         if ( | ||||
|           (syncAll || | ||||
|             !this.broadcastedElementVersions.has(element.id) || | ||||
|             element.version > | ||||
|               this.broadcastedElementVersions.get(element.id)!) && | ||||
|           isSyncableElement(element) | ||||
|         ) { | ||||
|           acc.push({ | ||||
|             ...element, | ||||
|             // z-index info for the reconciler | ||||
|             [PRECEDING_ELEMENT_KEY]: idx === 0 ? "^" : elements[idx - 1]?.id, | ||||
|           }); | ||||
|         } | ||||
|         return acc; | ||||
|       }, | ||||
|       [] as BroadcastedExcalidrawElement[], | ||||
|     ); | ||||
|  | ||||
|     const data: SocketUpdateDataSource[typeof updateType] = { | ||||
|       type: updateType, | ||||
| @@ -184,9 +183,9 @@ class Portal { | ||||
|   broadcastIdleChange = (userState: UserIdleState) => { | ||||
|     if (this.socket?.id) { | ||||
|       const data: SocketUpdateDataSource["IDLE_STATUS"] = { | ||||
|         type: WS_SUBTYPES.IDLE_STATUS, | ||||
|         type: "IDLE_STATUS", | ||||
|         payload: { | ||||
|           socketId: this.socket.id as SocketId, | ||||
|           socketId: this.socket.id, | ||||
|           userState, | ||||
|           username: this.collab.state.username, | ||||
|         }, | ||||
| @@ -204,9 +203,9 @@ class Portal { | ||||
|   }) => { | ||||
|     if (this.socket?.id) { | ||||
|       const data: SocketUpdateDataSource["MOUSE_LOCATION"] = { | ||||
|         type: WS_SUBTYPES.MOUSE_LOCATION, | ||||
|         type: "MOUSE_LOCATION", | ||||
|         payload: { | ||||
|           socketId: this.socket.id as SocketId, | ||||
|           socketId: this.socket.id, | ||||
|           pointer: payload.pointer, | ||||
|           button: payload.button || "up", | ||||
|           selectedElementIds: | ||||
| @@ -214,43 +213,12 @@ class Portal { | ||||
|           username: this.collab.state.username, | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       return this._broadcastSocketData( | ||||
|         data as SocketUpdateData, | ||||
|         true, // volatile | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   broadcastVisibleSceneBounds = ( | ||||
|     payload: { | ||||
|       sceneBounds: SocketUpdateDataSource["USER_VISIBLE_SCENE_BOUNDS"]["payload"]["sceneBounds"]; | ||||
|     }, | ||||
|     roomId: string, | ||||
|   ) => { | ||||
|     if (this.socket?.id) { | ||||
|       const data: SocketUpdateDataSource["USER_VISIBLE_SCENE_BOUNDS"] = { | ||||
|         type: WS_SUBTYPES.USER_VISIBLE_SCENE_BOUNDS, | ||||
|         payload: { | ||||
|           socketId: this.socket.id as SocketId, | ||||
|           username: this.collab.state.username, | ||||
|           sceneBounds: payload.sceneBounds, | ||||
|         }, | ||||
|       }; | ||||
|  | ||||
|       return this._broadcastSocketData( | ||||
|         data as SocketUpdateData, | ||||
|         true, // volatile | ||||
|         roomId, | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   broadcastUserFollowed = (payload: OnUserFollowedPayload) => { | ||||
|     if (this.socket?.id) { | ||||
|       this.socket.emit(WS_EVENTS.USER_FOLLOW_CHANGE, payload); | ||||
|     } | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export default Portal; | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| @import "../../packages/excalidraw/css/variables.module.scss"; | ||||
| @import "../../src/css/variables.module"; | ||||
| 
 | ||||
| .excalidraw { | ||||
|   .ShareDialog { | ||||
|   .RoomDialog { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     gap: 1.5rem; | ||||
| @@ -10,25 +10,8 @@ | ||||
|       height: calc(100vh - 5rem); | ||||
|     } | ||||
| 
 | ||||
|     &__separator { | ||||
|       border-top: 1px solid var(--dialog-border-color); | ||||
|       text-align: center; | ||||
|       display: flex; | ||||
|       justify-content: center; | ||||
|       align-items: center; | ||||
|       margin-top: 1em; | ||||
| 
 | ||||
|       span { | ||||
|         background: var(--island-bg-color); | ||||
|         padding: 0px 0.75rem; | ||||
|         transform: translateY(-1ch); | ||||
|         display: inline-flex; | ||||
|         line-height: 1; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     &__popover { | ||||
|       @keyframes ShareDialog__popover__scaleIn { | ||||
|       @keyframes RoomDialog__popover__scaleIn { | ||||
|         from { | ||||
|           opacity: 0; | ||||
|         } | ||||
| @@ -58,8 +41,8 @@ | ||||
|       font-size: 0.75rem; | ||||
|       line-height: 110%; | ||||
| 
 | ||||
|       background: var(--color-success); | ||||
|       color: var(--color-success-text); | ||||
|       background: var(--color-success-lighter); | ||||
|       color: var(--color-success); | ||||
| 
 | ||||
|       & > svg { | ||||
|         width: 0.875rem; | ||||
| @@ -67,10 +50,10 @@ | ||||
|       } | ||||
| 
 | ||||
|       transform-origin: var(--radix-popover-content-transform-origin); | ||||
|       animation: ShareDialog__popover__scaleIn 150ms ease-out; | ||||
|       animation: RoomDialog__popover__scaleIn 150ms ease-out; | ||||
|     } | ||||
| 
 | ||||
|     &__picker { | ||||
|     &__inactive { | ||||
|       font-family: "Assistant"; | ||||
| 
 | ||||
|       &__illustration { | ||||
| @@ -112,7 +95,7 @@ | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       &__button { | ||||
|       &__start_session { | ||||
|         display: flex; | ||||
| 
 | ||||
|         align-items: center; | ||||
							
								
								
									
										219
									
								
								excalidraw-app/collab/RoomDialog.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										219
									
								
								excalidraw-app/collab/RoomDialog.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,219 @@ | ||||
| import { useRef, useState } from "react"; | ||||
| import * as Popover from "@radix-ui/react-popover"; | ||||
|  | ||||
| import { copyTextToSystemClipboard } from "../../src/clipboard"; | ||||
| import { trackEvent } from "../../src/analytics"; | ||||
| import { getFrame } from "../../src/utils"; | ||||
| import { useI18n } from "../../src/i18n"; | ||||
| import { KEYS } from "../../src/keys"; | ||||
|  | ||||
| import { Dialog } from "../../src/components/Dialog"; | ||||
| import { | ||||
|   copyIcon, | ||||
|   playerPlayIcon, | ||||
|   playerStopFilledIcon, | ||||
|   share, | ||||
|   shareIOS, | ||||
|   shareWindows, | ||||
|   tablerCheckIcon, | ||||
| } from "../../src/components/icons"; | ||||
| import { TextField } from "../../src/components/TextField"; | ||||
| import { FilledButton } from "../../src/components/FilledButton"; | ||||
|  | ||||
| import { ReactComponent as CollabImage } from "../../src/assets/lock.svg"; | ||||
| import "./RoomDialog.scss"; | ||||
|  | ||||
| const getShareIcon = () => { | ||||
|   const navigator = window.navigator as any; | ||||
|   const isAppleBrowser = /Apple/.test(navigator.vendor); | ||||
|   const isWindowsBrowser = navigator.appVersion.indexOf("Win") !== -1; | ||||
|  | ||||
|   if (isAppleBrowser) { | ||||
|     return shareIOS; | ||||
|   } else if (isWindowsBrowser) { | ||||
|     return shareWindows; | ||||
|   } | ||||
|  | ||||
|   return share; | ||||
| }; | ||||
|  | ||||
| export type RoomModalProps = { | ||||
|   handleClose: () => void; | ||||
|   activeRoomLink: string; | ||||
|   username: string; | ||||
|   onUsernameChange: (username: string) => void; | ||||
|   onRoomCreate: () => void; | ||||
|   onRoomDestroy: () => void; | ||||
|   setErrorMessage: (message: string) => void; | ||||
| }; | ||||
|  | ||||
| export const RoomModal = ({ | ||||
|   activeRoomLink, | ||||
|   onRoomCreate, | ||||
|   onRoomDestroy, | ||||
|   setErrorMessage, | ||||
|   username, | ||||
|   onUsernameChange, | ||||
|   handleClose, | ||||
| }: RoomModalProps) => { | ||||
|   const { t } = useI18n(); | ||||
|   const [justCopied, setJustCopied] = useState(false); | ||||
|   const timerRef = useRef<number>(0); | ||||
|   const ref = useRef<HTMLInputElement>(null); | ||||
|   const isShareSupported = "share" in navigator; | ||||
|  | ||||
|   const copyRoomLink = async () => { | ||||
|     try { | ||||
|       await copyTextToSystemClipboard(activeRoomLink); | ||||
|  | ||||
|       setJustCopied(true); | ||||
|  | ||||
|       if (timerRef.current) { | ||||
|         window.clearTimeout(timerRef.current); | ||||
|       } | ||||
|  | ||||
|       timerRef.current = window.setTimeout(() => { | ||||
|         setJustCopied(false); | ||||
|       }, 3000); | ||||
|     } catch (error: any) { | ||||
|       setErrorMessage(error.message); | ||||
|     } | ||||
|  | ||||
|     ref.current?.select(); | ||||
|   }; | ||||
|  | ||||
|   const shareRoomLink = async () => { | ||||
|     try { | ||||
|       await navigator.share({ | ||||
|         title: t("roomDialog.shareTitle"), | ||||
|         text: t("roomDialog.shareTitle"), | ||||
|         url: activeRoomLink, | ||||
|       }); | ||||
|     } catch (error: any) { | ||||
|       // Just ignore. | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   if (activeRoomLink) { | ||||
|     return ( | ||||
|       <> | ||||
|         <h3 className="RoomDialog__active__header"> | ||||
|           {t("labels.liveCollaboration")} | ||||
|         </h3> | ||||
|         <TextField | ||||
|           value={username} | ||||
|           placeholder="Your name" | ||||
|           label="Your name" | ||||
|           onChange={onUsernameChange} | ||||
|           onKeyDown={(event) => event.key === KEYS.ENTER && handleClose()} | ||||
|         /> | ||||
|         <div className="RoomDialog__active__linkRow"> | ||||
|           <TextField | ||||
|             ref={ref} | ||||
|             label="Link" | ||||
|             readonly | ||||
|             fullWidth | ||||
|             value={activeRoomLink} | ||||
|           /> | ||||
|           {isShareSupported && ( | ||||
|             <FilledButton | ||||
|               size="large" | ||||
|               variant="icon" | ||||
|               label="Share" | ||||
|               startIcon={getShareIcon()} | ||||
|               className="RoomDialog__active__share" | ||||
|               onClick={shareRoomLink} | ||||
|             /> | ||||
|           )} | ||||
|           <Popover.Root open={justCopied}> | ||||
|             <Popover.Trigger asChild> | ||||
|               <FilledButton | ||||
|                 size="large" | ||||
|                 label="Copy link" | ||||
|                 startIcon={copyIcon} | ||||
|                 onClick={copyRoomLink} | ||||
|               /> | ||||
|             </Popover.Trigger> | ||||
|             <Popover.Content | ||||
|               onOpenAutoFocus={(event) => event.preventDefault()} | ||||
|               onCloseAutoFocus={(event) => event.preventDefault()} | ||||
|               className="RoomDialog__popover" | ||||
|               side="top" | ||||
|               align="end" | ||||
|               sideOffset={5.5} | ||||
|             > | ||||
|               {tablerCheckIcon} copied | ||||
|             </Popover.Content> | ||||
|           </Popover.Root> | ||||
|         </div> | ||||
|         <div className="RoomDialog__active__description"> | ||||
|           <p> | ||||
|             <span | ||||
|               role="img" | ||||
|               aria-hidden="true" | ||||
|               className="RoomDialog__active__description__emoji" | ||||
|             > | ||||
|               🔒{" "} | ||||
|             </span> | ||||
|             {t("roomDialog.desc_privacy")} | ||||
|           </p> | ||||
|           <p>{t("roomDialog.desc_exitSession")}</p> | ||||
|         </div> | ||||
|  | ||||
|         <div className="RoomDialog__active__actions"> | ||||
|           <FilledButton | ||||
|             size="large" | ||||
|             variant="outlined" | ||||
|             color="danger" | ||||
|             label={t("roomDialog.button_stopSession")} | ||||
|             startIcon={playerStopFilledIcon} | ||||
|             onClick={() => { | ||||
|               trackEvent("share", "room closed"); | ||||
|               onRoomDestroy(); | ||||
|             }} | ||||
|           /> | ||||
|         </div> | ||||
|       </> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <div className="RoomDialog__inactive__illustration"> | ||||
|         <CollabImage /> | ||||
|       </div> | ||||
|       <div className="RoomDialog__inactive__header"> | ||||
|         {t("labels.liveCollaboration")} | ||||
|       </div> | ||||
|  | ||||
|       <div className="RoomDialog__inactive__description"> | ||||
|         <strong>{t("roomDialog.desc_intro")}</strong> | ||||
|         {t("roomDialog.desc_privacy")} | ||||
|       </div> | ||||
|  | ||||
|       <div className="RoomDialog__inactive__start_session"> | ||||
|         <FilledButton | ||||
|           size="large" | ||||
|           label={t("roomDialog.button_startSession")} | ||||
|           startIcon={playerPlayIcon} | ||||
|           onClick={() => { | ||||
|             trackEvent("share", "room creation", `ui (${getFrame()})`); | ||||
|             onRoomCreate(); | ||||
|           }} | ||||
|         /> | ||||
|       </div> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const RoomDialog = (props: RoomModalProps) => { | ||||
|   return ( | ||||
|     <Dialog size="small" onCloseRequest={props.handleClose} title={false}> | ||||
|       <div className="RoomDialog"> | ||||
|         <RoomModal {...props} /> | ||||
|       </div> | ||||
|     </Dialog> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default RoomDialog; | ||||
							
								
								
									
										154
									
								
								excalidraw-app/collab/reconciliation.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								excalidraw-app/collab/reconciliation.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,154 @@ | ||||
| import { PRECEDING_ELEMENT_KEY } from "../../src/constants"; | ||||
| import { ExcalidrawElement } from "../../src/element/types"; | ||||
| import { AppState } from "../../src/types"; | ||||
| import { arrayToMapWithIndex } from "../../src/utils"; | ||||
|  | ||||
| export type ReconciledElements = readonly ExcalidrawElement[] & { | ||||
|   _brand: "reconciledElements"; | ||||
| }; | ||||
|  | ||||
| export type BroadcastedExcalidrawElement = ExcalidrawElement & { | ||||
|   [PRECEDING_ELEMENT_KEY]?: string; | ||||
| }; | ||||
|  | ||||
| const shouldDiscardRemoteElement = ( | ||||
|   localAppState: AppState, | ||||
|   local: ExcalidrawElement | undefined, | ||||
|   remote: BroadcastedExcalidrawElement, | ||||
| ): boolean => { | ||||
|   if ( | ||||
|     local && | ||||
|     // local element is being edited | ||||
|     (local.id === localAppState.editingElement?.id || | ||||
|       local.id === localAppState.resizingElement?.id || | ||||
|       local.id === localAppState.draggingElement?.id || | ||||
|       // local element is newer | ||||
|       local.version > remote.version || | ||||
|       // resolve conflicting edits deterministically by taking the one with | ||||
|       // the lowest versionNonce | ||||
|       (local.version === remote.version && | ||||
|         local.versionNonce < remote.versionNonce)) | ||||
|   ) { | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| }; | ||||
|  | ||||
| export const reconcileElements = ( | ||||
|   localElements: readonly ExcalidrawElement[], | ||||
|   remoteElements: readonly BroadcastedExcalidrawElement[], | ||||
|   localAppState: AppState, | ||||
| ): ReconciledElements => { | ||||
|   const localElementsData = | ||||
|     arrayToMapWithIndex<ExcalidrawElement>(localElements); | ||||
|  | ||||
|   const reconciledElements: ExcalidrawElement[] = localElements.slice(); | ||||
|  | ||||
|   const duplicates = new WeakMap<ExcalidrawElement, true>(); | ||||
|  | ||||
|   let cursor = 0; | ||||
|   let offset = 0; | ||||
|  | ||||
|   let remoteElementIdx = -1; | ||||
|   for (const remoteElement of remoteElements) { | ||||
|     remoteElementIdx++; | ||||
|  | ||||
|     const local = localElementsData.get(remoteElement.id); | ||||
|  | ||||
|     if (shouldDiscardRemoteElement(localAppState, local?.[0], remoteElement)) { | ||||
|       if (remoteElement[PRECEDING_ELEMENT_KEY]) { | ||||
|         delete remoteElement[PRECEDING_ELEMENT_KEY]; | ||||
|       } | ||||
|  | ||||
|       continue; | ||||
|     } | ||||
|  | ||||
|     // Mark duplicate for removal as it'll be replaced with the remote element | ||||
|     if (local) { | ||||
|       // Unless the remote and local elements are the same element in which case | ||||
|       // we need to keep it as we'd otherwise discard it from the resulting | ||||
|       // array. | ||||
|       if (local[0] === remoteElement) { | ||||
|         continue; | ||||
|       } | ||||
|       duplicates.set(local[0], true); | ||||
|     } | ||||
|  | ||||
|     // parent may not be defined in case the remote client is running an older | ||||
|     // excalidraw version | ||||
|     const parent = | ||||
|       remoteElement[PRECEDING_ELEMENT_KEY] || | ||||
|       remoteElements[remoteElementIdx - 1]?.id || | ||||
|       null; | ||||
|  | ||||
|     if (parent != null) { | ||||
|       delete remoteElement[PRECEDING_ELEMENT_KEY]; | ||||
|  | ||||
|       // ^ indicates the element is the first in elements array | ||||
|       if (parent === "^") { | ||||
|         offset++; | ||||
|         if (cursor === 0) { | ||||
|           reconciledElements.unshift(remoteElement); | ||||
|           localElementsData.set(remoteElement.id, [ | ||||
|             remoteElement, | ||||
|             cursor - offset, | ||||
|           ]); | ||||
|         } else { | ||||
|           reconciledElements.splice(cursor + 1, 0, remoteElement); | ||||
|           localElementsData.set(remoteElement.id, [ | ||||
|             remoteElement, | ||||
|             cursor + 1 - offset, | ||||
|           ]); | ||||
|           cursor++; | ||||
|         } | ||||
|       } else { | ||||
|         let idx = localElementsData.has(parent) | ||||
|           ? localElementsData.get(parent)![1] | ||||
|           : null; | ||||
|         if (idx != null) { | ||||
|           idx += offset; | ||||
|         } | ||||
|         if (idx != null && idx >= cursor) { | ||||
|           reconciledElements.splice(idx + 1, 0, remoteElement); | ||||
|           offset++; | ||||
|           localElementsData.set(remoteElement.id, [ | ||||
|             remoteElement, | ||||
|             idx + 1 - offset, | ||||
|           ]); | ||||
|           cursor = idx + 1; | ||||
|         } else if (idx != null) { | ||||
|           reconciledElements.splice(cursor + 1, 0, remoteElement); | ||||
|           offset++; | ||||
|           localElementsData.set(remoteElement.id, [ | ||||
|             remoteElement, | ||||
|             cursor + 1 - offset, | ||||
|           ]); | ||||
|           cursor++; | ||||
|         } else { | ||||
|           reconciledElements.push(remoteElement); | ||||
|           localElementsData.set(remoteElement.id, [ | ||||
|             remoteElement, | ||||
|             reconciledElements.length - 1 - offset, | ||||
|           ]); | ||||
|         } | ||||
|       } | ||||
|       // no parent z-index information, local element exists → replace in place | ||||
|     } else if (local) { | ||||
|       reconciledElements[local[1]] = remoteElement; | ||||
|       localElementsData.set(remoteElement.id, [remoteElement, local[1]]); | ||||
|       // otherwise push to the end | ||||
|     } else { | ||||
|       reconciledElements.push(remoteElement); | ||||
|       localElementsData.set(remoteElement.id, [ | ||||
|         remoteElement, | ||||
|         reconciledElements.length - 1 - offset, | ||||
|       ]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   const ret: readonly ExcalidrawElement[] = reconciledElements.filter( | ||||
|     (element) => !duplicates.has(element), | ||||
|   ); | ||||
|  | ||||
|   return ret as ReconciledElements; | ||||
| }; | ||||
| @@ -1,163 +0,0 @@ | ||||
| import type { ExcalidrawImperativeAPI } from "../../packages/excalidraw/types"; | ||||
| import { | ||||
|   DiagramToCodePlugin, | ||||
|   exportToBlob, | ||||
|   getTextFromElements, | ||||
|   MIME_TYPES, | ||||
|   TTDDialog, | ||||
| } from "../../packages/excalidraw"; | ||||
| import { getDataURL } from "../../packages/excalidraw/data/blob"; | ||||
| import { safelyParseJSON } from "../../packages/excalidraw/utils"; | ||||
|  | ||||
| export const AIComponents = ({ | ||||
|   excalidrawAPI, | ||||
| }: { | ||||
|   excalidrawAPI: ExcalidrawImperativeAPI; | ||||
| }) => { | ||||
|   return ( | ||||
|     <> | ||||
|       <DiagramToCodePlugin | ||||
|         generate={async ({ frame, children }) => { | ||||
|           const appState = excalidrawAPI.getAppState(); | ||||
|  | ||||
|           const blob = await exportToBlob({ | ||||
|             data: { | ||||
|               elements: children, | ||||
|               appState: { | ||||
|                 ...appState, | ||||
|                 exportBackground: true, | ||||
|                 viewBackgroundColor: appState.viewBackgroundColor, | ||||
|               }, | ||||
|               files: excalidrawAPI.getFiles(), | ||||
|             }, | ||||
|             config: { | ||||
|               exportingFrame: frame, | ||||
|               mimeType: MIME_TYPES.jpg, | ||||
|             }, | ||||
|           }); | ||||
|  | ||||
|           const dataURL = await getDataURL(blob); | ||||
|  | ||||
|           const textFromFrameChildren = getTextFromElements(children); | ||||
|  | ||||
|           const response = await fetch( | ||||
|             `${ | ||||
|               import.meta.env.VITE_APP_AI_BACKEND | ||||
|             }/v1/ai/diagram-to-code/generate`, | ||||
|             { | ||||
|               method: "POST", | ||||
|               headers: { | ||||
|                 Accept: "application/json", | ||||
|                 "Content-Type": "application/json", | ||||
|               }, | ||||
|               body: JSON.stringify({ | ||||
|                 texts: textFromFrameChildren, | ||||
|                 image: dataURL, | ||||
|                 theme: appState.theme, | ||||
|               }), | ||||
|             }, | ||||
|           ); | ||||
|  | ||||
|           if (!response.ok) { | ||||
|             const text = await response.text(); | ||||
|             const errorJSON = safelyParseJSON(text); | ||||
|  | ||||
|             if (!errorJSON) { | ||||
|               throw new Error(text); | ||||
|             } | ||||
|  | ||||
|             if (errorJSON.statusCode === 429) { | ||||
|               return { | ||||
|                 html: `<html> | ||||
|                 <body style="margin: 0; text-align: center"> | ||||
|                 <div style="display: flex; align-items: center; justify-content: center; flex-direction: column; height: 100vh; padding: 0 60px"> | ||||
|                   <div style="color:red">Too many requests today,</br>please try again tomorrow!</div> | ||||
|                   </br> | ||||
|                   </br> | ||||
|                   <div>You can also try <a href="${ | ||||
|                     import.meta.env.VITE_APP_PLUS_LP | ||||
|                   }/plus?utm_source=excalidraw&utm_medium=app&utm_content=d2c" target="_blank" rel="noreferrer noopener">Excalidraw+</a> to get more requests.</div> | ||||
|                 </div> | ||||
|                 </body> | ||||
|                 </html>`, | ||||
|               }; | ||||
|             } | ||||
|  | ||||
|             throw new Error(errorJSON.message || text); | ||||
|           } | ||||
|  | ||||
|           try { | ||||
|             const { html } = await response.json(); | ||||
|  | ||||
|             if (!html) { | ||||
|               throw new Error("Generation failed (invalid response)"); | ||||
|             } | ||||
|             return { | ||||
|               html, | ||||
|             }; | ||||
|           } catch (error: any) { | ||||
|             throw new Error("Generation failed (invalid response)"); | ||||
|           } | ||||
|         }} | ||||
|       /> | ||||
|  | ||||
|       <TTDDialog | ||||
|         onTextSubmit={async (input) => { | ||||
|           try { | ||||
|             const response = await fetch( | ||||
|               `${ | ||||
|                 import.meta.env.VITE_APP_AI_BACKEND | ||||
|               }/v1/ai/text-to-diagram/generate`, | ||||
|               { | ||||
|                 method: "POST", | ||||
|                 headers: { | ||||
|                   Accept: "application/json", | ||||
|                   "Content-Type": "application/json", | ||||
|                 }, | ||||
|                 body: JSON.stringify({ prompt: input }), | ||||
|               }, | ||||
|             ); | ||||
|  | ||||
|             const rateLimit = response.headers.has("X-Ratelimit-Limit") | ||||
|               ? parseInt(response.headers.get("X-Ratelimit-Limit") || "0", 10) | ||||
|               : undefined; | ||||
|  | ||||
|             const rateLimitRemaining = response.headers.has( | ||||
|               "X-Ratelimit-Remaining", | ||||
|             ) | ||||
|               ? parseInt( | ||||
|                   response.headers.get("X-Ratelimit-Remaining") || "0", | ||||
|                   10, | ||||
|                 ) | ||||
|               : undefined; | ||||
|  | ||||
|             const json = await response.json(); | ||||
|  | ||||
|             if (!response.ok) { | ||||
|               if (response.status === 429) { | ||||
|                 return { | ||||
|                   rateLimit, | ||||
|                   rateLimitRemaining, | ||||
|                   error: new Error( | ||||
|                     "Too many requests today, please try again tomorrow!", | ||||
|                   ), | ||||
|                 }; | ||||
|               } | ||||
|  | ||||
|               throw new Error(json.message || "Generation failed..."); | ||||
|             } | ||||
|  | ||||
|             const generatedResponse = json.generatedResponse; | ||||
|             if (!generatedResponse) { | ||||
|               throw new Error("Generation failed..."); | ||||
|             } | ||||
|  | ||||
|             return { generatedResponse, rateLimit, rateLimitRemaining }; | ||||
|           } catch (err: any) { | ||||
|             throw new Error("Request failed"); | ||||
|           } | ||||
|         }} | ||||
|       /> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
| @@ -1,29 +1,25 @@ | ||||
| import React from "react"; | ||||
| import { Footer } from "../../packages/excalidraw/index"; | ||||
| import { Footer } from "../../src/packages/excalidraw/index"; | ||||
| import { EncryptedIcon } from "./EncryptedIcon"; | ||||
| import { ExcalidrawPlusAppLink } from "./ExcalidrawPlusAppLink"; | ||||
| import { isExcalidrawPlusSignedUser } from "../app_constants"; | ||||
| import { DebugFooter, isVisualDebuggerEnabled } from "./DebugCanvas"; | ||||
|  | ||||
| export const AppFooter = React.memo( | ||||
|   ({ onChange }: { onChange: () => void }) => { | ||||
|     return ( | ||||
|       <Footer> | ||||
|         <div | ||||
|           style={{ | ||||
|             display: "flex", | ||||
|             gap: ".5rem", | ||||
|             alignItems: "center", | ||||
|           }} | ||||
|         > | ||||
|           {isVisualDebuggerEnabled() && <DebugFooter onChange={onChange} />} | ||||
|           {isExcalidrawPlusSignedUser ? ( | ||||
|             <ExcalidrawPlusAppLink /> | ||||
|           ) : ( | ||||
|             <EncryptedIcon /> | ||||
|           )} | ||||
|         </div> | ||||
|       </Footer> | ||||
|     ); | ||||
|   }, | ||||
| ); | ||||
| export const AppFooter = React.memo(() => { | ||||
|   return ( | ||||
|     <Footer> | ||||
|       <div | ||||
|         style={{ | ||||
|           display: "flex", | ||||
|           gap: ".5rem", | ||||
|           alignItems: "center", | ||||
|         }} | ||||
|       > | ||||
|         {isExcalidrawPlusSignedUser ? ( | ||||
|           <ExcalidrawPlusAppLink /> | ||||
|         ) : ( | ||||
|           <EncryptedIcon /> | ||||
|         )} | ||||
|       </div> | ||||
|     </Footer> | ||||
|   ); | ||||
| }); | ||||
|   | ||||
| @@ -1,22 +1,12 @@ | ||||
| import React from "react"; | ||||
| import { | ||||
|   loginIcon, | ||||
|   ExcalLogo, | ||||
|   eyeIcon, | ||||
| } from "../../packages/excalidraw/components/icons"; | ||||
| import type { Theme } from "../../packages/excalidraw/element/types"; | ||||
| import { MainMenu } from "../../packages/excalidraw/index"; | ||||
| import { isExcalidrawPlusSignedUser } from "../app_constants"; | ||||
| import { LanguageList } from "../app-language/LanguageList"; | ||||
| import { saveDebugState } from "./DebugCanvas"; | ||||
| import { PlusPromoIcon } from "../../src/components/icons"; | ||||
| import { MainMenu } from "../../src/packages/excalidraw/index"; | ||||
| import { LanguageList } from "./LanguageList"; | ||||
|  | ||||
| export const AppMainMenu: React.FC<{ | ||||
|   onCollabDialogOpen: () => any; | ||||
|   setCollabDialogShown: (toggle: boolean) => any; | ||||
|   isCollaborating: boolean; | ||||
|   isCollabEnabled: boolean; | ||||
|   theme: Theme | "system"; | ||||
|   setTheme: (theme: Theme | "system") => void; | ||||
|   refresh: () => void; | ||||
| }> = React.memo((props) => { | ||||
|   return ( | ||||
|     <MainMenu> | ||||
| @@ -27,56 +17,25 @@ export const AppMainMenu: React.FC<{ | ||||
|       {props.isCollabEnabled && ( | ||||
|         <MainMenu.DefaultItems.LiveCollaborationTrigger | ||||
|           isCollaborating={props.isCollaborating} | ||||
|           onSelect={() => props.onCollabDialogOpen()} | ||||
|           onSelect={() => props.setCollabDialogShown(true)} | ||||
|         /> | ||||
|       )} | ||||
|       <MainMenu.DefaultItems.CommandPalette className="highlighted" /> | ||||
|       <MainMenu.DefaultItems.SearchMenu /> | ||||
|  | ||||
|       <MainMenu.DefaultItems.Help /> | ||||
|       <MainMenu.DefaultItems.ClearCanvas /> | ||||
|       <MainMenu.Separator /> | ||||
|       <MainMenu.ItemLink | ||||
|         icon={ExcalLogo} | ||||
|         icon={PlusPromoIcon} | ||||
|         href={`${ | ||||
|           import.meta.env.VITE_APP_PLUS_LP | ||||
|         }/plus?utm_source=excalidraw&utm_medium=app&utm_content=hamburger`} | ||||
|         className="" | ||||
|         className="ExcalidrawPlus" | ||||
|       > | ||||
|         Excalidraw+ | ||||
|       </MainMenu.ItemLink> | ||||
|       <MainMenu.DefaultItems.Socials /> | ||||
|       <MainMenu.ItemLink | ||||
|         icon={loginIcon} | ||||
|         href={`${import.meta.env.VITE_APP_PLUS_APP}${ | ||||
|           isExcalidrawPlusSignedUser ? "" : "/sign-up" | ||||
|         }?utm_source=signin&utm_medium=app&utm_content=hamburger`} | ||||
|         className="highlighted" | ||||
|       > | ||||
|         {isExcalidrawPlusSignedUser ? "Sign in" : "Sign up"} | ||||
|       </MainMenu.ItemLink> | ||||
|       {import.meta.env.DEV && ( | ||||
|         <MainMenu.Item | ||||
|           icon={eyeIcon} | ||||
|           onClick={() => { | ||||
|             if (window.visualDebug) { | ||||
|               delete window.visualDebug; | ||||
|               saveDebugState({ enabled: false }); | ||||
|             } else { | ||||
|               window.visualDebug = { data: [] }; | ||||
|               saveDebugState({ enabled: true }); | ||||
|             } | ||||
|             props?.refresh(); | ||||
|           }} | ||||
|         > | ||||
|           Visual Debug | ||||
|         </MainMenu.Item> | ||||
|       )} | ||||
|       <MainMenu.Separator /> | ||||
|       <MainMenu.DefaultItems.ToggleTheme | ||||
|         allowSystemTheme | ||||
|         theme={props.theme} | ||||
|         onSelect={props.setTheme} | ||||
|       /> | ||||
|       <MainMenu.DefaultItems.ToggleTheme /> | ||||
|       <MainMenu.ItemCustom> | ||||
|         <LanguageList style={{ width: "100%" }} /> | ||||
|       </MainMenu.ItemCustom> | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| import React from "react"; | ||||
| import { loginIcon } from "../../packages/excalidraw/components/icons"; | ||||
| import { useI18n } from "../../packages/excalidraw/i18n"; | ||||
| import { WelcomeScreen } from "../../packages/excalidraw/index"; | ||||
| import { PlusPromoIcon } from "../../src/components/icons"; | ||||
| import { useI18n } from "../../src/i18n"; | ||||
| import { WelcomeScreen } from "../../src/packages/excalidraw/index"; | ||||
| import { isExcalidrawPlusSignedUser } from "../app_constants"; | ||||
| import { POINTER_EVENTS } from "../../packages/excalidraw/constants"; | ||||
| import { POINTER_EVENTS } from "../../src/constants"; | ||||
|  | ||||
| export const AppWelcomeScreen: React.FC<{ | ||||
|   onCollabDialogOpen: () => any; | ||||
|   setCollabDialogShown: (toggle: boolean) => any; | ||||
|   isCollabEnabled: boolean; | ||||
| }> = React.memo((props) => { | ||||
|   const { t } = useI18n(); | ||||
| @@ -52,7 +52,7 @@ export const AppWelcomeScreen: React.FC<{ | ||||
|           <WelcomeScreen.Center.MenuItemHelp /> | ||||
|           {props.isCollabEnabled && ( | ||||
|             <WelcomeScreen.Center.MenuItemLiveCollaborationTrigger | ||||
|               onSelect={() => props.onCollabDialogOpen()} | ||||
|               onSelect={() => props.setCollabDialogShown(true)} | ||||
|             /> | ||||
|           )} | ||||
|           {!isExcalidrawPlusSignedUser && ( | ||||
| @@ -61,9 +61,9 @@ export const AppWelcomeScreen: React.FC<{ | ||||
|                 import.meta.env.VITE_APP_PLUS_LP | ||||
|               }/plus?utm_source=excalidraw&utm_medium=app&utm_content=welcomeScreenGuest`} | ||||
|               shortcut={null} | ||||
|               icon={loginIcon} | ||||
|               icon={PlusPromoIcon} | ||||
|             > | ||||
|               Sign up | ||||
|               Try Excalidraw Plus! | ||||
|             </WelcomeScreen.Center.MenuItemLink> | ||||
|           )} | ||||
|         </WelcomeScreen.Center.Menu> | ||||
|   | ||||
| @@ -1,311 +0,0 @@ | ||||
| import { forwardRef, useCallback, useImperativeHandle, useRef } from "react"; | ||||
| import { type AppState } from "../../packages/excalidraw/types"; | ||||
| import { throttleRAF } from "../../packages/excalidraw/utils"; | ||||
| import { | ||||
|   bootstrapCanvas, | ||||
|   getNormalizedCanvasDimensions, | ||||
| } from "../../packages/excalidraw/renderer/helpers"; | ||||
| import type { DebugElement } from "../../packages/excalidraw/visualdebug"; | ||||
| import { | ||||
|   ArrowheadArrowIcon, | ||||
|   CloseIcon, | ||||
|   TrashIcon, | ||||
| } from "../../packages/excalidraw/components/icons"; | ||||
| import { STORAGE_KEYS } from "../app_constants"; | ||||
| import { | ||||
|   isLineSegment, | ||||
|   type GlobalPoint, | ||||
|   type LineSegment, | ||||
| } from "../../packages/math"; | ||||
|  | ||||
| const renderLine = ( | ||||
|   context: CanvasRenderingContext2D, | ||||
|   zoom: number, | ||||
|   segment: LineSegment<GlobalPoint>, | ||||
|   color: string, | ||||
| ) => { | ||||
|   context.save(); | ||||
|   context.strokeStyle = color; | ||||
|   context.beginPath(); | ||||
|   context.moveTo(segment[0][0] * zoom, segment[0][1] * zoom); | ||||
|   context.lineTo(segment[1][0] * zoom, segment[1][1] * zoom); | ||||
|   context.stroke(); | ||||
|   context.restore(); | ||||
| }; | ||||
|  | ||||
| const renderOrigin = (context: CanvasRenderingContext2D, zoom: number) => { | ||||
|   context.strokeStyle = "#888"; | ||||
|   context.save(); | ||||
|   context.beginPath(); | ||||
|   context.moveTo(-10 * zoom, -10 * zoom); | ||||
|   context.lineTo(10 * zoom, 10 * zoom); | ||||
|   context.moveTo(10 * zoom, -10 * zoom); | ||||
|   context.lineTo(-10 * zoom, 10 * zoom); | ||||
|   context.stroke(); | ||||
|   context.save(); | ||||
| }; | ||||
|  | ||||
| const render = ( | ||||
|   frame: DebugElement[], | ||||
|   context: CanvasRenderingContext2D, | ||||
|   appState: AppState, | ||||
| ) => { | ||||
|   frame.forEach((el: DebugElement) => { | ||||
|     switch (true) { | ||||
|       case isLineSegment(el.data): | ||||
|         renderLine( | ||||
|           context, | ||||
|           appState.zoom.value, | ||||
|           el.data as LineSegment<GlobalPoint>, | ||||
|           el.color, | ||||
|         ); | ||||
|         break; | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| const _debugRenderer = ( | ||||
|   canvas: HTMLCanvasElement, | ||||
|   appState: AppState, | ||||
|   scale: number, | ||||
|   refresh: () => void, | ||||
| ) => { | ||||
|   const [normalizedWidth, normalizedHeight] = getNormalizedCanvasDimensions( | ||||
|     canvas, | ||||
|     scale, | ||||
|   ); | ||||
|  | ||||
|   if (appState.height !== canvas.height || appState.width !== canvas.width) { | ||||
|     refresh(); | ||||
|   } | ||||
|  | ||||
|   const context = bootstrapCanvas({ | ||||
|     canvas, | ||||
|     scale, | ||||
|     normalizedWidth, | ||||
|     normalizedHeight, | ||||
|     canvasBackgroundColor: "transparent", | ||||
|   }); | ||||
|  | ||||
|   // Apply zoom | ||||
|   context.save(); | ||||
|   context.translate( | ||||
|     appState.scrollX * appState.zoom.value, | ||||
|     appState.scrollY * appState.zoom.value, | ||||
|   ); | ||||
|  | ||||
|   renderOrigin(context, appState.zoom.value); | ||||
|  | ||||
|   if ( | ||||
|     window.visualDebug?.currentFrame && | ||||
|     window.visualDebug?.data && | ||||
|     window.visualDebug.data.length > 0 | ||||
|   ) { | ||||
|     // Render only one frame | ||||
|     const [idx] = debugFrameData(); | ||||
|  | ||||
|     render(window.visualDebug.data[idx], context, appState); | ||||
|   } else { | ||||
|     // Render all debug frames | ||||
|     window.visualDebug?.data.forEach((frame) => { | ||||
|       render(frame, context, appState); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   if (window.visualDebug) { | ||||
|     window.visualDebug!.data = | ||||
|       window.visualDebug?.data.map((frame) => | ||||
|         frame.filter((el) => el.permanent), | ||||
|       ) ?? []; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const debugFrameData = (): [number, number] => { | ||||
|   const currentFrame = window.visualDebug?.currentFrame ?? 0; | ||||
|   const frameCount = window.visualDebug?.data.length ?? 0; | ||||
|  | ||||
|   if (frameCount > 0) { | ||||
|     return [currentFrame % frameCount, window.visualDebug?.currentFrame ?? 0]; | ||||
|   } | ||||
|  | ||||
|   return [0, 0]; | ||||
| }; | ||||
|  | ||||
| export const saveDebugState = (debug: { enabled: boolean }) => { | ||||
|   try { | ||||
|     localStorage.setItem( | ||||
|       STORAGE_KEYS.LOCAL_STORAGE_DEBUG, | ||||
|       JSON.stringify(debug), | ||||
|     ); | ||||
|   } catch (error: any) { | ||||
|     console.error(error); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export const debugRenderer = throttleRAF( | ||||
|   ( | ||||
|     canvas: HTMLCanvasElement, | ||||
|     appState: AppState, | ||||
|     scale: number, | ||||
|     refresh: () => void, | ||||
|   ) => { | ||||
|     _debugRenderer(canvas, appState, scale, refresh); | ||||
|   }, | ||||
|   { trailing: true }, | ||||
| ); | ||||
|  | ||||
| export const loadSavedDebugState = () => { | ||||
|   let debug; | ||||
|   try { | ||||
|     const savedDebugState = localStorage.getItem( | ||||
|       STORAGE_KEYS.LOCAL_STORAGE_DEBUG, | ||||
|     ); | ||||
|     if (savedDebugState) { | ||||
|       debug = JSON.parse(savedDebugState) as { enabled: boolean }; | ||||
|     } | ||||
|   } catch (error: any) { | ||||
|     console.error(error); | ||||
|   } | ||||
|  | ||||
|   return debug ?? { enabled: false }; | ||||
| }; | ||||
|  | ||||
| export const isVisualDebuggerEnabled = () => | ||||
|   Array.isArray(window.visualDebug?.data); | ||||
|  | ||||
| export const DebugFooter = ({ onChange }: { onChange: () => void }) => { | ||||
|   const moveForward = useCallback(() => { | ||||
|     if ( | ||||
|       !window.visualDebug?.currentFrame || | ||||
|       isNaN(window.visualDebug?.currentFrame ?? -1) | ||||
|     ) { | ||||
|       window.visualDebug!.currentFrame = 0; | ||||
|     } | ||||
|     window.visualDebug!.currentFrame += 1; | ||||
|     onChange(); | ||||
|   }, [onChange]); | ||||
|   const moveBackward = useCallback(() => { | ||||
|     if ( | ||||
|       !window.visualDebug?.currentFrame || | ||||
|       isNaN(window.visualDebug?.currentFrame ?? -1) || | ||||
|       window.visualDebug?.currentFrame < 1 | ||||
|     ) { | ||||
|       window.visualDebug!.currentFrame = 1; | ||||
|     } | ||||
|     window.visualDebug!.currentFrame -= 1; | ||||
|     onChange(); | ||||
|   }, [onChange]); | ||||
|   const reset = useCallback(() => { | ||||
|     window.visualDebug!.currentFrame = undefined; | ||||
|     onChange(); | ||||
|   }, [onChange]); | ||||
|   const trashFrames = useCallback(() => { | ||||
|     if (window.visualDebug) { | ||||
|       window.visualDebug.currentFrame = undefined; | ||||
|       window.visualDebug.data = []; | ||||
|     } | ||||
|     onChange(); | ||||
|   }, [onChange]); | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <button | ||||
|         className="ToolIcon_type_button" | ||||
|         data-testid="debug-forward" | ||||
|         aria-label="Move forward" | ||||
|         type="button" | ||||
|         onClick={trashFrames} | ||||
|       > | ||||
|         <div | ||||
|           className="ToolIcon__icon" | ||||
|           aria-hidden="true" | ||||
|           aria-disabled="false" | ||||
|         > | ||||
|           {TrashIcon} | ||||
|         </div> | ||||
|       </button> | ||||
|       <button | ||||
|         className="ToolIcon_type_button" | ||||
|         data-testid="debug-forward" | ||||
|         aria-label="Move forward" | ||||
|         type="button" | ||||
|         onClick={moveBackward} | ||||
|       > | ||||
|         <div | ||||
|           className="ToolIcon__icon" | ||||
|           aria-hidden="true" | ||||
|           aria-disabled="false" | ||||
|         > | ||||
|           <ArrowheadArrowIcon flip /> | ||||
|         </div> | ||||
|       </button> | ||||
|       <button | ||||
|         className="ToolIcon_type_button" | ||||
|         data-testid="debug-forward" | ||||
|         aria-label="Move forward" | ||||
|         type="button" | ||||
|         onClick={reset} | ||||
|       > | ||||
|         <div | ||||
|           className="ToolIcon__icon" | ||||
|           aria-hidden="true" | ||||
|           aria-disabled="false" | ||||
|         > | ||||
|           {CloseIcon} | ||||
|         </div> | ||||
|       </button> | ||||
|       <button | ||||
|         className="ToolIcon_type_button" | ||||
|         data-testid="debug-backward" | ||||
|         aria-label="Move backward" | ||||
|         type="button" | ||||
|         onClick={moveForward} | ||||
|       > | ||||
|         <div | ||||
|           className="ToolIcon__icon" | ||||
|           aria-hidden="true" | ||||
|           aria-disabled="false" | ||||
|         > | ||||
|           <ArrowheadArrowIcon /> | ||||
|         </div> | ||||
|       </button> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| interface DebugCanvasProps { | ||||
|   appState: AppState; | ||||
|   scale: number; | ||||
| } | ||||
|  | ||||
| const DebugCanvas = forwardRef<HTMLCanvasElement, DebugCanvasProps>( | ||||
|   ({ appState, scale }, ref) => { | ||||
|     const { width, height } = appState; | ||||
|  | ||||
|     const canvasRef = useRef<HTMLCanvasElement>(null); | ||||
|     useImperativeHandle<HTMLCanvasElement | null, HTMLCanvasElement | null>( | ||||
|       ref, | ||||
|       () => canvasRef.current, | ||||
|       [canvasRef], | ||||
|     ); | ||||
|  | ||||
|     return ( | ||||
|       <canvas | ||||
|         style={{ | ||||
|           width, | ||||
|           height, | ||||
|           position: "absolute", | ||||
|           zIndex: 2, | ||||
|           pointerEvents: "none", | ||||
|         }} | ||||
|         width={width * scale} | ||||
|         height={height * scale} | ||||
|         ref={canvasRef} | ||||
|       > | ||||
|         Debug Canvas | ||||
|       </canvas> | ||||
|     ); | ||||
|   }, | ||||
| ); | ||||
|  | ||||
| export default DebugCanvas; | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { shield } from "../../packages/excalidraw/components/icons"; | ||||
| import { Tooltip } from "../../packages/excalidraw/components/Tooltip"; | ||||
| import { useI18n } from "../../packages/excalidraw/i18n"; | ||||
| import { shield } from "../../src/components/icons"; | ||||
| import { Tooltip } from "../../src/components/Tooltip"; | ||||
| import { useI18n } from "../../src/i18n"; | ||||
|  | ||||
| export const EncryptedIcon = () => { | ||||
|   const { t } = useI18n(); | ||||
| @@ -8,7 +8,7 @@ export const EncryptedIcon = () => { | ||||
|   return ( | ||||
|     <a | ||||
|       className="encrypted-icon tooltip" | ||||
|       href="https://plus.excalidraw.com/blog/end-to-end-encryption/" | ||||
|       href="https://blog.excalidraw.com/end-to-end-encryption/" | ||||
|       target="_blank" | ||||
|       rel="noopener noreferrer" | ||||
|       aria-label={t("encrypted.link")} | ||||
|   | ||||
| @@ -1,36 +1,25 @@ | ||||
| import React from "react"; | ||||
| import { Card } from "../../packages/excalidraw/components/Card"; | ||||
| import { ToolButton } from "../../packages/excalidraw/components/ToolButton"; | ||||
| import { serializeAsJSON } from "../../packages/excalidraw/data/json"; | ||||
| import { Card } from "../../src/components/Card"; | ||||
| import { ToolButton } from "../../src/components/ToolButton"; | ||||
| import { serializeAsJSON } from "../../src/data/json"; | ||||
| import { loadFirebaseStorage, saveFilesToFirebase } from "../data/firebase"; | ||||
| import type { | ||||
|   FileId, | ||||
|   NonDeletedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import type { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFiles, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import { FileId, NonDeletedExcalidrawElement } from "../../src/element/types"; | ||||
| import { AppState, BinaryFileData, BinaryFiles } from "../../src/types"; | ||||
| import { nanoid } from "nanoid"; | ||||
| import { useI18n } from "../../packages/excalidraw/i18n"; | ||||
| import { | ||||
|   encryptData, | ||||
|   generateEncryptionKey, | ||||
| } from "../../packages/excalidraw/data/encryption"; | ||||
| import { isInitializedImageElement } from "../../packages/excalidraw/element/typeChecks"; | ||||
| import { useI18n } from "../../src/i18n"; | ||||
| import { encryptData, generateEncryptionKey } from "../../src/data/encryption"; | ||||
| import { isInitializedImageElement } from "../../src/element/typeChecks"; | ||||
| import { FILE_UPLOAD_MAX_BYTES } from "../app_constants"; | ||||
| import { encodeFilesForUpload } from "../data/FileManager"; | ||||
| import { MIME_TYPES } from "../../packages/excalidraw/constants"; | ||||
| import { trackEvent } from "../../packages/excalidraw/analytics"; | ||||
| import { getFrame } from "../../packages/excalidraw/utils"; | ||||
| import { ExcalidrawLogo } from "../../packages/excalidraw/components/ExcalidrawLogo"; | ||||
| import { MIME_TYPES } from "../../src/constants"; | ||||
| import { trackEvent } from "../../src/analytics"; | ||||
| import { getFrame } from "../../src/utils"; | ||||
| import { ExcalidrawLogo } from "../../src/components/ExcalidrawLogo"; | ||||
|  | ||||
| export const exportToExcalidrawPlus = async ( | ||||
|   elements: readonly NonDeletedExcalidrawElement[], | ||||
|   appState: Partial<AppState>, | ||||
|   files: BinaryFiles, | ||||
|   name: string, | ||||
| ) => { | ||||
|   const firebase = await loadFirebaseStorage(); | ||||
|  | ||||
| @@ -54,7 +43,7 @@ export const exportToExcalidrawPlus = async ( | ||||
|     .ref(`/migrations/scenes/${id}`) | ||||
|     .put(blob, { | ||||
|       customMetadata: { | ||||
|         data: JSON.stringify({ version: 2, name }), | ||||
|         data: JSON.stringify({ version: 2, name: appState.name }), | ||||
|         created: Date.now().toString(), | ||||
|       }, | ||||
|     }); | ||||
| @@ -90,10 +79,9 @@ export const ExportToExcalidrawPlus: React.FC<{ | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   appState: Partial<AppState>; | ||||
|   files: BinaryFiles; | ||||
|   name: string; | ||||
|   onError: (error: Error) => void; | ||||
|   onSuccess: () => void; | ||||
| }> = ({ elements, appState, files, name, onError, onSuccess }) => { | ||||
| }> = ({ elements, appState, files, onError, onSuccess }) => { | ||||
|   const { t } = useI18n(); | ||||
|   return ( | ||||
|     <Card color="primary"> | ||||
| @@ -119,7 +107,7 @@ export const ExportToExcalidrawPlus: React.FC<{ | ||||
|         onClick={async () => { | ||||
|           try { | ||||
|             trackEvent("export", "eplus", `ui (${getFrame()})`); | ||||
|             await exportToExcalidrawPlus(elements, appState, files, name); | ||||
|             await exportToExcalidrawPlus(elements, appState, files); | ||||
|             onSuccess(); | ||||
|           } catch (error: any) { | ||||
|             console.error(error); | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import oc from "open-color"; | ||||
| import React from "react"; | ||||
| import { THEME } from "../../packages/excalidraw/constants"; | ||||
| import type { Theme } from "../../packages/excalidraw/element/types"; | ||||
| import { THEME } from "../../src/constants"; | ||||
| import { Theme } from "../../src/element/types"; | ||||
|  | ||||
| // https://github.com/tholman/github-corners | ||||
| export const GitHubCorner = React.memo( | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| import { useSetAtom } from "jotai"; | ||||
| import React from "react"; | ||||
| import { useI18n, languages } from "../../packages/excalidraw/i18n"; | ||||
| import { appLangCodeAtom } from "./language-state"; | ||||
| import { appLangCodeAtom } from ".."; | ||||
| import { useI18n } from "../../src/i18n"; | ||||
| import { languages } from "../../src/i18n"; | ||||
| 
 | ||||
| export const LanguageList = ({ style }: { style?: React.CSSProperties }) => { | ||||
|   const { t, langCode } = useI18n(); | ||||
| @@ -1,41 +1,28 @@ | ||||
| import { StoreAction } from "../../packages/excalidraw"; | ||||
| import { compressData } from "../../packages/excalidraw/data/encode"; | ||||
| import { newElementWith } from "../../packages/excalidraw/element/mutateElement"; | ||||
| import { isInitializedImageElement } from "../../packages/excalidraw/element/typeChecks"; | ||||
| import type { | ||||
| import { compressData } from "../../src/data/encode"; | ||||
| import { newElementWith } from "../../src/element/mutateElement"; | ||||
| import { isInitializedImageElement } from "../../src/element/typeChecks"; | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawImageElement, | ||||
|   FileId, | ||||
|   InitializedExcalidrawImageElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { t } from "../../packages/excalidraw/i18n"; | ||||
| import type { | ||||
| } from "../../src/element/types"; | ||||
| import { t } from "../../src/i18n"; | ||||
| import { | ||||
|   BinaryFileData, | ||||
|   BinaryFileMetadata, | ||||
|   ExcalidrawImperativeAPI, | ||||
|   BinaryFiles, | ||||
| } from "../../packages/excalidraw/types"; | ||||
|  | ||||
| type FileVersion = Required<BinaryFileData>["version"]; | ||||
| } from "../../src/types"; | ||||
|  | ||||
| export class FileManager { | ||||
|   /** files being fetched */ | ||||
|   private fetchingFiles = new Map<ExcalidrawImageElement["fileId"], true>(); | ||||
|   private erroredFiles_fetch = new Map< | ||||
|     ExcalidrawImageElement["fileId"], | ||||
|     true | ||||
|   >(); | ||||
|   /** files being saved */ | ||||
|   private savingFiles = new Map< | ||||
|     ExcalidrawImageElement["fileId"], | ||||
|     FileVersion | ||||
|   >(); | ||||
|   private savingFiles = new Map<ExcalidrawImageElement["fileId"], true>(); | ||||
|   /* files already saved to persistent storage */ | ||||
|   private savedFiles = new Map<ExcalidrawImageElement["fileId"], FileVersion>(); | ||||
|   private erroredFiles_save = new Map< | ||||
|     ExcalidrawImageElement["fileId"], | ||||
|     FileVersion | ||||
|   >(); | ||||
|   private savedFiles = new Map<ExcalidrawImageElement["fileId"], true>(); | ||||
|   private erroredFiles = new Map<ExcalidrawImageElement["fileId"], true>(); | ||||
|  | ||||
|   private _getFiles; | ||||
|   private _saveFiles; | ||||
| @@ -49,8 +36,8 @@ export class FileManager { | ||||
|       erroredFiles: Map<FileId, true>; | ||||
|     }>; | ||||
|     saveFiles: (data: { addedFiles: Map<FileId, BinaryFileData> }) => Promise<{ | ||||
|       savedFiles: Map<FileId, BinaryFileData>; | ||||
|       erroredFiles: Map<FileId, BinaryFileData>; | ||||
|       savedFiles: Map<FileId, true>; | ||||
|       erroredFiles: Map<FileId, true>; | ||||
|     }>; | ||||
|   }) { | ||||
|     this._getFiles = getFiles; | ||||
| @@ -58,28 +45,19 @@ export class FileManager { | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * returns whether file is saved/errored, or being processed | ||||
|    * returns whether file is already saved or being processed | ||||
|    */ | ||||
|   isFileTracked = (id: FileId) => { | ||||
|   isFileHandled = (id: FileId) => { | ||||
|     return ( | ||||
|       this.savedFiles.has(id) || | ||||
|       this.savingFiles.has(id) || | ||||
|       this.fetchingFiles.has(id) || | ||||
|       this.erroredFiles_fetch.has(id) || | ||||
|       this.erroredFiles_save.has(id) | ||||
|       this.savingFiles.has(id) || | ||||
|       this.erroredFiles.has(id) | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   isFileSavedOrBeingSaved = (file: BinaryFileData) => { | ||||
|     const fileVersion = this.getFileVersion(file); | ||||
|     return ( | ||||
|       this.savedFiles.get(file.id) === fileVersion || | ||||
|       this.savingFiles.get(file.id) === fileVersion | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   getFileVersion = (file: BinaryFileData) => { | ||||
|     return file.version ?? 1; | ||||
|   isFileSaved = (id: FileId) => { | ||||
|     return this.savedFiles.has(id); | ||||
|   }; | ||||
|  | ||||
|   saveFiles = async ({ | ||||
| @@ -92,16 +70,13 @@ export class FileManager { | ||||
|     const addedFiles: Map<FileId, BinaryFileData> = new Map(); | ||||
|  | ||||
|     for (const element of elements) { | ||||
|       const fileData = | ||||
|         isInitializedImageElement(element) && files[element.fileId]; | ||||
|  | ||||
|       if ( | ||||
|         fileData && | ||||
|         // NOTE if errored during save, won't retry due to this check | ||||
|         !this.isFileSavedOrBeingSaved(fileData) | ||||
|         isInitializedImageElement(element) && | ||||
|         files[element.fileId] && | ||||
|         !this.isFileHandled(element.fileId) | ||||
|       ) { | ||||
|         addedFiles.set(element.fileId, files[element.fileId]); | ||||
|         this.savingFiles.set(element.fileId, this.getFileVersion(fileData)); | ||||
|         this.savingFiles.set(element.fileId, true); | ||||
|       } | ||||
|     } | ||||
|  | ||||
| @@ -110,12 +85,8 @@ export class FileManager { | ||||
|         addedFiles, | ||||
|       }); | ||||
|  | ||||
|       for (const [fileId, fileData] of savedFiles) { | ||||
|         this.savedFiles.set(fileId, this.getFileVersion(fileData)); | ||||
|       } | ||||
|  | ||||
|       for (const [fileId, fileData] of erroredFiles) { | ||||
|         this.erroredFiles_save.set(fileId, this.getFileVersion(fileData)); | ||||
|       for (const [fileId] of savedFiles) { | ||||
|         this.savedFiles.set(fileId, true); | ||||
|       } | ||||
|  | ||||
|       return { | ||||
| @@ -149,10 +120,10 @@ export class FileManager { | ||||
|       const { loadedFiles, erroredFiles } = await this._getFiles(ids); | ||||
|  | ||||
|       for (const file of loadedFiles) { | ||||
|         this.savedFiles.set(file.id, this.getFileVersion(file)); | ||||
|         this.savedFiles.set(file.id, true); | ||||
|       } | ||||
|       for (const [fileId] of erroredFiles) { | ||||
|         this.erroredFiles_fetch.set(fileId, true); | ||||
|         this.erroredFiles.set(fileId, true); | ||||
|       } | ||||
|  | ||||
|       return { loadedFiles, erroredFiles }; | ||||
| @@ -188,7 +159,7 @@ export class FileManager { | ||||
|   ): element is InitializedExcalidrawImageElement => { | ||||
|     return ( | ||||
|       isInitializedImageElement(element) && | ||||
|       this.savedFiles.has(element.fileId) && | ||||
|       this.isFileSaved(element.fileId) && | ||||
|       element.status === "pending" | ||||
|     ); | ||||
|   }; | ||||
| @@ -197,8 +168,7 @@ export class FileManager { | ||||
|     this.fetchingFiles.clear(); | ||||
|     this.savingFiles.clear(); | ||||
|     this.savedFiles.clear(); | ||||
|     this.erroredFiles_fetch.clear(); | ||||
|     this.erroredFiles_save.clear(); | ||||
|     this.erroredFiles.clear(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -268,6 +238,5 @@ export const updateStaleImageStatuses = (params: { | ||||
|         } | ||||
|         return element; | ||||
|       }), | ||||
|     storeAction: StoreAction.UPDATE, | ||||
|   }); | ||||
| }; | ||||
|   | ||||
| @@ -10,34 +10,12 @@ | ||||
|  *   (localStorage, indexedDB). | ||||
|  */ | ||||
|  | ||||
| import { | ||||
|   createStore, | ||||
|   entries, | ||||
|   del, | ||||
|   getMany, | ||||
|   set, | ||||
|   setMany, | ||||
|   get, | ||||
| } from "idb-keyval"; | ||||
| import { clearAppStateForLocalStorage } from "../../packages/excalidraw/appState"; | ||||
| import { | ||||
|   CANVAS_SEARCH_TAB, | ||||
|   DEFAULT_SIDEBAR, | ||||
| } from "../../packages/excalidraw/constants"; | ||||
| import type { LibraryPersistedData } from "../../packages/excalidraw/data/library"; | ||||
| import type { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import { clearElementsForLocalStorage } from "../../packages/excalidraw/element"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import type { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFiles, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import type { MaybePromise } from "../../packages/excalidraw/utility-types"; | ||||
| import { debounce } from "../../packages/excalidraw/utils"; | ||||
| import { createStore, entries, del, getMany, set, setMany } from "idb-keyval"; | ||||
| import { clearAppStateForLocalStorage } from "../../src/appState"; | ||||
| import { clearElementsForLocalStorage } from "../../src/element"; | ||||
| import { ExcalidrawElement, FileId } from "../../src/element/types"; | ||||
| import { AppState, BinaryFileData, BinaryFiles } from "../../src/types"; | ||||
| import { debounce } from "../../src/utils"; | ||||
| import { SAVE_TO_LOCAL_STORAGE_TIMEOUT, STORAGE_KEYS } from "../app_constants"; | ||||
| import { FileManager } from "./FileManager"; | ||||
| import { Locker } from "./Locker"; | ||||
| @@ -70,22 +48,13 @@ const saveDataStateToLocalStorage = ( | ||||
|   appState: AppState, | ||||
| ) => { | ||||
|   try { | ||||
|     const _appState = clearAppStateForLocalStorage(appState); | ||||
|  | ||||
|     if ( | ||||
|       _appState.openSidebar?.name === DEFAULT_SIDEBAR.name && | ||||
|       _appState.openSidebar.tab === CANVAS_SEARCH_TAB | ||||
|     ) { | ||||
|       _appState.openSidebar = null; | ||||
|     } | ||||
|  | ||||
|     localStorage.setItem( | ||||
|       STORAGE_KEYS.LOCAL_STORAGE_ELEMENTS, | ||||
|       JSON.stringify(clearElementsForLocalStorage(elements)), | ||||
|     ); | ||||
|     localStorage.setItem( | ||||
|       STORAGE_KEYS.LOCAL_STORAGE_APP_STATE, | ||||
|       JSON.stringify(_appState), | ||||
|       JSON.stringify(clearAppStateForLocalStorage(appState)), | ||||
|     ); | ||||
|     updateBrowserStateVersion(STORAGE_KEYS.VERSION_DATA_STATE); | ||||
|   } catch (error: any) { | ||||
| @@ -183,8 +152,8 @@ export class LocalData { | ||||
|       ); | ||||
|     }, | ||||
|     async saveFiles({ addedFiles }) { | ||||
|       const savedFiles = new Map<FileId, BinaryFileData>(); | ||||
|       const erroredFiles = new Map<FileId, BinaryFileData>(); | ||||
|       const savedFiles = new Map<FileId, true>(); | ||||
|       const erroredFiles = new Map<FileId, true>(); | ||||
|  | ||||
|       // before we use `storage` event synchronization, let's update the flag | ||||
|       // optimistically. Hopefully nothing fails, and an IDB read executed | ||||
| @@ -195,10 +164,10 @@ export class LocalData { | ||||
|         [...addedFiles].map(async ([id, fileData]) => { | ||||
|           try { | ||||
|             await set(id, fileData, filesStore); | ||||
|             savedFiles.set(id, fileData); | ||||
|             savedFiles.set(id, true); | ||||
|           } catch (error: any) { | ||||
|             console.error(error); | ||||
|             erroredFiles.set(id, fileData); | ||||
|             erroredFiles.set(id, true); | ||||
|           } | ||||
|         }), | ||||
|       ); | ||||
| @@ -207,52 +176,3 @@ export class LocalData { | ||||
|     }, | ||||
|   }); | ||||
| } | ||||
| export class LibraryIndexedDBAdapter { | ||||
|   /** IndexedDB database and store name */ | ||||
|   private static idb_name = STORAGE_KEYS.IDB_LIBRARY; | ||||
|   /** library data store key */ | ||||
|   private static key = "libraryData"; | ||||
|  | ||||
|   private static store = createStore( | ||||
|     `${LibraryIndexedDBAdapter.idb_name}-db`, | ||||
|     `${LibraryIndexedDBAdapter.idb_name}-store`, | ||||
|   ); | ||||
|  | ||||
|   static async load() { | ||||
|     const IDBData = await get<LibraryPersistedData>( | ||||
|       LibraryIndexedDBAdapter.key, | ||||
|       LibraryIndexedDBAdapter.store, | ||||
|     ); | ||||
|  | ||||
|     return IDBData || null; | ||||
|   } | ||||
|  | ||||
|   static save(data: LibraryPersistedData): MaybePromise<void> { | ||||
|     return set( | ||||
|       LibraryIndexedDBAdapter.key, | ||||
|       data, | ||||
|       LibraryIndexedDBAdapter.store, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** LS Adapter used only for migrating LS library data | ||||
|  * to indexedDB */ | ||||
| export class LibraryLocalStorageMigrationAdapter { | ||||
|   static load() { | ||||
|     const LSData = localStorage.getItem( | ||||
|       STORAGE_KEYS.__LEGACY_LOCAL_STORAGE_LIBRARY, | ||||
|     ); | ||||
|     if (LSData != null) { | ||||
|       const libraryItems: ImportedDataState["libraryItems"] = | ||||
|         JSON.parse(LSData); | ||||
|       if (libraryItems) { | ||||
|         return { libraryItems }; | ||||
|       } | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
|   static clear() { | ||||
|     localStorage.removeItem(STORAGE_KEYS.__LEGACY_LOCAL_STORAGE_LIBRARY); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,30 +1,20 @@ | ||||
| import { reconcileElements } from "../../packages/excalidraw"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { getSceneVersion } from "../../packages/excalidraw/element"; | ||||
| import type Portal from "../collab/Portal"; | ||||
| import { restoreElements } from "../../packages/excalidraw/data/restore"; | ||||
| import type { | ||||
| import { ExcalidrawElement, FileId } from "../../src/element/types"; | ||||
| import { getSceneVersion } from "../../src/element"; | ||||
| import Portal from "../collab/Portal"; | ||||
| import { restoreElements } from "../../src/data/restore"; | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFileMetadata, | ||||
|   DataURL, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| } from "../../src/types"; | ||||
| import { FILE_CACHE_MAX_AGE_SEC } from "../app_constants"; | ||||
| import { decompressData } from "../../packages/excalidraw/data/encode"; | ||||
| import { | ||||
|   encryptData, | ||||
|   decryptData, | ||||
| } from "../../packages/excalidraw/data/encryption"; | ||||
| import { MIME_TYPES } from "../../packages/excalidraw/constants"; | ||||
| import type { SyncableExcalidrawElement } from "."; | ||||
| import { getSyncableElements } from "."; | ||||
| import type { ResolutionType } from "../../packages/excalidraw/utility-types"; | ||||
| import type { Socket } from "socket.io-client"; | ||||
| import type { RemoteExcalidrawElement } from "../../packages/excalidraw/data/reconcile"; | ||||
| import { decompressData } from "../../src/data/encode"; | ||||
| import { encryptData, decryptData } from "../../src/data/encryption"; | ||||
| import { MIME_TYPES } from "../../src/constants"; | ||||
| import { reconcileElements } from "../collab/reconciliation"; | ||||
| import { getSyncableElements, SyncableExcalidrawElement } from "."; | ||||
| import { ResolutionType } from "../../src/utility-types"; | ||||
|  | ||||
| // private | ||||
| // ----------------------------------------------------------------------------- | ||||
| @@ -142,12 +132,12 @@ const decryptElements = async ( | ||||
| }; | ||||
|  | ||||
| class FirebaseSceneVersionCache { | ||||
|   private static cache = new WeakMap<Socket, number>(); | ||||
|   static get = (socket: Socket) => { | ||||
|   private static cache = new WeakMap<SocketIOClient.Socket, number>(); | ||||
|   static get = (socket: SocketIOClient.Socket) => { | ||||
|     return FirebaseSceneVersionCache.cache.get(socket); | ||||
|   }; | ||||
|   static set = ( | ||||
|     socket: Socket, | ||||
|     socket: SocketIOClient.Socket, | ||||
|     elements: readonly SyncableExcalidrawElement[], | ||||
|   ) => { | ||||
|     FirebaseSceneVersionCache.cache.set(socket, getSceneVersion(elements)); | ||||
| @@ -177,8 +167,8 @@ export const saveFilesToFirebase = async ({ | ||||
| }) => { | ||||
|   const firebase = await loadFirebaseStorage(); | ||||
|  | ||||
|   const erroredFiles: FileId[] = []; | ||||
|   const savedFiles: FileId[] = []; | ||||
|   const erroredFiles = new Map<FileId, true>(); | ||||
|   const savedFiles = new Map<FileId, true>(); | ||||
|  | ||||
|   await Promise.all( | ||||
|     files.map(async ({ id, buffer }) => { | ||||
| @@ -194,9 +184,9 @@ export const saveFilesToFirebase = async ({ | ||||
|               cacheControl: `public, max-age=${FILE_CACHE_MAX_AGE_SEC}`, | ||||
|             }, | ||||
|           ); | ||||
|         savedFiles.push(id); | ||||
|         savedFiles.set(id, true); | ||||
|       } catch (error: any) { | ||||
|         erroredFiles.push(id); | ||||
|         erroredFiles.set(id, true); | ||||
|       } | ||||
|     }), | ||||
|   ); | ||||
| @@ -233,7 +223,7 @@ export const saveToFirebase = async ( | ||||
|     !socket || | ||||
|     isSavedToFirebase(portal, elements) | ||||
|   ) { | ||||
|     return null; | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   const firebase = await loadFirestore(); | ||||
| @@ -241,59 +231,56 @@ export const saveToFirebase = async ( | ||||
|  | ||||
|   const docRef = firestore.collection("scenes").doc(roomId); | ||||
|  | ||||
|   const storedScene = await firestore.runTransaction(async (transaction) => { | ||||
|   const savedData = await firestore.runTransaction(async (transaction) => { | ||||
|     const snapshot = await transaction.get(docRef); | ||||
|  | ||||
|     if (!snapshot.exists) { | ||||
|       const storedScene = await createFirebaseSceneDocument( | ||||
|       const sceneDocument = await createFirebaseSceneDocument( | ||||
|         firebase, | ||||
|         elements, | ||||
|         roomKey, | ||||
|       ); | ||||
|  | ||||
|       transaction.set(docRef, storedScene); | ||||
|       transaction.set(docRef, sceneDocument); | ||||
|  | ||||
|       return storedScene; | ||||
|       return { | ||||
|         elements, | ||||
|         reconciledElements: null, | ||||
|       }; | ||||
|     } | ||||
|  | ||||
|     const prevStoredScene = snapshot.data() as FirebaseStoredScene; | ||||
|     const prevStoredElements = getSyncableElements( | ||||
|       restoreElements(await decryptElements(prevStoredScene, roomKey), null), | ||||
|     ); | ||||
|     const reconciledElements = getSyncableElements( | ||||
|       reconcileElements( | ||||
|         elements, | ||||
|         prevStoredElements as OrderedExcalidrawElement[] as RemoteExcalidrawElement[], | ||||
|         appState, | ||||
|       ), | ||||
|     const prevDocData = snapshot.data() as FirebaseStoredScene; | ||||
|     const prevElements = getSyncableElements( | ||||
|       await decryptElements(prevDocData, roomKey), | ||||
|     ); | ||||
|  | ||||
|     const storedScene = await createFirebaseSceneDocument( | ||||
|     const reconciledElements = getSyncableElements( | ||||
|       reconcileElements(elements, prevElements, appState), | ||||
|     ); | ||||
|  | ||||
|     const sceneDocument = await createFirebaseSceneDocument( | ||||
|       firebase, | ||||
|       reconciledElements, | ||||
|       roomKey, | ||||
|     ); | ||||
|  | ||||
|     transaction.update(docRef, storedScene); | ||||
|  | ||||
|     // Return the stored elements as the in memory `reconciledElements` could have mutated in the meantime | ||||
|     return storedScene; | ||||
|     transaction.update(docRef, sceneDocument); | ||||
|     return { | ||||
|       elements, | ||||
|       reconciledElements, | ||||
|     }; | ||||
|   }); | ||||
|  | ||||
|   const storedElements = getSyncableElements( | ||||
|     restoreElements(await decryptElements(storedScene, roomKey), null), | ||||
|   ); | ||||
|   FirebaseSceneVersionCache.set(socket, savedData.elements); | ||||
|  | ||||
|   FirebaseSceneVersionCache.set(socket, storedElements); | ||||
|  | ||||
|   return storedElements; | ||||
|   return { reconciledElements: savedData.reconciledElements }; | ||||
| }; | ||||
|  | ||||
| export const loadFromFirebase = async ( | ||||
|   roomId: string, | ||||
|   roomKey: string, | ||||
|   socket: Socket | null, | ||||
| ): Promise<readonly SyncableExcalidrawElement[] | null> => { | ||||
|   socket: SocketIOClient.Socket | null, | ||||
| ): Promise<readonly ExcalidrawElement[] | null> => { | ||||
|   const firebase = await loadFirestore(); | ||||
|   const db = firebase.firestore(); | ||||
|  | ||||
| @@ -304,14 +291,14 @@ export const loadFromFirebase = async ( | ||||
|   } | ||||
|   const storedScene = doc.data() as FirebaseStoredScene; | ||||
|   const elements = getSyncableElements( | ||||
|     restoreElements(await decryptElements(storedScene, roomKey), null), | ||||
|     await decryptElements(storedScene, roomKey), | ||||
|   ); | ||||
|  | ||||
|   if (socket) { | ||||
|     FirebaseSceneVersionCache.set(socket, elements); | ||||
|   } | ||||
|  | ||||
|   return elements; | ||||
|   return restoreElements(elements, null); | ||||
| }; | ||||
|  | ||||
| export const loadFilesFromFirebase = async ( | ||||
|   | ||||
| @@ -1,34 +1,23 @@ | ||||
| import { | ||||
|   compressData, | ||||
|   decompressData, | ||||
| } from "../../packages/excalidraw/data/encode"; | ||||
| import { compressData, decompressData } from "../../src/data/encode"; | ||||
| import { | ||||
|   decryptData, | ||||
|   generateEncryptionKey, | ||||
|   IV_LENGTH_BYTES, | ||||
| } from "../../packages/excalidraw/data/encryption"; | ||||
| import { serializeAsJSON } from "../../packages/excalidraw/data/json"; | ||||
| import { restore } from "../../packages/excalidraw/data/restore"; | ||||
| import type { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import type { SceneBounds } from "../../packages/excalidraw/element/bounds"; | ||||
| import { isInvisiblySmallElement } from "../../packages/excalidraw/element/sizeHelpers"; | ||||
| import { isInitializedImageElement } from "../../packages/excalidraw/element/typeChecks"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { t } from "../../packages/excalidraw/i18n"; | ||||
| import type { | ||||
| } from "../../src/data/encryption"; | ||||
| import { serializeAsJSON } from "../../src/data/json"; | ||||
| import { restore } from "../../src/data/restore"; | ||||
| import { ImportedDataState } from "../../src/data/types"; | ||||
| import { isInvisiblySmallElement } from "../../src/element/sizeHelpers"; | ||||
| import { isInitializedImageElement } from "../../src/element/typeChecks"; | ||||
| import { ExcalidrawElement, FileId } from "../../src/element/types"; | ||||
| import { t } from "../../src/i18n"; | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFiles, | ||||
|   SocketId, | ||||
|   UserIdleState, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import type { MakeBrand } from "../../packages/excalidraw/utility-types"; | ||||
| import { bytesToHexString } from "../../packages/excalidraw/utils"; | ||||
| import type { WS_SUBTYPES } from "../app_constants"; | ||||
| } from "../../src/types"; | ||||
| import { bytesToHexString } from "../../src/utils"; | ||||
| import { | ||||
|   DELETED_ELEMENT_TIMEOUT, | ||||
|   FILE_UPLOAD_MAX_BYTES, | ||||
| @@ -37,11 +26,12 @@ import { | ||||
| import { encodeFilesForUpload } from "./FileManager"; | ||||
| import { saveFilesToFirebase } from "./firebase"; | ||||
|  | ||||
| export type SyncableExcalidrawElement = OrderedExcalidrawElement & | ||||
|   MakeBrand<"SyncableExcalidrawElement">; | ||||
| export type SyncableExcalidrawElement = ExcalidrawElement & { | ||||
|   _brand: "SyncableExcalidrawElement"; | ||||
| }; | ||||
|  | ||||
| export const isSyncableElement = ( | ||||
|   element: OrderedExcalidrawElement, | ||||
|   element: ExcalidrawElement, | ||||
| ): element is SyncableExcalidrawElement => { | ||||
|   if (element.isDeleted) { | ||||
|     if (element.updated > Date.now() - DELETED_ELEMENT_TIMEOUT) { | ||||
| @@ -52,9 +42,7 @@ export const isSyncableElement = ( | ||||
|   return !isInvisiblySmallElement(element); | ||||
| }; | ||||
|  | ||||
| export const getSyncableElements = ( | ||||
|   elements: readonly OrderedExcalidrawElement[], | ||||
| ) => | ||||
| export const getSyncableElements = (elements: readonly ExcalidrawElement[]) => | ||||
|   elements.filter((element) => | ||||
|     isSyncableElement(element), | ||||
|   ) as SyncableExcalidrawElement[]; | ||||
| @@ -68,49 +56,67 @@ const generateRoomId = async () => { | ||||
|   return bytesToHexString(buffer); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Right now the reason why we resolve connection params (url, polling...) | ||||
|  * from upstream is to allow changing the params immediately when needed without | ||||
|  * having to wait for clients to update the SW. | ||||
|  * | ||||
|  * If REACT_APP_WS_SERVER_URL env is set, we use that instead (useful for forks) | ||||
|  */ | ||||
| export const getCollabServer = async (): Promise<{ | ||||
|   url: string; | ||||
|   polling: boolean; | ||||
| }> => { | ||||
|   if (import.meta.env.VITE_APP_WS_SERVER_URL) { | ||||
|     return { | ||||
|       url: import.meta.env.VITE_APP_WS_SERVER_URL, | ||||
|       polling: true, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   try { | ||||
|     const resp = await fetch( | ||||
|       `${import.meta.env.VITE_APP_PORTAL_URL}/collab-server`, | ||||
|     ); | ||||
|     return await resp.json(); | ||||
|   } catch (error) { | ||||
|     console.error(error); | ||||
|     throw new Error(t("errors.cannotResolveCollabServer")); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export type EncryptedData = { | ||||
|   data: ArrayBuffer; | ||||
|   iv: Uint8Array; | ||||
| }; | ||||
|  | ||||
| export type SocketUpdateDataSource = { | ||||
|   INVALID_RESPONSE: { | ||||
|     type: WS_SUBTYPES.INVALID_RESPONSE; | ||||
|   }; | ||||
|   SCENE_INIT: { | ||||
|     type: WS_SUBTYPES.INIT; | ||||
|     type: "SCENE_INIT"; | ||||
|     payload: { | ||||
|       elements: readonly ExcalidrawElement[]; | ||||
|     }; | ||||
|   }; | ||||
|   SCENE_UPDATE: { | ||||
|     type: WS_SUBTYPES.UPDATE; | ||||
|     type: "SCENE_UPDATE"; | ||||
|     payload: { | ||||
|       elements: readonly ExcalidrawElement[]; | ||||
|     }; | ||||
|   }; | ||||
|   MOUSE_LOCATION: { | ||||
|     type: WS_SUBTYPES.MOUSE_LOCATION; | ||||
|     type: "MOUSE_LOCATION"; | ||||
|     payload: { | ||||
|       socketId: SocketId; | ||||
|       socketId: string; | ||||
|       pointer: { x: number; y: number; tool: "pointer" | "laser" }; | ||||
|       button: "down" | "up"; | ||||
|       selectedElementIds: AppState["selectedElementIds"]; | ||||
|       username: string; | ||||
|     }; | ||||
|   }; | ||||
|   USER_VISIBLE_SCENE_BOUNDS: { | ||||
|     type: WS_SUBTYPES.USER_VISIBLE_SCENE_BOUNDS; | ||||
|     payload: { | ||||
|       socketId: SocketId; | ||||
|       username: string; | ||||
|       sceneBounds: SceneBounds; | ||||
|     }; | ||||
|   }; | ||||
|   IDLE_STATUS: { | ||||
|     type: WS_SUBTYPES.IDLE_STATUS; | ||||
|     type: "IDLE_STATUS"; | ||||
|     payload: { | ||||
|       socketId: SocketId; | ||||
|       socketId: string; | ||||
|       userState: UserIdleState; | ||||
|       username: string; | ||||
|     }; | ||||
| @@ -118,7 +124,10 @@ export type SocketUpdateDataSource = { | ||||
| }; | ||||
|  | ||||
| export type SocketUpdateDataIncoming = | ||||
|   SocketUpdateDataSource[keyof SocketUpdateDataSource]; | ||||
|   | SocketUpdateDataSource[keyof SocketUpdateDataSource] | ||||
|   | { | ||||
|       type: "INVALID_RESPONSE"; | ||||
|     }; | ||||
|  | ||||
| export type SocketUpdateData = | ||||
|   SocketUpdateDataSource[keyof SocketUpdateDataSource] & { | ||||
| @@ -269,6 +278,7 @@ export const loadScene = async ( | ||||
|     // in the scene database/localStorage, and instead fetch them async | ||||
|     // from a different database | ||||
|     files: data.files, | ||||
|     commitToHistory: false, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,11 +1,12 @@ | ||||
| import type { ExcalidrawElement } from "../../packages/excalidraw/element/types"; | ||||
| import type { AppState } from "../../packages/excalidraw/types"; | ||||
| import { ExcalidrawElement } from "../../src/element/types"; | ||||
| import { AppState } from "../../src/types"; | ||||
| import { | ||||
|   clearAppStateForLocalStorage, | ||||
|   getDefaultAppState, | ||||
| } from "../../packages/excalidraw/appState"; | ||||
| import { clearElementsForLocalStorage } from "../../packages/excalidraw/element"; | ||||
| } from "../../src/appState"; | ||||
| import { clearElementsForLocalStorage } from "../../src/element"; | ||||
| import { STORAGE_KEYS } from "../app_constants"; | ||||
| import { ImportedDataState } from "../../src/data/types"; | ||||
|  | ||||
| export const saveUsernameToLocalStorage = (username: string) => { | ||||
|   try { | ||||
| @@ -87,13 +88,28 @@ export const getTotalStorageSize = () => { | ||||
|   try { | ||||
|     const appState = localStorage.getItem(STORAGE_KEYS.LOCAL_STORAGE_APP_STATE); | ||||
|     const collab = localStorage.getItem(STORAGE_KEYS.LOCAL_STORAGE_COLLAB); | ||||
|     const library = localStorage.getItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY); | ||||
|  | ||||
|     const appStateSize = appState?.length || 0; | ||||
|     const collabSize = collab?.length || 0; | ||||
|     const librarySize = library?.length || 0; | ||||
|  | ||||
|     return appStateSize + collabSize + getElementsStorageSize(); | ||||
|     return appStateSize + collabSize + librarySize + getElementsStorageSize(); | ||||
|   } catch (error: any) { | ||||
|     console.error(error); | ||||
|     return 0; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export const getLibraryItemsFromStorage = () => { | ||||
|   try { | ||||
|     const libraryItems: ImportedDataState["libraryItems"] = JSON.parse( | ||||
|       localStorage.getItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY) as string, | ||||
|     ); | ||||
|  | ||||
|     return libraryItems || []; | ||||
|   } catch (error) { | ||||
|     console.error(error); | ||||
|     return []; | ||||
|   } | ||||
| }; | ||||
|   | ||||
							
								
								
									
										3
									
								
								excalidraw-app/global.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								excalidraw-app/global.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +0,0 @@ | ||||
| interface Window { | ||||
|   __EXCALIDRAW_SHA__: string | undefined; | ||||
| } | ||||
| @@ -4,13 +4,6 @@ | ||||
|   &.theme--dark { | ||||
|     --color-primary-contrast-offset: #726dff; // to offset Chubb illusion | ||||
|   } | ||||
|  | ||||
|   .top-right-ui { | ||||
|     display: flex; | ||||
|     justify-content: center; | ||||
|     align-items: flex-start; | ||||
|   } | ||||
|  | ||||
|   .footer-center { | ||||
|     justify-content: flex-end; | ||||
|     margin-top: auto; | ||||
| @@ -25,7 +18,6 @@ | ||||
|     margin-bottom: auto; | ||||
|     margin-inline-start: auto; | ||||
|     margin-inline-end: 0.6em; | ||||
|     z-index: var(--zIndex-layerUI); | ||||
|  | ||||
|     svg { | ||||
|       width: 1.2rem; | ||||
| @@ -39,12 +31,8 @@ | ||||
|         background-color: #ecfdf5; | ||||
|         color: #064e3c; | ||||
|       } | ||||
|       &.highlighted { | ||||
|       &.ExcalidrawPlus { | ||||
|         color: var(--color-promo); | ||||
|         font-weight: 700; | ||||
|         .dropdown-menu-item__icon g { | ||||
|           stroke-width: 2; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user