mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-25 08:54:20 +02:00 
			
		
		
		
	Compare commits
	
		
			6 Commits
		
	
	
		
			mrazator/f
			...
			docs-next
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 77c4eb6db4 | ||
|   | 1ac2626e47 | ||
|   | 43d6a7e286 | ||
|   | 15b7d141c1 | ||
|   | 550e23a2ab | ||
|   | d4c6462ab1 | 
| @@ -4,16 +4,8 @@ | ||||
| !.eslintrc.json | ||||
| !.npmrc | ||||
| !.prettierrc | ||||
| !excalidraw-app/ | ||||
| !package.json | ||||
| !public/ | ||||
| !packages/ | ||||
| !scripts/ | ||||
| !tsconfig.json | ||||
| !yarn.lock | ||||
|  | ||||
| # keep (sub)sub directories at the end to exclude from explicit included | ||||
| # e.g. ./packages/excalidraw/{dist,node_modules} | ||||
| **/build | ||||
| **/dist | ||||
| **/node_modules | ||||
|   | ||||
| @@ -22,7 +22,7 @@ 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 | ||||
|  | ||||
|   | ||||
| @@ -14,4 +14,4 @@ VITE_APP_WS_SERVER_URL=https://oss-collab.excalidraw.com | ||||
|  | ||||
| VITE_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyAd15pYlMci_xIp9ko6wkEsDzAAA0Dn0RU","authDomain":"excalidraw-room-persistence.firebaseapp.com","databaseURL":"https://excalidraw-room-persistence.firebaseio.com","projectId":"excalidraw-room-persistence","storageBucket":"excalidraw-room-persistence.appspot.com","messagingSenderId":"654800341332","appId":"1:654800341332:web:4a692de832b55bd57ce0c1"}' | ||||
|  | ||||
| VITE_APP_ENABLE_TRACKING=false | ||||
| VITE_APP_DISABLE_TRACKING= | ||||
|   | ||||
| @@ -6,5 +6,3 @@ firebase/ | ||||
| dist/ | ||||
| public/workbox | ||||
| packages/excalidraw/types | ||||
| examples/**/public | ||||
| dev-dist | ||||
|   | ||||
| @@ -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" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,16 +1,14 @@ | ||||
| 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 | ||||
|   | ||||
							
								
								
									
										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 | ||||
|   | ||||
| @@ -8,15 +8,15 @@ | ||||
| import { FONT_FAMILY } from "@excalidraw/excalidraw"; | ||||
| ``` | ||||
|  | ||||
| `FONT_FAMILY` contains all the font families used in `Excalidraw` as explained below | ||||
| `FONT_FAMILY` contains all the font families used in `Excalidraw`. The default families are the following: | ||||
|  | ||||
| | Font Family | Description            | | ||||
| | ----------- | ---------------------- | | ||||
| | `Virgil`    | The `handwritten` font | | ||||
| | `Helvetica` | The `Normal` Font      | | ||||
| | `Cascadia`  | The `Code` Font        | | ||||
| | `Excalifont`    | The `Hand-drawn` font | | ||||
| | `Nunito` | The `Normal` Font      | | ||||
| | `Comic Shanns`  | The `Code` Font        | | ||||
|  | ||||
| Defaults to `FONT_FAMILY.Virgil` unless passed in `initialData.appState.currentItemFontFamily`. | ||||
| Pre-selected family is `FONT_FAMILY.Excalifont`, unless it's overriden with `initialData.appState.currentItemFontFamily`. | ||||
|  | ||||
| ### THEME | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ 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)}} />; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| @@ -65,7 +65,7 @@ You can use this function to update the scene with the sceneData. It accepts the | ||||
| | `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`. | | ||||
| | `storeAction` | [`StoreAction`](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/store.ts#L40) | Parameter to control which updates should be captured by the `Store`. Captured updates are emmitted as increments and listened to by other components, such as `History` for undo / redo purposes. | | ||||
|  | ||||
| ```jsx live | ||||
| function App() { | ||||
| @@ -105,6 +105,7 @@ function App() { | ||||
|       appState: { | ||||
|         viewBackgroundColor: "#edf2ff", | ||||
|       }, | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|     }; | ||||
|     excalidrawAPI.updateScene(sceneData); | ||||
|   }; | ||||
| @@ -115,12 +116,25 @@ function App() { | ||||
|       <button className="custom-button" onClick={updateScene}> | ||||
|         Update Scene | ||||
|       </button> | ||||
|       <Excalidraw excalidrawAPI={(api) => setExcalidrawAPI(api)} /> | ||||
|       <Excalidraw ref={(api) => setExcalidrawAPI(api)} /> | ||||
|     </div> | ||||
|   ); | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### storeAction | ||||
|  | ||||
| You can use the `storeAction` to influence undo / redo behaviour. | ||||
|  | ||||
| > **NOTE**: Some updates are not observed by the store / history - i.e. updates to `collaborators` object or parts of `AppState` which are not observed (not `ObservedAppState`). Such updates will never make it to the undo / redo stacks, regardless of the passed `storeAction` value. | ||||
|  | ||||
| |  | `storeAction` value | Notes | | ||||
| | --- | --- | --- | | ||||
| | _Immediately undoable_ | `StoreAction.CAPTURE` | Use for updates which should be captured. Should be used for most of the local updates. These updates will _immediately_ make it to the local undo / redo stacks. | | ||||
| | _Eventually undoable_ | `StoreAction.NONE` | Use for updates which should not be captured immediately - likely exceptions which are part of some async multi-step process. Otherwise, all such updates would end up being captured with the next `StoreAction.CAPTURE` - triggered either by the next `updateScene` or internally by the editor. These updates will _eventually_ make it to the local undo / redo stacks. | | ||||
| | _Never undoable_ | `StoreAction.UPDATE` | Use for updates which should never be recorded, such as remote updates or scene initialization. These updates will _never_ make it to the local undo / redo stacks. | | ||||
|  | ||||
|  | ||||
| ### updateLibrary | ||||
|  | ||||
| <pre> | ||||
| @@ -188,7 +202,7 @@ function App() { | ||||
|         Update Library | ||||
|       </button> | ||||
|       <Excalidraw | ||||
|         excalidrawAPI={(api) => setExcalidrawAPI(api)} | ||||
|         ref={(api) => setExcalidrawAPI(api)} | ||||
|         // initial data retrieved from https://github.com/excalidraw/excalidraw/blob/master/dev-docs/packages/excalidraw/initialData.js | ||||
|         initialData={{ | ||||
|           libraryItems: initialData.libraryItems, | ||||
|   | ||||
| @@ -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 | | ||||
| @@ -26,7 +26,7 @@ All `props` are _optional_. | ||||
| | [`UIOptions`](/docs/@excalidraw/excalidraw/api/props/ui-options) | `object` | [DEFAULT UI OPTIONS](https://github.com/excalidraw/excalidraw/blob/master/packages/excalidraw/constants.ts#L151) | To customise UI options. Currently we support customising [`canvas actions`](/docs/@excalidraw/excalidraw/api/props/ui-options#canvasactions) | | ||||
| | [`detectScroll`](#detectscroll) | `boolean` | `true` | Indicates whether to update the offsets when nearest ancestor is scrolled. | | ||||
| | [`handleKeyboardGlobally`](#handlekeyboardglobally) | `boolean` | `false` | Indicates whether to bind the keyboard events to document. | | ||||
| | [`autoFocus`](#autofocus) | `boolean` | `false` | Indicates whether to focus the Excalidraw component on page load | | ||||
| | [`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>` | | ||||
|   | ||||
| @@ -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> | ||||
|     </> | ||||
|   | ||||
| @@ -31,7 +31,7 @@ You can pass `null` / `undefined` if not applicable. | ||||
| 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/>  | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean }<br/> | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean, normalizeIndices?: boolean }<br/> | ||||
| ) | ||||
| </pre> | ||||
|  | ||||
| @@ -51,8 +51,9 @@ The extra optional parameter to configure restored elements. It has the followin | ||||
|  | ||||
| | Prop | Type | Description| | ||||
| | --- | --- | ------| | ||||
| | `refreshDimensions` | `boolean` | Indicates whether we should also `recalculate` text element dimensions. Since this is a potentially costly operation, you may want to disable it if you restore elements in tight loops, such as during collaboration. | | ||||
| | `repairBindings` |`boolean` | Indicates whether the `bindings` for the elements should be repaired. This is to make sure there are no containers with non existent bound text element id and no bound text elements with non existent container id. | | ||||
| | `refreshDimensions` | `boolean` | Indicates whether we should also _recalculate_ text element dimensions. Since this is a potentially costly operation, you may want to disable it if you restore elements in tight loops, such as during collaboration. | | ||||
| | `repairBindings` |`boolean` | Indicates whether the _bindings_ for the elements should be repaired. This is to make sure there are no containers with non existent bound text element id and no bound text elements with non existent container id. | | ||||
| | `normalizeIndices` |`boolean` | Indicates whether _fractional indices_ for the elements should be normalized. This is to prevent possible issues caused by using stale (too old, too long) indices. | | ||||
|  | ||||
| **_How to use_** | ||||
|  | ||||
| @@ -73,7 +74,7 @@ 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/> | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean }<br/> | ||||
|   opts: { refreshDimensions?: boolean, repairBindings?: boolean, normalizeIndices?: boolean }<br/> | ||||
|  | ||||
| ) | ||||
| </pre> | ||||
|   | ||||
| @@ -24,7 +24,7 @@ yarn add react react-dom @excalidraw/excalidraw | ||||
|  | ||||
| Excalidraw depends on assets such as localization files (if you opt to use them), fonts, and others. | ||||
|  | ||||
| By default these assets are loaded from a public CDN [`https://unpkg.com/@excalidraw/excalidraw/dist/`](https://unpkg.com/@excalidraw/excalidraw/dist), so you don't need to do anything on your end. | ||||
| By default these assets are loaded from a public CDN [`https://unpkg.com/@excalidraw/excalidraw/dist/prod/`](https://unpkg.com/@excalidraw/excalidraw/dist/prod/), so you don't need to do anything on your end. | ||||
|  | ||||
| However, if you want to host these files yourself, you can find them in your `node_modules/@excalidraw/excalidraw/dist` directory, in folders `excalidraw-assets` (for production) and `excalidraw-assets-dev` (for development). | ||||
|  | ||||
| @@ -34,6 +34,26 @@ Copy these folders to your static assets directory, and add a `window.EXCALIDRAW | ||||
| window.EXCALIDRAW_ASSET_PATH = "/"; | ||||
| ``` | ||||
|  | ||||
| or, if you serve your assets from the root of your CDN, you would do: | ||||
|  | ||||
| ```js | ||||
| // Vanilla | ||||
| <head> | ||||
|     <script> | ||||
|         window.EXCALIDRAW_ASSET_PATH = "https://my.cdn.com/assets/"; | ||||
|     </script>     | ||||
| </head> | ||||
| ``` | ||||
|  | ||||
| or, if you prefer the path to be dynamicly set based on the `location.origin`, you could do the following: | ||||
|  | ||||
| ```jsx | ||||
| // Next.js | ||||
| <Script id="load-env-variables" strategy="beforeInteractive" > | ||||
|     { `window["EXCALIDRAW_ASSET_PATH"] = location.origin;` } // or use just "/"! | ||||
| </Script> | ||||
| ``` | ||||
|  | ||||
| ### Dimensions of Excalidraw | ||||
|  | ||||
| Excalidraw takes _100%_ of `width` and `height` of the containing block so make sure the container in which you render Excalidraw has non zero dimensions. | ||||
|   | ||||
| @@ -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; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -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" | ||||
|   | ||||
| @@ -12,9 +12,9 @@ import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
|  | ||||
| import { nanoid } from "nanoid"; | ||||
|  | ||||
| import type { ResolvablePromise } from "../utils"; | ||||
| import { | ||||
|   resolvablePromise, | ||||
|   ResolvablePromise, | ||||
|   distance2d, | ||||
|   fileOpen, | ||||
|   withBatchedUpdates, | ||||
| @@ -872,7 +872,7 @@ export default function App({ | ||||
|                 files: excalidrawAPI.getFiles(), | ||||
|               }); | ||||
|               const ctx = canvas.getContext("2d")!; | ||||
|               ctx.font = "30px Excalifont"; | ||||
|               ctx.font = "30px Virgil"; | ||||
|               ctx.strokeText("My custom text", 50, 60); | ||||
|               setCanvasUrl(canvas.toDataURL()); | ||||
|             }} | ||||
| @@ -893,7 +893,7 @@ export default function App({ | ||||
|                 files: excalidrawAPI.getFiles(), | ||||
|               }); | ||||
|               const ctx = canvas.getContext("2d")!; | ||||
|               ctx.font = "30px Excalifont"; | ||||
|               ctx.font = "30px Virgil"; | ||||
|               ctx.strokeText("My custom text", 50, 60); | ||||
|               setCanvasUrl(canvas.toDataURL()); | ||||
|             }} | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import type { ExcalidrawImperativeAPI } from "@excalidraw/excalidraw/dist/excalidraw/types"; | ||||
| import { ExcalidrawImperativeAPI } from "@excalidraw/excalidraw/dist/excalidraw/types"; | ||||
| import CustomFooter from "./CustomFooter"; | ||||
| import type * as TExcalidraw from "@excalidraw/excalidraw"; | ||||
|  | ||||
|   | ||||
| @@ -46,7 +46,7 @@ const elements: ExcalidrawElementSkeleton[] = [ | ||||
| ]; | ||||
| export default { | ||||
|   elements, | ||||
|   appState: { viewBackgroundColor: "#AFEEEE", currentItemFontFamily: 5 }, | ||||
|   appState: { viewBackgroundColor: "#AFEEEE", currentItemFontFamily: 1 }, | ||||
|   scrollToContent: true, | ||||
|   libraryItems: [ | ||||
|     [ | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { unstable_batchedUpdates } from "react-dom"; | ||||
| import { fileOpen as _fileOpen } from "browser-fs-access"; | ||||
| import { MIME_TYPES } from "@excalidraw/excalidraw"; | ||||
| import type { MIME_TYPES } from "@excalidraw/excalidraw"; | ||||
| import { AbortError } from "../../packages/excalidraw/errors"; | ||||
|  | ||||
| type FILE_EXTENSION = Exclude<keyof typeof MIME_TYPES, "binary">; | ||||
|   | ||||
							
								
								
									
										3
									
								
								examples/excalidraw/with-nextjs/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								examples/excalidraw/with-nextjs/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -34,6 +34,3 @@ yarn-error.log* | ||||
| # typescript | ||||
| *.tsbuildinfo | ||||
| next-env.d.ts | ||||
|  | ||||
| # copied assets | ||||
| public/*.woff2 | ||||
| @@ -3,8 +3,7 @@ | ||||
|   "version": "0.1.0", | ||||
|   "private": true, | ||||
|   "scripts": { | ||||
|     "build:workspace": "yarn workspace @excalidraw/excalidraw run build:esm && yarn copy:assets", | ||||
|     "copy:assets": "cp ../../../packages/excalidraw/dist/browser/prod/excalidraw-assets/*.woff2 ./public", | ||||
|     "build:workspace": "yarn workspace @excalidraw/excalidraw run build:esm", | ||||
|     "dev": "yarn build:workspace && next dev -p 3005", | ||||
|     "build": "yarn build:workspace && next build", | ||||
|     "start": "next start -p 3006", | ||||
| @@ -13,13 +12,13 @@ | ||||
|   "dependencies": { | ||||
|     "@excalidraw/excalidraw": "*", | ||||
|     "next": "14.1", | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0" | ||||
|     "react": "^18", | ||||
|     "react-dom": "^18" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/node": "^20", | ||||
|     "@types/react": "18.2.0", | ||||
|     "@types/react-dom": "18.2.0", | ||||
|     "@types/react": "^18", | ||||
|     "@types/react-dom": "^18", | ||||
|     "path2d-polyfill": "2.0.1", | ||||
|     "typescript": "^5" | ||||
|   } | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| 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 | ||||
| @@ -16,9 +15,7 @@ export default function Page() { | ||||
|     <> | ||||
|       <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 /> | ||||
|     </> | ||||
|   | ||||
| @@ -7,7 +7,7 @@ a { | ||||
|   color: #1c7ed6; | ||||
|   font-size: 20px; | ||||
|   text-decoration: none; | ||||
|   font-weight: 500; | ||||
|   font-weight: 550; | ||||
| } | ||||
|  | ||||
| .page-title { | ||||
|   | ||||
| @@ -1,2 +0,0 @@ | ||||
| # copied assets | ||||
| public/*.woff2 | ||||
| @@ -11,7 +11,6 @@ | ||||
|     <title>React App</title> | ||||
|     <script> | ||||
|       window.name = "codesandbox"; | ||||
|       window.EXCALIDRAW_ASSET_PATH = window.origin; | ||||
|     </script> | ||||
|     <link rel="stylesheet" href="/dist/browser/dev/index.css" /> | ||||
|   </head> | ||||
|   | ||||
| @@ -12,10 +12,8 @@ | ||||
|     "typescript": "^5" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "build:workspace": "yarn workspace @excalidraw/excalidraw run build:esm && yarn copy:assets", | ||||
|     "copy:assets": "cp ../../../packages/excalidraw/dist/browser/prod/excalidraw-assets/*.woff2 ./public", | ||||
|     "start": "yarn build:workspace && vite", | ||||
|     "build": "yarn build:workspace && vite build", | ||||
|     "start": "yarn workspace @excalidraw/excalidraw run build:esm && vite", | ||||
|     "build": "yarn workspace @excalidraw/excalidraw run build:esm && vite build", | ||||
|     "build:preview": "yarn build && vite preview --port 5002" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import polyfill from "../packages/excalidraw/polyfill"; | ||||
| import LanguageDetector from "i18next-browser-languagedetector"; | ||||
| import { useCallback, useEffect, useRef, useState } from "react"; | ||||
| import { trackEvent } from "../packages/excalidraw/analytics"; | ||||
| import { getDefaultAppState } from "../packages/excalidraw/appState"; | ||||
| @@ -12,7 +13,7 @@ import { | ||||
|   VERSION_TIMEOUT, | ||||
| } from "../packages/excalidraw/constants"; | ||||
| import { loadFromBlob } from "../packages/excalidraw/data/blob"; | ||||
| import type { | ||||
| import { | ||||
|   FileId, | ||||
|   NonDeletedExcalidrawElement, | ||||
|   OrderedExcalidrawElement, | ||||
| @@ -21,26 +22,25 @@ import { useCallbackRefState } from "../packages/excalidraw/hooks/useCallbackRef | ||||
| import { t } from "../packages/excalidraw/i18n"; | ||||
| import { | ||||
|   Excalidraw, | ||||
|   defaultLang, | ||||
|   LiveCollaborationTrigger, | ||||
|   TTDDialog, | ||||
|   TTDDialogTrigger, | ||||
|   StoreAction, | ||||
|   reconcileElements, | ||||
| } from "../packages/excalidraw"; | ||||
| import type { | ||||
| } from "../packages/excalidraw/index"; | ||||
| import { | ||||
|   AppState, | ||||
|   ExcalidrawImperativeAPI, | ||||
|   BinaryFiles, | ||||
|   ExcalidrawInitialDataState, | ||||
|   UIAppState, | ||||
| } from "../packages/excalidraw/types"; | ||||
| import type { ResolvablePromise } from "../packages/excalidraw/utils"; | ||||
| import { | ||||
|   debounce, | ||||
|   getVersion, | ||||
|   getFrame, | ||||
|   isTestEnv, | ||||
|   preventUnload, | ||||
|   ResolvablePromise, | ||||
|   resolvablePromise, | ||||
|   isRunningInIframe, | ||||
| } from "../packages/excalidraw/utils"; | ||||
| @@ -50,8 +50,8 @@ import { | ||||
|   STORAGE_KEYS, | ||||
|   SYNC_BROWSER_TABS_TIMEOUT, | ||||
| } from "./app_constants"; | ||||
| import type { CollabAPI } from "./collab/Collab"; | ||||
| import Collab, { | ||||
|   CollabAPI, | ||||
|   collabAPIAtom, | ||||
|   isCollaboratingAtom, | ||||
|   isOfflineAtom, | ||||
| @@ -67,8 +67,11 @@ import { | ||||
|   importUsernameFromLocalStorage, | ||||
| } from "./data/localStorage"; | ||||
| import CustomStats from "./CustomStats"; | ||||
| import type { RestoredDataState } from "../packages/excalidraw/data/restore"; | ||||
| import { restore, restoreAppState } from "../packages/excalidraw/data/restore"; | ||||
| import { | ||||
|   restore, | ||||
|   restoreAppState, | ||||
|   RestoredDataState, | ||||
| } from "../packages/excalidraw/data/restore"; | ||||
| import { | ||||
|   ExportToExcalidrawPlus, | ||||
|   exportToExcalidrawPlus, | ||||
| @@ -91,19 +94,22 @@ import { | ||||
| import { AppMainMenu } from "./components/AppMainMenu"; | ||||
| import { AppWelcomeScreen } from "./components/AppWelcomeScreen"; | ||||
| import { AppFooter } from "./components/AppFooter"; | ||||
| import { Provider, useAtom, useAtomValue } from "jotai"; | ||||
| import { atom, Provider, useAtom, useAtomValue } from "jotai"; | ||||
| import { useAtomWithInitialValue } from "../packages/excalidraw/jotai"; | ||||
| import { appJotaiStore } from "./app-jotai"; | ||||
|  | ||||
| import "./index.scss"; | ||||
| import type { ResolutionType } from "../packages/excalidraw/utility-types"; | ||||
| import { ResolutionType } from "../packages/excalidraw/utility-types"; | ||||
| import { ShareableLinkDialog } from "../packages/excalidraw/components/ShareableLinkDialog"; | ||||
| import { openConfirmModal } from "../packages/excalidraw/components/OverwriteConfirm/OverwriteConfirmState"; | ||||
| import { OverwriteConfirmDialog } from "../packages/excalidraw/components/OverwriteConfirm/OverwriteConfirm"; | ||||
| import Trans from "../packages/excalidraw/components/Trans"; | ||||
| import { ShareDialog, shareDialogStateAtom } from "./share/ShareDialog"; | ||||
| import CollabError, { collabErrorIndicatorAtom } from "./collab/CollabError"; | ||||
| import type { RemoteExcalidrawElement } from "../packages/excalidraw/data/reconcile"; | ||||
| import { | ||||
|   RemoteExcalidrawElement, | ||||
|   reconcileElements, | ||||
| } from "../packages/excalidraw/data/reconcile"; | ||||
| import { | ||||
|   CommandPalette, | ||||
|   DEFAULT_CATEGORIES, | ||||
| @@ -119,45 +125,11 @@ import { | ||||
|   youtubeIcon, | ||||
| } from "../packages/excalidraw/components/icons"; | ||||
| import { appThemeAtom, useHandleAppTheme } from "./useHandleAppTheme"; | ||||
| import { getPreferredLanguage } from "./app-language/language-detector"; | ||||
| import { useAppLangCode } from "./app-language/language-state"; | ||||
|  | ||||
| polyfill(); | ||||
|  | ||||
| window.EXCALIDRAW_THROTTLE_RENDER = true; | ||||
|  | ||||
| declare global { | ||||
|   interface BeforeInstallPromptEventChoiceResult { | ||||
|     outcome: "accepted" | "dismissed"; | ||||
|   } | ||||
|  | ||||
|   interface BeforeInstallPromptEvent extends Event { | ||||
|     prompt(): Promise<void>; | ||||
|     userChoice: Promise<BeforeInstallPromptEventChoiceResult>; | ||||
|   } | ||||
|  | ||||
|   interface WindowEventMap { | ||||
|     beforeinstallprompt: BeforeInstallPromptEvent; | ||||
|   } | ||||
| } | ||||
|  | ||||
| let pwaEvent: BeforeInstallPromptEvent | null = null; | ||||
|  | ||||
| // Adding a listener outside of the component as it may (?) need to be | ||||
| // subscribed early to catch the event. | ||||
| // | ||||
| // Also note that it will fire only if certain heuristics are met (user has | ||||
| // used the app for some time, etc.) | ||||
| window.addEventListener( | ||||
|   "beforeinstallprompt", | ||||
|   (event: BeforeInstallPromptEvent) => { | ||||
|     // prevent Chrome <= 67 from automatically showing the prompt | ||||
|     event.preventDefault(); | ||||
|     // cache for later use | ||||
|     pwaEvent = event; | ||||
|   }, | ||||
| ); | ||||
|  | ||||
| let isSelfEmbedding = false; | ||||
|  | ||||
| if (window.self !== window.top) { | ||||
| @@ -172,6 +144,11 @@ if (window.self !== window.top) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| const languageDetector = new LanguageDetector(); | ||||
| languageDetector.init({ | ||||
|   languageUtils: {}, | ||||
| }); | ||||
|  | ||||
| const shareableLinkConfirmDialog = { | ||||
|   title: t("overwriteConfirm.modal.shareableLink.title"), | ||||
|   description: ( | ||||
| @@ -317,15 +294,19 @@ const initializeScene = async (opts: { | ||||
|   return { scene: null, isExternalScene: false }; | ||||
| }; | ||||
|  | ||||
| const detectedLangCode = languageDetector.detect() || defaultLang.code; | ||||
| export const appLangCodeAtom = atom( | ||||
|   Array.isArray(detectedLangCode) ? detectedLangCode[0] : detectedLangCode, | ||||
| ); | ||||
|  | ||||
| const ExcalidrawWrapper = () => { | ||||
|   const [errorMessage, setErrorMessage] = useState(""); | ||||
|   const [langCode, setLangCode] = useAtom(appLangCodeAtom); | ||||
|   const isCollabDisabled = isRunningInIframe(); | ||||
|  | ||||
|   const [appTheme, setAppTheme] = useAtom(appThemeAtom); | ||||
|   const { editorTheme } = useHandleAppTheme(); | ||||
|  | ||||
|   const [langCode, setLangCode] = useAppLangCode(); | ||||
|  | ||||
|   // initial state | ||||
|   // --------------------------------------------------------------------------- | ||||
|  | ||||
| @@ -457,7 +438,7 @@ const ExcalidrawWrapper = () => { | ||||
|             excalidrawAPI.updateScene({ | ||||
|               ...data.scene, | ||||
|               ...restore(data.scene, null, null, { repairBindings: true }), | ||||
|               storeAction: StoreAction.CAPTURE, | ||||
|               commitToStore: true, | ||||
|             }); | ||||
|           } | ||||
|         }); | ||||
| @@ -481,10 +462,13 @@ const ExcalidrawWrapper = () => { | ||||
|         if (isBrowserStorageStateNewer(STORAGE_KEYS.VERSION_DATA_STATE)) { | ||||
|           const localDataState = importFromLocalStorage(); | ||||
|           const username = importUsernameFromLocalStorage(); | ||||
|           setLangCode(getPreferredLanguage()); | ||||
|           let langCode = languageDetector.detect() || defaultLang.code; | ||||
|           if (Array.isArray(langCode)) { | ||||
|             langCode = langCode[0]; | ||||
|           } | ||||
|           setLangCode(langCode); | ||||
|           excalidrawAPI.updateScene({ | ||||
|             ...localDataState, | ||||
|             storeAction: StoreAction.UPDATE, | ||||
|           }); | ||||
|           LibraryIndexedDBAdapter.load().then((data) => { | ||||
|             if (data) { | ||||
| @@ -582,6 +566,10 @@ const ExcalidrawWrapper = () => { | ||||
|     }; | ||||
|   }, [excalidrawAPI]); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     languageDetector.cacheUserLanguage(langCode); | ||||
|   }, [langCode]); | ||||
|  | ||||
|   const onChange = ( | ||||
|     elements: readonly OrderedExcalidrawElement[], | ||||
|     appState: AppState, | ||||
| @@ -616,7 +604,6 @@ const ExcalidrawWrapper = () => { | ||||
|           if (didChange) { | ||||
|             excalidrawAPI.updateScene({ | ||||
|               elements, | ||||
|               storeAction: StoreAction.UPDATE, | ||||
|             }); | ||||
|           } | ||||
|         } | ||||
| @@ -1115,21 +1102,6 @@ const ExcalidrawWrapper = () => { | ||||
|                 ); | ||||
|               }, | ||||
|             }, | ||||
|             { | ||||
|               label: t("labels.installPWA"), | ||||
|               category: DEFAULT_CATEGORIES.app, | ||||
|               predicate: () => !!pwaEvent, | ||||
|               perform: () => { | ||||
|                 if (pwaEvent) { | ||||
|                   pwaEvent.prompt(); | ||||
|                   pwaEvent.userChoice.then(() => { | ||||
|                     // event cannot be reused, but we'll hopefully | ||||
|                     // grab new one as the event should be fired again | ||||
|                     pwaEvent = null; | ||||
|                   }); | ||||
|                 } | ||||
|               }, | ||||
|             }, | ||||
|           ]} | ||||
|         /> | ||||
|       </Excalidraw> | ||||
|   | ||||
| @@ -7,8 +7,8 @@ import { | ||||
| 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 { NonDeletedExcalidrawElement } from "../packages/excalidraw/element/types"; | ||||
| import { UIAppState } from "../packages/excalidraw/types"; | ||||
|  | ||||
| type StorageSizes = { scene: number; total: number }; | ||||
|  | ||||
|   | ||||
| @@ -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; | ||||
| }; | ||||
| @@ -1,25 +1,23 @@ | ||||
| import throttle from "lodash.throttle"; | ||||
| import { PureComponent } from "react"; | ||||
| import type { | ||||
| import { | ||||
|   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 { | ||||
| import { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   InitializedExcalidrawImageElement, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { | ||||
|   StoreAction, | ||||
|   getSceneVersion, | ||||
|   restoreElements, | ||||
|   zoomToFitBounds, | ||||
|   reconcileElements, | ||||
| } from "../../packages/excalidraw"; | ||||
| import type { Collaborator, Gesture } from "../../packages/excalidraw/types"; | ||||
| } from "../../packages/excalidraw/index"; | ||||
| import { Collaborator, Gesture } from "../../packages/excalidraw/types"; | ||||
| import { | ||||
|   assertNever, | ||||
|   preventUnload, | ||||
| @@ -36,14 +34,12 @@ import { | ||||
|   SYNC_FULL_SCENE_INTERVAL_MS, | ||||
|   WS_EVENTS, | ||||
| } from "../app_constants"; | ||||
| import type { | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
| import { | ||||
|   generateCollaborationLinkData, | ||||
|   getCollaborationLink, | ||||
|   getSyncableElements, | ||||
|   SocketUpdateDataSource, | ||||
|   SyncableExcalidrawElement, | ||||
| } from "../data"; | ||||
| import { | ||||
|   isSavedToFirebase, | ||||
| @@ -79,13 +75,14 @@ import { resetBrowserStateVersions } from "../data/tabSync"; | ||||
| import { LocalData } from "../data/LocalData"; | ||||
| import { atom } from "jotai"; | ||||
| import { appJotaiStore } from "../app-jotai"; | ||||
| import type { Mutable, ValueOf } from "../../packages/excalidraw/utility-types"; | ||||
| import { 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 { | ||||
| import { | ||||
|   ReconciledExcalidrawElement, | ||||
|   RemoteExcalidrawElement, | ||||
|   reconcileElements, | ||||
| } from "../../packages/excalidraw/data/reconcile"; | ||||
|  | ||||
| export const collabAPIAtom = atom<CollabAPI | null>(null); | ||||
| @@ -359,7 +356,6 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|  | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         elements, | ||||
|         storeAction: StoreAction.UPDATE, | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
| @@ -510,7 +506,6 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|       // to database even if deleted before creating the room. | ||||
|       this.excalidrawAPI.updateScene({ | ||||
|         elements, | ||||
|         storeAction: StoreAction.UPDATE, | ||||
|       }); | ||||
|  | ||||
|       this.saveCollabRoomToFirebase(getSyncableElements(elements)); | ||||
| @@ -748,7 +743,6 @@ class Collab extends PureComponent<CollabProps, CollabState> { | ||||
|   ) => { | ||||
|     this.excalidrawAPI.updateScene({ | ||||
|       elements, | ||||
|       storeAction: StoreAction.UPDATE, | ||||
|     }); | ||||
|  | ||||
|     this.loadImageFiles(); | ||||
|   | ||||
| @@ -1,15 +1,15 @@ | ||||
| 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 { OrderedExcalidrawElement } from "../../packages/excalidraw/element/types"; | ||||
| import { WS_EVENTS, FILE_UPLOAD_TIMEOUT, WS_SUBTYPES } from "../app_constants"; | ||||
| import type { | ||||
| import { | ||||
|   OnUserFollowedPayload, | ||||
|   SocketId, | ||||
|   UserIdleState, | ||||
| @@ -19,7 +19,6 @@ 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"; | ||||
|  | ||||
| class Portal { | ||||
|   collab: TCollabClass; | ||||
| @@ -128,7 +127,6 @@ class Portal { | ||||
|           } | ||||
|           return element; | ||||
|         }), | ||||
|       storeAction: StoreAction.UPDATE, | ||||
|     }); | ||||
|   }, FILE_UPLOAD_TIMEOUT); | ||||
|  | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| import React from "react"; | ||||
| import { | ||||
|   loginIcon, | ||||
|   arrowBarToLeftIcon, | ||||
|   ExcalLogo, | ||||
| } from "../../packages/excalidraw/components/icons"; | ||||
| import type { Theme } from "../../packages/excalidraw/element/types"; | ||||
| import { Theme } from "../../packages/excalidraw/element/types"; | ||||
| import { MainMenu } from "../../packages/excalidraw/index"; | ||||
| import { isExcalidrawPlusSignedUser } from "../app_constants"; | ||||
| import { LanguageList } from "../app-language/LanguageList"; | ||||
| import { LanguageList } from "./LanguageList"; | ||||
|  | ||||
| export const AppMainMenu: React.FC<{ | ||||
|   onCollabDialogOpen: () => any; | ||||
| @@ -34,7 +34,7 @@ export const AppMainMenu: React.FC<{ | ||||
|       <MainMenu.ItemLink | ||||
|         icon={ExcalLogo} | ||||
|         href={`${ | ||||
|           import.meta.env.VITE_APP_PLUS_LP | ||||
|           import.meta.env.VITE_APP_PLUS_APP | ||||
|         }/plus?utm_source=excalidraw&utm_medium=app&utm_content=hamburger`} | ||||
|         className="" | ||||
|       > | ||||
| @@ -42,7 +42,7 @@ export const AppMainMenu: React.FC<{ | ||||
|       </MainMenu.ItemLink> | ||||
|       <MainMenu.DefaultItems.Socials /> | ||||
|       <MainMenu.ItemLink | ||||
|         icon={loginIcon} | ||||
|         icon={arrowBarToLeftIcon} | ||||
|         href={`${import.meta.env.VITE_APP_PLUS_APP}${ | ||||
|           isExcalidrawPlusSignedUser ? "" : "/sign-up" | ||||
|         }?utm_source=signin&utm_medium=app&utm_content=hamburger`} | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from "react"; | ||||
| import { loginIcon } from "../../packages/excalidraw/components/icons"; | ||||
| import { arrowBarToLeftIcon } from "../../packages/excalidraw/components/icons"; | ||||
| import { useI18n } from "../../packages/excalidraw/i18n"; | ||||
| import { WelcomeScreen } from "../../packages/excalidraw/index"; | ||||
| import { isExcalidrawPlusSignedUser } from "../app_constants"; | ||||
| @@ -61,7 +61,7 @@ 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={arrowBarToLeftIcon} | ||||
|             > | ||||
|               Sign up | ||||
|             </WelcomeScreen.Center.MenuItemLink> | ||||
|   | ||||
| @@ -3,11 +3,11 @@ import { Card } from "../../packages/excalidraw/components/Card"; | ||||
| import { ToolButton } from "../../packages/excalidraw/components/ToolButton"; | ||||
| import { serializeAsJSON } from "../../packages/excalidraw/data/json"; | ||||
| import { loadFirebaseStorage, saveFilesToFirebase } from "../data/firebase"; | ||||
| import type { | ||||
| import { | ||||
|   FileId, | ||||
|   NonDeletedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import type { | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFiles, | ||||
|   | ||||
| @@ -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 "../../packages/excalidraw/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 "../App"; | ||||
| import { useI18n } from "../../packages/excalidraw/i18n"; | ||||
| import { languages } from "../../packages/excalidraw/i18n"; | ||||
| 
 | ||||
| export const LanguageList = ({ style }: { style?: React.CSSProperties }) => { | ||||
|   const { t, langCode } = useI18n(); | ||||
| @@ -1,15 +1,14 @@ | ||||
| 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 { | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawImageElement, | ||||
|   FileId, | ||||
|   InitializedExcalidrawImageElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { t } from "../../packages/excalidraw/i18n"; | ||||
| import type { | ||||
| import { | ||||
|   BinaryFileData, | ||||
|   BinaryFileMetadata, | ||||
|   ExcalidrawImperativeAPI, | ||||
| @@ -239,6 +238,5 @@ export const updateStaleImageStatuses = (params: { | ||||
|         } | ||||
|         return element; | ||||
|       }), | ||||
|     storeAction: StoreAction.UPDATE, | ||||
|   }); | ||||
| }; | ||||
|   | ||||
| @@ -20,19 +20,19 @@ import { | ||||
|   get, | ||||
| } from "idb-keyval"; | ||||
| import { clearAppStateForLocalStorage } from "../../packages/excalidraw/appState"; | ||||
| import type { LibraryPersistedData } from "../../packages/excalidraw/data/library"; | ||||
| import type { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import { LibraryPersistedData } from "../../packages/excalidraw/data/library"; | ||||
| import { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import { clearElementsForLocalStorage } from "../../packages/excalidraw/element"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import type { | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFiles, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import type { MaybePromise } from "../../packages/excalidraw/utility-types"; | ||||
| import { MaybePromise } from "../../packages/excalidraw/utility-types"; | ||||
| import { debounce } from "../../packages/excalidraw/utils"; | ||||
| import { SAVE_TO_LOCAL_STORAGE_TIMEOUT, STORAGE_KEYS } from "../app_constants"; | ||||
| import { FileManager } from "./FileManager"; | ||||
|   | ||||
| @@ -1,13 +1,12 @@ | ||||
| import { reconcileElements } from "../../packages/excalidraw"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { getSceneVersion } from "../../packages/excalidraw/element"; | ||||
| import type Portal from "../collab/Portal"; | ||||
| import Portal from "../collab/Portal"; | ||||
| import { restoreElements } from "../../packages/excalidraw/data/restore"; | ||||
| import type { | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFileMetadata, | ||||
| @@ -20,11 +19,13 @@ import { | ||||
|   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 { getSyncableElements, SyncableExcalidrawElement } from "."; | ||||
| import { ResolutionType } from "../../packages/excalidraw/utility-types"; | ||||
| import type { Socket } from "socket.io-client"; | ||||
| import type { RemoteExcalidrawElement } from "../../packages/excalidraw/data/reconcile"; | ||||
| import { | ||||
|   RemoteExcalidrawElement, | ||||
|   reconcileElements, | ||||
| } from "../../packages/excalidraw/data/reconcile"; | ||||
|  | ||||
| // private | ||||
| // ----------------------------------------------------------------------------- | ||||
|   | ||||
| @@ -9,30 +9,30 @@ import { | ||||
| } 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 { ImportedDataState } from "../../packages/excalidraw/data/types"; | ||||
| import { SceneBounds } from "../../packages/excalidraw/element/bounds"; | ||||
| import { isInvisiblySmallElement } from "../../packages/excalidraw/element/sizeHelpers"; | ||||
| import { isInitializedImageElement } from "../../packages/excalidraw/element/typeChecks"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   FileId, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../../packages/excalidraw/element/types"; | ||||
| import { t } from "../../packages/excalidraw/i18n"; | ||||
| import type { | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFileData, | ||||
|   BinaryFiles, | ||||
|   SocketId, | ||||
|   UserIdleState, | ||||
| } from "../../packages/excalidraw/types"; | ||||
| import type { MakeBrand } from "../../packages/excalidraw/utility-types"; | ||||
| import { MakeBrand } from "../../packages/excalidraw/utility-types"; | ||||
| import { bytesToHexString } from "../../packages/excalidraw/utils"; | ||||
| import type { WS_SUBTYPES } from "../app_constants"; | ||||
| import { | ||||
|   DELETED_ELEMENT_TIMEOUT, | ||||
|   FILE_UPLOAD_MAX_BYTES, | ||||
|   ROOM_ID_BYTES, | ||||
|   WS_SUBTYPES, | ||||
| } from "../app_constants"; | ||||
| import { encodeFilesForUpload } from "./FileManager"; | ||||
| import { saveFilesToFirebase } from "./firebase"; | ||||
| @@ -269,6 +269,7 @@ export const loadScene = async ( | ||||
|     // in the scene database/localStorage, and instead fetch them async | ||||
|     // from a different database | ||||
|     files: data.files, | ||||
|     commitToStore: false, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import type { ExcalidrawElement } from "../../packages/excalidraw/element/types"; | ||||
| import type { AppState } from "../../packages/excalidraw/types"; | ||||
| import { ExcalidrawElement } from "../../packages/excalidraw/element/types"; | ||||
| import { AppState } from "../../packages/excalidraw/types"; | ||||
| import { | ||||
|   clearAppStateForLocalStorage, | ||||
|   getDefaultAppState, | ||||
|   | ||||
| @@ -20,7 +20,7 @@ | ||||
|       name="description" | ||||
|       content="Excalidraw is a virtual collaborative whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them." | ||||
|     /> | ||||
|     <meta name="image" content="https://excalidraw.com/og-image-3.png" /> | ||||
|     <meta name="image" content="https://excalidraw.com/og-image-2.png" /> | ||||
|  | ||||
|     <!-- Open Graph / Facebook --> | ||||
|     <meta property="og:site_name" content="Excalidraw" /> | ||||
| @@ -35,7 +35,7 @@ | ||||
|       property="og:description" | ||||
|       content="Excalidraw is a virtual collaborative whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them." | ||||
|     /> | ||||
|     <meta property="og:image" content="https://excalidraw.com/og-image-3.png" /> | ||||
|     <meta property="og:image" content="https://excalidraw.com/og-image-2.png" /> | ||||
|  | ||||
|     <!-- Twitter --> | ||||
|     <meta property="twitter:card" content="summary_large_image" /> | ||||
| @@ -51,7 +51,7 @@ | ||||
|     /> | ||||
|     <meta | ||||
|       property="twitter:image" | ||||
|       content="https://excalidraw.com/og-image-3.png" | ||||
|       content="https://excalidraw.com/og-twitter-v2.png" | ||||
|     /> | ||||
|  | ||||
|     <!-- General tags --> | ||||
| @@ -95,11 +95,6 @@ | ||||
|         color: #fff; | ||||
|       } | ||||
|     </style> | ||||
|  | ||||
|     <!-- Warmup the connection for Google fonts --> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com" /> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin /> | ||||
|  | ||||
|     <!-------------------------------------------------------------------------> | ||||
|     <% if (typeof PROD != 'undefined' && PROD == true) { %> | ||||
|     <script> | ||||
| @@ -120,32 +115,8 @@ | ||||
|         window.location.href = "https://app.excalidraw.com"; | ||||
|       } | ||||
|     </script> | ||||
|  | ||||
|     <!-- Following placeholder is replaced during the build step --> | ||||
|     <!-- PLACEHOLDER:EXCALIDRAW_APP_FONTS --> | ||||
|  | ||||
|     <% } else { %> | ||||
|     <script> | ||||
|       window.EXCALIDRAW_ASSET_PATH = window.origin; | ||||
|     </script> | ||||
|     <% } %> | ||||
|  | ||||
|     <!-- For Nunito only preload the latin range, which should be good enough for now --> | ||||
|     <link | ||||
|       rel="preload" | ||||
|       href="https://fonts.gstatic.com/s/nunito/v26/XRXI3I6Li01BKofiOc5wtlZ2di8HDIkhdTQ3j6zbXWjgeg.woff2" | ||||
|       as="font" | ||||
|       type="font/woff2" | ||||
|       crossorigin="anonymous" | ||||
|     /> | ||||
|  | ||||
|     <!-- Register Assistant as the UI font, before the scene inits --> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="../packages/excalidraw/fonts/assets/fonts.css" | ||||
|       type="text/css" | ||||
|     /> | ||||
|  | ||||
|     <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" /> | ||||
|     <link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png" /> | ||||
|     <link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png" /> | ||||
| @@ -153,6 +124,22 @@ | ||||
|     <!-- Excalidraw version --> | ||||
|     <meta name="version" content="{version}" /> | ||||
|  | ||||
|     <link | ||||
|       rel="preload" | ||||
|       href="/Virgil.woff2" | ||||
|       as="font" | ||||
|       type="font/woff2" | ||||
|       crossorigin="anonymous" | ||||
|     /> | ||||
|     <link | ||||
|       rel="preload" | ||||
|       href="/Cascadia.woff2" | ||||
|       as="font" | ||||
|       type="font/woff2" | ||||
|       crossorigin="anonymous" | ||||
|     /> | ||||
|  | ||||
|     <link rel="stylesheet" href="/fonts/fonts.css" type="text/css" /> | ||||
|     <% if (typeof VITE_APP_DEV_DISABLE_LIVE_RELOAD != 'undefined' && | ||||
|     VITE_APP_DEV_DISABLE_LIVE_RELOAD == true) { %> | ||||
|     <script> | ||||
| @@ -171,6 +158,7 @@ | ||||
|     </script> | ||||
|     <% } %> | ||||
|     <script> | ||||
|       window.EXCALIDRAW_ASSET_PATH = "/"; | ||||
|       // setting this so that libraries installation reuses this window tab. | ||||
|       window.name = "_excalidraw"; | ||||
|     </script> | ||||
|   | ||||
| @@ -25,7 +25,6 @@ | ||||
|     margin-bottom: auto; | ||||
|     margin-inline-start: auto; | ||||
|     margin-inline-end: 0.6em; | ||||
|     z-index: var(--zIndex-layerUI); | ||||
|  | ||||
|     svg { | ||||
|       width: 1.2rem; | ||||
| @@ -41,10 +40,6 @@ | ||||
|       } | ||||
|       &.highlighted { | ||||
|         color: var(--color-promo); | ||||
|         font-weight: 700; | ||||
|         .dropdown-menu-item__icon g { | ||||
|           stroke-width: 2; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -26,28 +26,17 @@ | ||||
|     "node": ">=18.0.0" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "firebase": "8.3.3", | ||||
|     "idb-keyval": "6.0.3", | ||||
|     "jotai": "1.13.1", | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0", | ||||
|     "vite-plugin-html": "3.2.2", | ||||
|     "@excalidraw/random-username": "1.0.0", | ||||
|     "@sentry/browser": "6.2.5", | ||||
|     "@sentry/integrations": "6.2.5", | ||||
|     "i18next-browser-languagedetector": "6.1.4", | ||||
|     "socket.io-client": "4.7.2" | ||||
|     "vite-plugin-html": "3.2.2" | ||||
|   }, | ||||
|   "prettier": "@excalidraw/prettier-config", | ||||
|   "scripts": { | ||||
|     "build-node": "node ./scripts/build-node.js", | ||||
|     "build:app:docker": "cross-env VITE_APP_DISABLE_SENTRY=true vite build", | ||||
|     "build:app": "cross-env VITE_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA cross-env VITE_APP_ENABLE_TRACKING=true vite build", | ||||
|     "build:app:docker": "cross-env VITE_APP_DISABLE_SENTRY=true VITE_APP_DISABLE_TRACKING=true vite build", | ||||
|     "build:app": "cross-env VITE_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA vite build", | ||||
|     "build:version": "node ../scripts/build-version.js", | ||||
|     "build": "yarn build:app && yarn build:version", | ||||
|     "start": "yarn && vite", | ||||
|     "start:production": "yarn build && yarn serve", | ||||
|     "serve": "npx http-server build -a localhost -p 5001 -o", | ||||
|     "start:production": "npm run build && npx http-server build -a localhost -p 5001 -o", | ||||
|     "build:preview": "yarn build && vite preview --port 5000" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -18,8 +18,7 @@ import { | ||||
| } from "../../packages/excalidraw/components/icons"; | ||||
| import { TextField } from "../../packages/excalidraw/components/TextField"; | ||||
| import { FilledButton } from "../../packages/excalidraw/components/FilledButton"; | ||||
| import type { CollabAPI } from "../collab/Collab"; | ||||
| import { activeRoomLinkAtom } from "../collab/Collab"; | ||||
| import { activeRoomLinkAtom, CollabAPI } from "../collab/Collab"; | ||||
| import { atom, useAtom, useAtomValue } from "jotai"; | ||||
|  | ||||
| import "./ShareDialog.scss"; | ||||
|   | ||||
| @@ -5,7 +5,7 @@ exports[`Test MobileMenu > should initialize with welcome screen and hide once u | ||||
|   class="welcome-screen-center" | ||||
| > | ||||
|   <div | ||||
|     class="welcome-screen-center__logo excalifont welcome-screen-decor" | ||||
|     class="welcome-screen-center__logo virgil welcome-screen-decor" | ||||
|   > | ||||
|     <div | ||||
|       class="ExcalidrawLogo is-small" | ||||
| @@ -48,7 +48,7 @@ exports[`Test MobileMenu > should initialize with welcome screen and hide once u | ||||
|     </div> | ||||
|   </div> | ||||
|   <div | ||||
|     class="welcome-screen-center__heading welcome-screen-decor excalifont" | ||||
|     class="welcome-screen-center__heading welcome-screen-decor virgil" | ||||
|   > | ||||
|     All your data is saved locally in your browser. | ||||
|   </div> | ||||
| @@ -216,22 +216,23 @@ exports[`Test MobileMenu > should initialize with welcome screen and hide once u | ||||
|           stroke-width="2" | ||||
|           viewBox="0 0 24 24" | ||||
|         > | ||||
|           <g | ||||
|             stroke-width="1.5" | ||||
|           > | ||||
|           <g> | ||||
|             <path | ||||
|               d="M0 0h24v24H0z" | ||||
|               fill="none" | ||||
|               stroke="none" | ||||
|             /> | ||||
|             <path | ||||
|               d="M15 8v-2a2 2 0 0 0 -2 -2h-7a2 2 0 0 0 -2 2v12a2 2 0 0 0 2 2h7a2 2 0 0 0 2 -2v-2" | ||||
|               d="M10 12l10 0" | ||||
|             /> | ||||
|             <path | ||||
|               d="M21 12h-13l3 -3" | ||||
|               d="M10 12l4 4" | ||||
|             /> | ||||
|             <path | ||||
|               d="M11 15l-3 -3" | ||||
|               d="M10 12l4 -4" | ||||
|             /> | ||||
|             <path | ||||
|               d="M4 4l0 16" | ||||
|             /> | ||||
|           </g> | ||||
|         </svg> | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import { vi } from "vitest"; | ||||
| import { | ||||
|   act, | ||||
|   render, | ||||
|   updateSceneData, | ||||
|   waitFor, | ||||
| } from "../../packages/excalidraw/tests/test-utils"; | ||||
| import ExcalidrawApp from "../App"; | ||||
| @@ -11,7 +12,7 @@ import { | ||||
|   createRedoAction, | ||||
|   createUndoAction, | ||||
| } from "../../packages/excalidraw/actions/actionHistory"; | ||||
| import { StoreAction, newElementWith } from "../../packages/excalidraw"; | ||||
| import { newElementWith } from "../../packages/excalidraw"; | ||||
|  | ||||
| const { h } = window; | ||||
|  | ||||
| @@ -87,17 +88,17 @@ describe("collaboration", () => { | ||||
|     const rect1 = API.createElement({ ...rect1Props }); | ||||
|     const rect2 = API.createElement({ ...rect2Props }); | ||||
|  | ||||
|     API.updateScene({ | ||||
|     updateSceneData({ | ||||
|       elements: syncInvalidIndices([rect1, rect2]), | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|       commitToStore: true, | ||||
|     }); | ||||
|  | ||||
|     API.updateScene({ | ||||
|     updateSceneData({ | ||||
|       elements: syncInvalidIndices([ | ||||
|         rect1, | ||||
|         newElementWith(h.elements[1], { isDeleted: true }), | ||||
|       ]), | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|       commitToStore: true, | ||||
|     }); | ||||
|  | ||||
|     await waitFor(() => { | ||||
| @@ -142,9 +143,8 @@ describe("collaboration", () => { | ||||
|     }); | ||||
|  | ||||
|     // simulate force deleting the element remotely | ||||
|     API.updateScene({ | ||||
|     updateSceneData({ | ||||
|       elements: syncInvalidIndices([rect1]), | ||||
|       storeAction: StoreAction.UPDATE, | ||||
|     }); | ||||
|  | ||||
|     await waitFor(() => { | ||||
| @@ -177,12 +177,12 @@ describe("collaboration", () => { | ||||
|     act(() => h.app.actionManager.executeAction(undoAction)); | ||||
|  | ||||
|     // simulate local update | ||||
|     API.updateScene({ | ||||
|     updateSceneData({ | ||||
|       elements: syncInvalidIndices([ | ||||
|         h.elements[0], | ||||
|         newElementWith(h.elements[1], { x: 100 }), | ||||
|       ]), | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|       commitToStore: true, | ||||
|     }); | ||||
|  | ||||
|     await waitFor(() => { | ||||
| @@ -215,9 +215,8 @@ describe("collaboration", () => { | ||||
|     }); | ||||
|  | ||||
|     // simulate force deleting the element remotely | ||||
|     API.updateScene({ | ||||
|     updateSceneData({ | ||||
|       elements: syncInvalidIndices([rect1]), | ||||
|       storeAction: StoreAction.UPDATE, | ||||
|     }); | ||||
|  | ||||
|     // snapshot was correctly updated and marked the element as deleted | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { atom, useAtom } from "jotai"; | ||||
| import { useEffect, useLayoutEffect, useState } from "react"; | ||||
| import { THEME } from "../packages/excalidraw"; | ||||
| import { EVENT } from "../packages/excalidraw/constants"; | ||||
| import type { Theme } from "../packages/excalidraw/element/types"; | ||||
| import { Theme } from "../packages/excalidraw/element/types"; | ||||
| import { CODES, KEYS } from "../packages/excalidraw/keys"; | ||||
| import { STORAGE_KEYS } from "./app_constants"; | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,6 @@ import { ViteEjsPlugin } from "vite-plugin-ejs"; | ||||
| import { VitePWA } from "vite-plugin-pwa"; | ||||
| import checker from "vite-plugin-checker"; | ||||
| import { createHtmlPlugin } from "vite-plugin-html"; | ||||
| import { woff2BrowserPlugin } from "../scripts/woff2/woff2-vite-plugins"; | ||||
|  | ||||
| // To load .env.local variables | ||||
| const envVars = loadEnv("", `../`); | ||||
| @@ -23,14 +22,6 @@ export default defineConfig({ | ||||
|     outDir: "build", | ||||
|     rollupOptions: { | ||||
|       output: { | ||||
|         assetFileNames(chunkInfo) { | ||||
|           if (chunkInfo?.name?.endsWith(".woff2")) { | ||||
|             // put on root so we are flexible about the CDN path | ||||
|             return "[name]-[hash][extname]"; | ||||
|           } | ||||
|  | ||||
|           return "assets/[name]-[hash][extname]"; | ||||
|         }, | ||||
|         // Creating separate chunk for locales except for en and percentages.json so they | ||||
|         // can be cached at runtime and not merged with | ||||
|         // app precache. en.json and percentages.json are needed for first load | ||||
| @@ -50,7 +41,6 @@ export default defineConfig({ | ||||
|     sourcemap: true, | ||||
|   }, | ||||
|   plugins: [ | ||||
|     woff2BrowserPlugin(), | ||||
|     react(), | ||||
|     checker({ | ||||
|       typescript: true, | ||||
| @@ -73,8 +63,8 @@ export default defineConfig({ | ||||
|       }, | ||||
|  | ||||
|       workbox: { | ||||
|         // Don't push fonts, locales and wasm to app precache | ||||
|         globIgnores: ["fonts.css", "**/locales/**", "service-worker.js", "**/*.wasm-*.js"], | ||||
|         // Don't push fonts and locales to app precache | ||||
|         globIgnores: ["fonts.css", "**/locales/**", "service-worker.js"], | ||||
|         runtimeCaching: [ | ||||
|           { | ||||
|             urlPattern: new RegExp("/.+.(ttf|woff2|otf)"), | ||||
| @@ -108,17 +98,6 @@ export default defineConfig({ | ||||
|               }, | ||||
|             }, | ||||
|           }, | ||||
|           { | ||||
|             urlPattern: new RegExp(".wasm-.+.js"), | ||||
|             handler: "CacheFirst", | ||||
|             options: { | ||||
|               cacheName: "wasm", | ||||
|               expiration: { | ||||
|                 maxEntries: 50, | ||||
|                 maxAgeSeconds: 60 * 60 * 24 * 90, // <== 90 days | ||||
|               }, | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       manifest: { | ||||
|   | ||||
							
								
								
									
										34
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,7 +1,6 @@ | ||||
| { | ||||
|   "private": true, | ||||
|   "name": "excalidraw-monorepo", | ||||
|   "packageManager": "yarn@1.22.22", | ||||
|   "workspaces": [ | ||||
|     "excalidraw-app", | ||||
|     "packages/excalidraw", | ||||
| @@ -9,15 +8,26 @@ | ||||
|     "examples/excalidraw", | ||||
|     "examples/excalidraw/*" | ||||
|   ], | ||||
|   "dependencies": { | ||||
|     "@excalidraw/random-username": "1.0.0", | ||||
|     "@sentry/browser": "6.2.5", | ||||
|     "@sentry/integrations": "6.2.5", | ||||
|     "firebase": "8.3.3", | ||||
|     "i18next-browser-languagedetector": "6.1.4", | ||||
|     "idb-keyval": "6.0.3", | ||||
|     "jotai": "1.13.1", | ||||
|     "react": "18.2.0", | ||||
|     "react-dom": "18.2.0", | ||||
|     "socket.io-client": "4.7.2" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@babel/plugin-proposal-private-property-in-object": "7.21.11", | ||||
|     "@excalidraw/eslint-config": "1.0.3", | ||||
|     "@excalidraw/prettier-config": "1.0.2", | ||||
|     "@types/chai": "4.3.0", | ||||
|     "@types/jest": "27.4.0", | ||||
|     "@types/lodash.throttle": "4.1.7", | ||||
|     "@types/react": "18.2.0", | ||||
|     "@types/react-dom": "18.2.0", | ||||
|     "@types/react": "18.0.15", | ||||
|     "@types/react-dom": "18.0.6", | ||||
|     "@types/socket.io-client": "3.0.0", | ||||
|     "@vitejs/plugin-react": "3.1.0", | ||||
|     "@vitest/coverage-v8": "0.33.0", | ||||
| @@ -40,7 +50,7 @@ | ||||
|     "vite-plugin-ejs": "1.7.0", | ||||
|     "vite-plugin-pwa": "0.17.4", | ||||
|     "vite-plugin-svgr": "2.4.0", | ||||
|     "vitest": "1.6.0", | ||||
|     "vitest": "1.0.1", | ||||
|     "vitest-canvas-mock": "0.3.2" | ||||
|   }, | ||||
|   "engines": { | ||||
| @@ -50,9 +60,9 @@ | ||||
|   "prettier": "@excalidraw/prettier-config", | ||||
|   "scripts": { | ||||
|     "build-node": "node ./scripts/build-node.js", | ||||
|     "build:app:docker": "yarn --cwd ./excalidraw-app build:app:docker", | ||||
|     "build:app": "yarn --cwd ./excalidraw-app build:app", | ||||
|     "build:version": "yarn --cwd ./excalidraw-app build:version", | ||||
|     "build:app:docker": "cross-env VITE_APP_DISABLE_SENTRY=true VITE_APP_DISABLE_TRACKING=true vite build", | ||||
|     "build:app": "cross-env VITE_APP_GIT_SHA=$VERCEL_GIT_COMMIT_SHA vite build", | ||||
|     "build:version": "node ./scripts/build-version.js", | ||||
|     "build": "yarn --cwd ./excalidraw-app build", | ||||
|     "fix:code": "yarn test:code --fix", | ||||
|     "fix:other": "yarn prettier --write", | ||||
| @@ -76,12 +86,6 @@ | ||||
|     "autorelease": "node scripts/autorelease.js", | ||||
|     "prerelease:excalidraw": "node scripts/prerelease.js", | ||||
|     "build:preview": "yarn build && vite preview --port 5000", | ||||
|     "release:excalidraw": "node scripts/release.js", | ||||
|     "rm:build": "rm -rf excalidraw-app/{build,dist,dev-dist} && rm -rf packages/*/{dist,build} && rm -rf examples/*/*/{build,dist}", | ||||
|     "rm:node_modules": "rm -rf node_modules && rm -rf excalidraw-app/node_modules && rm -rf packages/*/node_modules", | ||||
|     "clean-install": "yarn rm:node_modules && yarn install" | ||||
|   }, | ||||
|   "resolutions": { | ||||
|     "@types/react": "18.2.0" | ||||
|     "release:excalidraw": "node scripts/release.js" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -15,12 +15,8 @@ Please add the latest change on the top under the correct section. | ||||
|  | ||||
| ### Features | ||||
|  | ||||
| - `props.initialData` can now be a function that returns `ExcalidrawInitialDataState` or `Promise<ExcalidrawInitialDataState>`. [#8107](https://github.com/excalidraw/excalidraw/pull/8135) | ||||
|  | ||||
| - Added support for multiplayer undo/redo, by calculating invertible increments and storing them inside the local-only undo/redo stacks. [#7348](https://github.com/excalidraw/excalidraw/pull/7348) | ||||
|  | ||||
| - Added font picker component to have the ability to choose from a range of different fonts. Also, changed the default fonts to `Excalifont`, `Nunito` and `Comic Shanns` and deprecated `Virgil`, `Helvetica` and `Cascadia`. | ||||
|  | ||||
| - `MainMenu.DefaultItems.ToggleTheme` now supports `onSelect(theme: string)` callback, and optionally `allowSystemTheme: boolean` alongside `theme: string` to indicate you want to allow users to set to system theme (you need to handle this yourself). [#7853](https://github.com/excalidraw/excalidraw/pull/7853) | ||||
|  | ||||
| - Add `useHandleLibrary`'s `opts.adapter` as the new recommended pattern to handle library initialization and persistence on library updates. [#7655](https://github.com/excalidraw/excalidraw/pull/7655) | ||||
| @@ -33,19 +29,17 @@ Please add the latest change on the top under the correct section. | ||||
|  | ||||
| - Expose `getVisibleSceneBounds` helper to get scene bounds of visible canvas area. [#7450](https://github.com/excalidraw/excalidraw/pull/7450) | ||||
|  | ||||
| - Extended `window.EXCALIDRAW_ASSET_PATH` to accept array of paths `string[]` as a value, allowing to specify multiple base `URL` fallbacks. [#8286](https://github.com/excalidraw/excalidraw/pull/8286) | ||||
|  | ||||
| ### Fixes | ||||
|  | ||||
| - Keep customData when converting to ExcalidrawElement. [#7656](https://github.com/excalidraw/excalidraw/pull/7656) | ||||
|  | ||||
| ### Breaking Changes | ||||
|  | ||||
| - `updateScene` API has changed due to the added `Store` component as part of the multiplayer undo / redo initiative. Specifically, `sceneData` property `commitToHistory: boolean` was replaced with `storeAction: StoreActionType`. Make sure to update all instances of `updateScene` according to the _before / after_ table below. [#7898](https://github.com/excalidraw/excalidraw/pull/7898) | ||||
| - Renamed required `updatedScene` parameter from `commitToHistory` into `commitToStore` [#7348](https://github.com/excalidraw/excalidraw/pull/7348). | ||||
|  | ||||
| |  | Before `commitToHistory` | After `storeAction` | Notes | | ||||
| | --- | --- | --- | --- | | ||||
| | _Immediately undoable_ | `true` | `"capture"` | As before, use for all updates which should be recorded by the store & history. Should be used for the most of the local updates. These updates will _immediately_ make it to the local undo / redo stacks. | | ||||
| | _Eventually undoable_ | `false` | `"none"` | Similar to before, use for all updates which should not be recorded immediately (likely exceptions which are part of some async multi-step process) or those not meant to be recorded at all (i.e. updates to `collaborators` object, parts of `AppState` which are not observed by the store & history - not `ObservedAppState`).<br/><br/>**IMPORTANT** It's likely you should switch to `"update"` in all the other cases. Otherwise, all such updates would end up being recorded with the next `"capture"` - triggered either by the next `updateScene` or internally by the editor. These updates will _eventually_ make it to the local undo / redo stacks. | | ||||
| | _Never undoable_ | n/a | `"update"` | **NEW**: previously there was no equivalent for this value. Now, it's recommended to use `"update"` for all remote updates (from the other clients), scene initialization, or those updates, which should not be locally "undoable". These updates will _never_ make it to the local undo / redo stacks. | | ||||
| ### Breaking Changes | ||||
|  | ||||
| - `ExcalidrawEmbeddableElement.validated` was removed and moved to private editor state. This should largely not affect your apps unless you were reading from this attribute. We keep validating embeddable urls internally, and the public [`props.validateEmbeddable`](https://docs.excalidraw.com/docs/@excalidraw/excalidraw/api/props#validateembeddable) still applies. [#7539](https://github.com/excalidraw/excalidraw/pull/7539) | ||||
|  | ||||
|   | ||||
| @@ -20,7 +20,7 @@ After installation you will see a folder `excalidraw-assets` and `excalidraw-ass | ||||
|  | ||||
| Move the folder `excalidraw-assets` and `excalidraw-assets-dev` to the path where your assets are served. | ||||
|  | ||||
| By default it will try to load the files from [`https://unpkg.com/@excalidraw/excalidraw/dist/`](https://unpkg.com/@excalidraw/excalidraw/dist) | ||||
| By default it will try to load the files from [`https://unpkg.com/@excalidraw/excalidraw/dist/prod/`](https://unpkg.com/@excalidraw/excalidraw/dist/prod/) | ||||
|  | ||||
| If you want to load assets from a different path you can set a variable `window.EXCALIDRAW_ASSET_PATH` depending on environment (for example if you have different URL's for dev and prod) to the url from where you want to load the assets. | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| import type { Alignment } from "../align"; | ||||
| import { alignElements } from "../align"; | ||||
| import { alignElements, Alignment } from "../align"; | ||||
| import { | ||||
|   AlignBottomIcon, | ||||
|   AlignLeftIcon, | ||||
| @@ -11,13 +10,13 @@ import { | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { isFrameLikeElement } from "../element/typeChecks"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { updateFrameMembershipOfSelectedElements } from "../frame"; | ||||
| import { t } from "../i18n"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { isSomeElementSelected } from "../scene"; | ||||
| import { StoreAction } from "../store"; | ||||
| import type { AppClassProperties, AppState, UIAppState } from "../types"; | ||||
| import { AppClassProperties, AppState, UIAppState } from "../types"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { | ||||
|   BOUND_TEXT_PADDING, | ||||
|   ROUNDNESS, | ||||
|   TEXT_ALIGN, | ||||
|   VERTICAL_ALIGN, | ||||
|   TEXT_ALIGN, | ||||
| } from "../constants"; | ||||
| import { isTextElement, newElement } from "../element"; | ||||
| import { mutateElement } from "../element/mutateElement"; | ||||
| @@ -23,14 +23,14 @@ import { | ||||
|   isTextBindableContainer, | ||||
|   isUsingAdaptiveRadius, | ||||
| } from "../element/typeChecks"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawLinearElement, | ||||
|   ExcalidrawTextContainer, | ||||
|   ExcalidrawTextElement, | ||||
| } from "../element/types"; | ||||
| import type { AppState } from "../types"; | ||||
| import type { Mutable } from "../utility-types"; | ||||
| import { AppState } from "../types"; | ||||
| import { Mutable } from "../utility-types"; | ||||
| import { arrayToMap, getFontString } from "../utils"; | ||||
| import { register } from "./register"; | ||||
| import { syncMovedIndices } from "../fractionalIndex"; | ||||
| @@ -142,7 +142,6 @@ export const actionBindText = register({ | ||||
|       containerId: container.id, | ||||
|       verticalAlign: VERTICAL_ALIGN.MIDDLE, | ||||
|       textAlign: TEXT_ALIGN.CENTER, | ||||
|       autoResize: true, | ||||
|     }); | ||||
|     mutateElement(container, { | ||||
|       boundElements: (container.boundElements || []).concat({ | ||||
| @@ -297,7 +296,6 @@ export const actionWrapTextInContainer = register({ | ||||
|             verticalAlign: VERTICAL_ALIGN.MIDDLE, | ||||
|             boundElements: null, | ||||
|             textAlign: TEXT_ALIGN.CENTER, | ||||
|             autoResize: true, | ||||
|           }, | ||||
|           false, | ||||
|         ); | ||||
|   | ||||
| @@ -18,13 +18,13 @@ import { | ||||
|   ZOOM_STEP, | ||||
| } from "../constants"; | ||||
| import { getCommonBounds, getNonDeletedElements } from "../element"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { getNormalizedZoom } from "../scene"; | ||||
| import { centerScrollOn } from "../scene/scroll"; | ||||
| import { getStateForZoom } from "../scene/zoom"; | ||||
| import type { AppState, NormalizedZoomValue } from "../types"; | ||||
| import { AppState, NormalizedZoomValue } from "../types"; | ||||
| import { getShortcutKey, updateActiveTool } from "../utils"; | ||||
| import { register } from "./register"; | ||||
| import { Tooltip } from "../components/Tooltip"; | ||||
| @@ -35,7 +35,7 @@ import { | ||||
|   isHandToolActive, | ||||
| } from "../appState"; | ||||
| import { DEFAULT_CANVAS_BACKGROUND_PICKS } from "../colors"; | ||||
| import type { SceneBounds } from "../element/bounds"; | ||||
| import { SceneBounds } from "../element/bounds"; | ||||
| import { setCursor } from "../cursor"; | ||||
| import { StoreAction } from "../store"; | ||||
|  | ||||
| @@ -104,7 +104,7 @@ export const actionClearCanvas = register({ | ||||
|         exportBackground: appState.exportBackground, | ||||
|         exportEmbedScene: appState.exportEmbedScene, | ||||
|         gridSize: appState.gridSize, | ||||
|         stats: appState.stats, | ||||
|         showStats: appState.showStats, | ||||
|         pasteDialog: appState.pasteDialog, | ||||
|         activeTool: | ||||
|           appState.activeTool.type === "image" | ||||
|   | ||||
| @@ -4,28 +4,21 @@ import { ToolButton } from "../components/ToolButton"; | ||||
| import { t } from "../i18n"; | ||||
| import { register } from "./register"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import type { AppClassProperties, AppState } from "../types"; | ||||
| import { mutateElement, newElementWith } from "../element/mutateElement"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppState } from "../types"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { getElementsInGroup } from "../groups"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
| import { fixBindingsAfterDeletion } from "../element/binding"; | ||||
| import { | ||||
|   isBoundToContainer, | ||||
|   isElbowArrow, | ||||
|   isFrameLikeElement, | ||||
| } from "../element/typeChecks"; | ||||
| import { isBoundToContainer, isFrameLikeElement } from "../element/typeChecks"; | ||||
| import { updateActiveTool } from "../utils"; | ||||
| import { TrashIcon } from "../components/icons"; | ||||
| import { StoreAction } from "../store"; | ||||
| import { mutateElbowArrow } from "../element/routing"; | ||||
|  | ||||
| const deleteSelectedElements = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   appState: AppState, | ||||
|   app: AppClassProperties, | ||||
| ) => { | ||||
|   const elementsMap = app.scene.getNonDeletedElementsMap(); | ||||
|   const framesToBeDeleted = new Set( | ||||
|     getSelectedElements( | ||||
|       elements.filter((el) => isFrameLikeElement(el)), | ||||
| @@ -36,26 +29,6 @@ const deleteSelectedElements = ( | ||||
|   return { | ||||
|     elements: elements.map((el) => { | ||||
|       if (appState.selectedElementIds[el.id]) { | ||||
|         if (el.boundElements) { | ||||
|           el.boundElements.forEach((candidate) => { | ||||
|             const bound = app.scene | ||||
|               .getNonDeletedElementsMap() | ||||
|               .get(candidate.id); | ||||
|             if (bound && isElbowArrow(bound)) { | ||||
|               mutateElement(bound, { | ||||
|                 startBinding: | ||||
|                   el.id === bound.startBinding?.elementId | ||||
|                     ? null | ||||
|                     : bound.startBinding, | ||||
|                 endBinding: | ||||
|                   el.id === bound.endBinding?.elementId | ||||
|                     ? null | ||||
|                     : bound.endBinding, | ||||
|               }); | ||||
|               mutateElbowArrow(bound, elementsMap, bound.points); | ||||
|             } | ||||
|           }); | ||||
|         } | ||||
|         return newElementWith(el, { isDeleted: true }); | ||||
|       } | ||||
|  | ||||
| @@ -157,11 +130,7 @@ export const actionDeleteSelected = register({ | ||||
|           : endBindingElement, | ||||
|       }; | ||||
|  | ||||
|       LinearElementEditor.deletePoints( | ||||
|         element, | ||||
|         selectedPointsIndices, | ||||
|         elementsMap, | ||||
|       ); | ||||
|       LinearElementEditor.deletePoints(element, selectedPointsIndices); | ||||
|  | ||||
|       return { | ||||
|         elements, | ||||
| @@ -180,7 +149,7 @@ export const actionDeleteSelected = register({ | ||||
|       }; | ||||
|     } | ||||
|     let { elements: nextElements, appState: nextAppState } = | ||||
|       deleteSelectedElements(elements, appState, app); | ||||
|       deleteSelectedElements(elements, appState); | ||||
|     fixBindingsAfterDeletion( | ||||
|       nextElements, | ||||
|       elements.filter(({ id }) => appState.selectedElementIds[id]), | ||||
|   | ||||
| @@ -3,17 +3,16 @@ import { | ||||
|   DistributeVerticallyIcon, | ||||
| } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import type { Distribution } from "../distribute"; | ||||
| import { distributeElements } from "../distribute"; | ||||
| import { distributeElements, Distribution } from "../distribute"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { isFrameLikeElement } from "../element/typeChecks"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { updateFrameMembershipOfSelectedElements } from "../frame"; | ||||
| import { t } from "../i18n"; | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { isSomeElementSelected } from "../scene"; | ||||
| import { StoreAction } from "../store"; | ||||
| import type { AppClassProperties, AppState } from "../types"; | ||||
| import { AppClassProperties, AppState } from "../types"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { duplicateElement, getNonDeletedElements } from "../element"; | ||||
| import { isSomeElementSelected } from "../scene"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| @@ -12,9 +12,9 @@ import { | ||||
|   getSelectedGroupForElement, | ||||
|   getElementsInGroup, | ||||
| } from "../groups"; | ||||
| import type { AppState } from "../types"; | ||||
| import { AppState } from "../types"; | ||||
| import { fixBindingsAfterDuplication } from "../element/binding"; | ||||
| import type { ActionResult } from "./types"; | ||||
| import { ActionResult } from "./types"; | ||||
| import { GRID_SIZE } from "../constants"; | ||||
| import { | ||||
|   bindTextToShapeAfterDuplication, | ||||
| @@ -40,11 +40,12 @@ export const actionDuplicateSelection = register({ | ||||
|   icon: DuplicateIcon, | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState, formData, app) => { | ||||
|     const elementsMap = app.scene.getNonDeletedElementsMap(); | ||||
|     // duplicate selected point(s) if editing a line | ||||
|     if (appState.editingLinearElement) { | ||||
|       const ret = LinearElementEditor.duplicateSelectedPoints( | ||||
|         appState, | ||||
|         app.scene.getNonDeletedElementsMap(), | ||||
|         elementsMap, | ||||
|       ); | ||||
|  | ||||
|       if (!ret) { | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import React from "react"; | ||||
| import { Excalidraw } from "../index"; | ||||
| import { queryByTestId, fireEvent } from "@testing-library/react"; | ||||
| import { render } from "../tests/test-utils"; | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { LockedIcon, UnlockedIcon } from "../components/icons"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { isFrameLikeElement } from "../element/typeChecks"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { StoreAction } from "../store"; | ||||
|   | ||||
| @@ -16,7 +16,7 @@ import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { isImageFileHandle } from "../data/blob"; | ||||
| import { nativeFileSystemSupported } from "../data/filesystem"; | ||||
| import type { Theme } from "../element/types"; | ||||
| import { Theme } from "../element/types"; | ||||
|  | ||||
| import "../components/ToolIcon.scss"; | ||||
| import { StoreAction } from "../store"; | ||||
|   | ||||
| @@ -13,7 +13,7 @@ import { | ||||
|   bindOrUnbindLinearElement, | ||||
| } from "../element/binding"; | ||||
| import { isBindingElement, isLinearElement } from "../element/typeChecks"; | ||||
| import type { AppState } from "../types"; | ||||
| import { AppState } from "../types"; | ||||
| import { resetCursor } from "../cursor"; | ||||
| import { StoreAction } from "../store"; | ||||
|  | ||||
| @@ -38,7 +38,6 @@ export const actionFinalize = register({ | ||||
|             startBindingElement, | ||||
|             endBindingElement, | ||||
|             elementsMap, | ||||
|             scene, | ||||
|           ); | ||||
|         } | ||||
|         return { | ||||
| @@ -73,8 +72,8 @@ export const actionFinalize = register({ | ||||
|  | ||||
|     const multiPointElement = appState.multiElement | ||||
|       ? appState.multiElement | ||||
|       : appState.newElement?.type === "freedraw" | ||||
|       ? appState.newElement | ||||
|       : appState.editingElement?.type === "freedraw" | ||||
|       ? appState.editingElement | ||||
|       : null; | ||||
|  | ||||
|     if (multiPointElement) { | ||||
| @@ -132,13 +131,7 @@ export const actionFinalize = register({ | ||||
|           -1, | ||||
|           arrayToMap(elements), | ||||
|         ); | ||||
|         maybeBindLinearElement( | ||||
|           multiPointElement, | ||||
|           appState, | ||||
|           { x, y }, | ||||
|           elementsMap, | ||||
|           elements, | ||||
|         ); | ||||
|         maybeBindLinearElement(multiPointElement, appState, { x, y }, app); | ||||
|       } | ||||
|     } | ||||
|  | ||||
| @@ -176,8 +169,7 @@ export const actionFinalize = register({ | ||||
|             ? appState.activeTool | ||||
|             : activeTool, | ||||
|         activeEmbeddable: null, | ||||
|         newElement: null, | ||||
|         selectionElement: null, | ||||
|         draggingElement: null, | ||||
|         multiElement: null, | ||||
|         editingElement: null, | ||||
|         startBoundElement: null, | ||||
| @@ -205,7 +197,7 @@ export const actionFinalize = register({ | ||||
|   keyTest: (event, appState) => | ||||
|     (event.key === KEYS.ESCAPE && | ||||
|       (appState.editingLinearElement !== null || | ||||
|         (!appState.newElement && appState.multiElement === null))) || | ||||
|         (!appState.draggingElement && appState.multiElement === null))) || | ||||
|     ((event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) && | ||||
|       appState.multiElement !== null), | ||||
|   PanelComponent: ({ appState, updateData, data }) => ( | ||||
|   | ||||
| @@ -1,24 +1,24 @@ | ||||
| import { register } from "./register"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   NonDeleted, | ||||
|   NonDeletedSceneElementsMap, | ||||
| } from "../element/types"; | ||||
| import { resizeMultipleElements } from "../element/resizeElements"; | ||||
| import type { AppClassProperties, AppState } from "../types"; | ||||
| import { AppClassProperties, AppState } from "../types"; | ||||
| import { arrayToMap } from "../utils"; | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { getCommonBoundingBox } from "../element/bounds"; | ||||
| import { | ||||
|   bindOrUnbindLinearElements, | ||||
|   bindOrUnbindSelectedElements, | ||||
|   isBindingEnabled, | ||||
|   unbindLinearElements, | ||||
| } from "../element/binding"; | ||||
| import { updateFrameMembershipOfSelectedElements } from "../frame"; | ||||
| import { flipHorizontal, flipVertical } from "../components/icons"; | ||||
| import { StoreAction } from "../store"; | ||||
| import { isLinearElement } from "../element/typeChecks"; | ||||
|  | ||||
| export const actionFlipHorizontal = register({ | ||||
|   name: "flipHorizontal", | ||||
| @@ -89,6 +89,7 @@ const flipSelectedElements = ( | ||||
|  | ||||
|   const updatedElements = flipElements( | ||||
|     selectedElements, | ||||
|     elements, | ||||
|     elementsMap, | ||||
|     appState, | ||||
|     flipDirection, | ||||
| @@ -104,6 +105,7 @@ const flipSelectedElements = ( | ||||
|  | ||||
| const flipElements = ( | ||||
|   selectedElements: NonDeleted<ExcalidrawElement>[], | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   elementsMap: NonDeletedSceneElementsMap, | ||||
|   appState: AppState, | ||||
|   flipDirection: "horizontal" | "vertical", | ||||
| @@ -117,19 +119,13 @@ const flipElements = ( | ||||
|     elementsMap, | ||||
|     "nw", | ||||
|     true, | ||||
|     true, | ||||
|     flipDirection === "horizontal" ? maxX : minX, | ||||
|     flipDirection === "horizontal" ? minY : maxY, | ||||
|   ); | ||||
|  | ||||
|   bindOrUnbindLinearElements( | ||||
|     selectedElements.filter(isLinearElement), | ||||
|     elementsMap, | ||||
|     app.scene.getNonDeletedElements(), | ||||
|     app.scene, | ||||
|     isBindingEnabled(appState), | ||||
|     [], | ||||
|   ); | ||||
|   isBindingEnabled(appState) | ||||
|     ? bindOrUnbindSelectedElements(selectedElements, app) | ||||
|     : unbindLinearElements(selectedElements, elementsMap); | ||||
|  | ||||
|   return selectedElements; | ||||
| }; | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { removeAllElementsFromFrame } from "../frame"; | ||||
| import { getFrameChildren } from "../frame"; | ||||
| import { KEYS } from "../keys"; | ||||
| import type { AppClassProperties, AppState, UIAppState } from "../types"; | ||||
| import { AppClassProperties, AppState, UIAppState } from "../types"; | ||||
| import { updateActiveTool } from "../utils"; | ||||
| import { setCursorForShape } from "../cursor"; | ||||
| import { register } from "./register"; | ||||
|   | ||||
| @@ -17,12 +17,12 @@ import { | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { randomId } from "../random"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawTextElement, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../element/types"; | ||||
| import type { AppClassProperties, AppState } from "../types"; | ||||
| import { AppClassProperties, AppState } from "../types"; | ||||
| import { isBoundToContainer } from "../element/typeChecks"; | ||||
| import { | ||||
|   getElementsInResizingFrame, | ||||
|   | ||||
| @@ -1,20 +1,17 @@ | ||||
| import type { Action, ActionResult } from "./types"; | ||||
| import { Action, ActionResult } from "./types"; | ||||
| import { UndoIcon, RedoIcon } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { t } from "../i18n"; | ||||
| import type { History } from "../history"; | ||||
| import { HistoryChangedEvent } from "../history"; | ||||
| import type { AppClassProperties, AppState } from "../types"; | ||||
| import { History, HistoryChangedEvent } from "../history"; | ||||
| import { AppState } from "../types"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { arrayToMap } from "../utils"; | ||||
| import { isWindows } from "../constants"; | ||||
| import type { SceneElementsMap } from "../element/types"; | ||||
| import type { Store } from "../store"; | ||||
| import { StoreAction } from "../store"; | ||||
| import { SceneElementsMap } from "../element/types"; | ||||
| import { IStore, StoreAction } from "../store"; | ||||
| import { useEmitter } from "../hooks/useEmitter"; | ||||
|  | ||||
| const executeHistoryAction = ( | ||||
|   app: AppClassProperties, | ||||
| const writeData = ( | ||||
|   appState: Readonly<AppState>, | ||||
|   updater: () => [SceneElementsMap, AppState] | void, | ||||
| ): ActionResult => { | ||||
| @@ -22,10 +19,7 @@ const executeHistoryAction = ( | ||||
|     !appState.multiElement && | ||||
|     !appState.resizingElement && | ||||
|     !appState.editingElement && | ||||
|     !appState.newElement && | ||||
|     !appState.selectedElementsAreBeingDragged && | ||||
|     !appState.selectionElement && | ||||
|     !app.flowChartCreator.isCreatingChart | ||||
|     !appState.draggingElement | ||||
|   ) { | ||||
|     const result = updater(); | ||||
|  | ||||
| @@ -46,7 +40,7 @@ const executeHistoryAction = ( | ||||
|   return { storeAction: StoreAction.NONE }; | ||||
| }; | ||||
|  | ||||
| type ActionCreator = (history: History, store: Store) => Action; | ||||
| type ActionCreator = (history: History, store: IStore) => Action; | ||||
|  | ||||
| export const createUndoAction: ActionCreator = (history, store) => ({ | ||||
|   name: "undo", | ||||
| @@ -54,8 +48,8 @@ export const createUndoAction: ActionCreator = (history, store) => ({ | ||||
|   icon: UndoIcon, | ||||
|   trackEvent: { category: "history" }, | ||||
|   viewMode: false, | ||||
|   perform: (elements, appState, value, app) => | ||||
|     executeHistoryAction(app, appState, () => | ||||
|   perform: (elements, appState) => | ||||
|     writeData(appState, () => | ||||
|       history.undo( | ||||
|         arrayToMap(elements) as SceneElementsMap, // TODO: #7348 refactor action manager to already include `SceneElementsMap` | ||||
|         appState, | ||||
| @@ -69,10 +63,7 @@ export const createUndoAction: ActionCreator = (history, store) => ({ | ||||
|   PanelComponent: ({ updateData, data }) => { | ||||
|     const { isUndoStackEmpty } = useEmitter<HistoryChangedEvent>( | ||||
|       history.onHistoryChangedEmitter, | ||||
|       new HistoryChangedEvent( | ||||
|         history.isUndoStackEmpty, | ||||
|         history.isRedoStackEmpty, | ||||
|       ), | ||||
|       new HistoryChangedEvent(), | ||||
|     ); | ||||
|  | ||||
|     return ( | ||||
| @@ -83,7 +74,6 @@ export const createUndoAction: ActionCreator = (history, store) => ({ | ||||
|         onClick={updateData} | ||||
|         size={data?.size || "medium"} | ||||
|         disabled={isUndoStackEmpty} | ||||
|         data-testid="button-undo" | ||||
|       /> | ||||
|     ); | ||||
|   }, | ||||
| @@ -95,8 +85,8 @@ export const createRedoAction: ActionCreator = (history, store) => ({ | ||||
|   icon: RedoIcon, | ||||
|   trackEvent: { category: "history" }, | ||||
|   viewMode: false, | ||||
|   perform: (elements, appState, _, app) => | ||||
|     executeHistoryAction(app, appState, () => | ||||
|   perform: (elements, appState) => | ||||
|     writeData(appState, () => | ||||
|       history.redo( | ||||
|         arrayToMap(elements) as SceneElementsMap, // TODO: #7348 refactor action manager to already include `SceneElementsMap` | ||||
|         appState, | ||||
| @@ -111,10 +101,7 @@ export const createRedoAction: ActionCreator = (history, store) => ({ | ||||
|   PanelComponent: ({ updateData, data }) => { | ||||
|     const { isRedoStackEmpty } = useEmitter( | ||||
|       history.onHistoryChangedEmitter, | ||||
|       new HistoryChangedEvent( | ||||
|         history.isUndoStackEmpty, | ||||
|         history.isRedoStackEmpty, | ||||
|       ), | ||||
|       new HistoryChangedEvent(), | ||||
|     ); | ||||
|  | ||||
|     return ( | ||||
| @@ -125,7 +112,6 @@ export const createRedoAction: ActionCreator = (history, store) => ({ | ||||
|         onClick={updateData} | ||||
|         size={data?.size || "medium"} | ||||
|         disabled={isRedoStackEmpty} | ||||
|         data-testid="button-redo" | ||||
|       /> | ||||
|     ); | ||||
|   }, | ||||
|   | ||||
| @@ -1,12 +1,9 @@ | ||||
| import { DEFAULT_CATEGORIES } from "../components/CommandPalette/CommandPalette"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
| import { isElbowArrow, isLinearElement } from "../element/typeChecks"; | ||||
| import type { ExcalidrawLinearElement } from "../element/types"; | ||||
| import { isLinearElement } from "../element/typeChecks"; | ||||
| import { ExcalidrawLinearElement } from "../element/types"; | ||||
| import { StoreAction } from "../store"; | ||||
| import { register } from "./register"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { t } from "../i18n"; | ||||
| import { lineEditorIcon } from "../components/icons"; | ||||
| 
 | ||||
| export const actionToggleLinearEditor = register({ | ||||
|   name: "toggleLinearEditor", | ||||
| @@ -14,24 +11,18 @@ export const actionToggleLinearEditor = register({ | ||||
|   label: (elements, appState, app) => { | ||||
|     const selectedElement = app.scene.getSelectedElements({ | ||||
|       selectedElementIds: appState.selectedElementIds, | ||||
|     })[0] as ExcalidrawLinearElement | undefined; | ||||
| 
 | ||||
|     return selectedElement?.type === "arrow" | ||||
|       ? "labels.lineEditor.editArrow" | ||||
|       includeBoundTextElement: true, | ||||
|     })[0] as ExcalidrawLinearElement; | ||||
|     return appState.editingLinearElement?.elementId === selectedElement?.id | ||||
|       ? "labels.lineEditor.exit" | ||||
|       : "labels.lineEditor.edit"; | ||||
|   }, | ||||
|   keywords: ["line"], | ||||
|   trackEvent: { | ||||
|     category: "element", | ||||
|   }, | ||||
|   predicate: (elements, appState, _, app) => { | ||||
|     const selectedElements = app.scene.getSelectedElements(appState); | ||||
|     if ( | ||||
|       !appState.editingLinearElement && | ||||
|       selectedElements.length === 1 && | ||||
|       isLinearElement(selectedElements[0]) && | ||||
|       !isElbowArrow(selectedElements[0]) | ||||
|     ) { | ||||
|     if (selectedElements.length === 1 && isLinearElement(selectedElements[0])) { | ||||
|       return true; | ||||
|     } | ||||
|     return false; | ||||
| @@ -54,24 +45,4 @@ export const actionToggleLinearEditor = register({ | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ appState, updateData, app }) => { | ||||
|     const selectedElement = app.scene.getSelectedElements({ | ||||
|       selectedElementIds: appState.selectedElementIds, | ||||
|     })[0] as ExcalidrawLinearElement; | ||||
| 
 | ||||
|     const label = t( | ||||
|       selectedElement.type === "arrow" | ||||
|         ? "labels.lineEditor.editArrow" | ||||
|         : "labels.lineEditor.edit", | ||||
|     ); | ||||
|     return ( | ||||
|       <ToolButton | ||||
|         type="button" | ||||
|         icon={lineEditorIcon} | ||||
|         title={label} | ||||
|         aria-label={label} | ||||
|         onClick={() => updateData(null)} | ||||
|       /> | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { getClientColor } from "../clients"; | ||||
| import { Avatar } from "../components/Avatar"; | ||||
| import type { GoToCollaboratorComponentProps } from "../components/UserList"; | ||||
| import { GoToCollaboratorComponentProps } from "../components/UserList"; | ||||
| import { | ||||
|   eyeIcon, | ||||
|   microphoneIcon, | ||||
| @@ -8,7 +8,7 @@ import { | ||||
| } from "../components/icons"; | ||||
| import { t } from "../i18n"; | ||||
| import { StoreAction } from "../store"; | ||||
| import type { Collaborator } from "../types"; | ||||
| import { Collaborator } from "../types"; | ||||
| import { register } from "./register"; | ||||
| import clsx from "clsx"; | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import React from "react"; | ||||
| import { Excalidraw } from "../index"; | ||||
| import { queryByTestId } from "@testing-library/react"; | ||||
| import { render } from "../tests/test-utils"; | ||||
| @@ -7,6 +6,8 @@ import { API } from "../tests/helpers/api"; | ||||
| import { COLOR_PALETTE, DEFAULT_ELEMENT_BACKGROUND_PICKS } from "../colors"; | ||||
| import { FONT_FAMILY, STROKE_WIDTH } from "../constants"; | ||||
|  | ||||
| const { h } = window; | ||||
|  | ||||
| describe("element locking", () => { | ||||
|   beforeEach(async () => { | ||||
|     await render(<Excalidraw />); | ||||
| @@ -21,7 +22,7 @@ describe("element locking", () => { | ||||
|       // just in case we change it in the future | ||||
|       expect(color).not.toBe(COLOR_PALETTE.transparent); | ||||
|  | ||||
|       API.setAppState({ | ||||
|       h.setState({ | ||||
|         currentItemBackgroundColor: color, | ||||
|       }); | ||||
|       const activeColor = queryByTestId( | ||||
| @@ -39,14 +40,14 @@ describe("element locking", () => { | ||||
|       // just in case we change it in the future | ||||
|       expect(color).not.toBe(COLOR_PALETTE.transparent); | ||||
|  | ||||
|       API.setAppState({ | ||||
|       h.setState({ | ||||
|         currentItemBackgroundColor: color, | ||||
|         currentItemFillStyle: "hachure", | ||||
|       }); | ||||
|       const hachureFillButton = queryByTestId(document.body, `fill-hachure`); | ||||
|  | ||||
|       expect(hachureFillButton).toHaveClass("active"); | ||||
|       API.setAppState({ | ||||
|       h.setState({ | ||||
|         currentItemFillStyle: "solid", | ||||
|       }); | ||||
|       const solidFillStyle = queryByTestId(document.body, `fill-solid`); | ||||
| @@ -56,7 +57,7 @@ describe("element locking", () => { | ||||
|     it("should not show fill style when background transparent", () => { | ||||
|       UI.clickTool("rectangle"); | ||||
|  | ||||
|       API.setAppState({ | ||||
|       h.setState({ | ||||
|         currentItemBackgroundColor: COLOR_PALETTE.transparent, | ||||
|         currentItemFillStyle: "hachure", | ||||
|       }); | ||||
| @@ -68,7 +69,7 @@ describe("element locking", () => { | ||||
|     it("should show horizontal text align for text tool", () => { | ||||
|       UI.clickTool("text"); | ||||
|  | ||||
|       API.setAppState({ | ||||
|       h.setState({ | ||||
|         currentItemTextAlign: "right", | ||||
|       }); | ||||
|  | ||||
| @@ -84,7 +85,7 @@ describe("element locking", () => { | ||||
|         backgroundColor: "red", | ||||
|         fillStyle: "cross-hatch", | ||||
|       }); | ||||
|       API.setElements([rect]); | ||||
|       h.elements = [rect]; | ||||
|       API.setSelectedElements([rect]); | ||||
|  | ||||
|       const crossHatchButton = queryByTestId(document.body, `fill-cross-hatch`); | ||||
| @@ -97,7 +98,7 @@ describe("element locking", () => { | ||||
|         backgroundColor: COLOR_PALETTE.transparent, | ||||
|         fillStyle: "cross-hatch", | ||||
|       }); | ||||
|       API.setElements([rect]); | ||||
|       h.elements = [rect]; | ||||
|       API.setSelectedElements([rect]); | ||||
|  | ||||
|       const crossHatchButton = queryByTestId(document.body, `fill-cross-hatch`); | ||||
| @@ -113,7 +114,7 @@ describe("element locking", () => { | ||||
|         type: "rectangle", | ||||
|         strokeWidth: STROKE_WIDTH.thin, | ||||
|       }); | ||||
|       API.setElements([rect1, rect2]); | ||||
|       h.elements = [rect1, rect2]; | ||||
|       API.setSelectedElements([rect1, rect2]); | ||||
|  | ||||
|       const thinStrokeWidthButton = queryByTestId( | ||||
| @@ -132,7 +133,7 @@ describe("element locking", () => { | ||||
|         type: "rectangle", | ||||
|         strokeWidth: STROKE_WIDTH.bold, | ||||
|       }); | ||||
|       API.setElements([rect1, rect2]); | ||||
|       h.elements = [rect1, rect2]; | ||||
|       API.setSelectedElements([rect1, rect2]); | ||||
|  | ||||
|       expect(queryByTestId(document.body, `strokeWidth-thin`)).not.toBe(null); | ||||
| @@ -154,15 +155,13 @@ describe("element locking", () => { | ||||
|       }); | ||||
|       const text = API.createElement({ | ||||
|         type: "text", | ||||
|         fontFamily: FONT_FAMILY["Comic Shanns"], | ||||
|         fontFamily: FONT_FAMILY.Cascadia, | ||||
|       }); | ||||
|       API.setElements([rect, text]); | ||||
|       h.elements = [rect, text]; | ||||
|       API.setSelectedElements([rect, text]); | ||||
|  | ||||
|       expect(queryByTestId(document.body, `strokeWidth-bold`)).toBeChecked(); | ||||
|       expect(queryByTestId(document.body, `font-family-code`)).toHaveClass( | ||||
|         "active", | ||||
|       ); | ||||
|       expect(queryByTestId(document.body, `font-family-code`)).toBeChecked(); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -1,6 +1,4 @@ | ||||
| import { useEffect, useMemo, useRef, useState } from "react"; | ||||
| import type { AppClassProperties, AppState, Point, Primitive } from "../types"; | ||||
| import type { StoreActionType } from "../store"; | ||||
| import { AppClassProperties, AppState, Primitive } from "../types"; | ||||
| import { | ||||
|   DEFAULT_ELEMENT_BACKGROUND_COLOR_PALETTE, | ||||
|   DEFAULT_ELEMENT_BACKGROUND_PICKS, | ||||
| @@ -11,7 +9,6 @@ import { trackEvent } from "../analytics"; | ||||
| import { ButtonIconSelect } from "../components/ButtonIconSelect"; | ||||
| import { ColorPicker } from "../components/ColorPicker/ColorPicker"; | ||||
| import { IconPicker } from "../components/IconPicker"; | ||||
| import { FontPicker } from "../components/FontPicker/FontPicker"; | ||||
| // TODO barnabasmolnar/editor-redesign | ||||
| // TextAlignTopIcon, TextAlignBottomIcon,TextAlignMiddleIcon, | ||||
| // ArrowHead icons | ||||
| @@ -41,6 +38,9 @@ import { | ||||
|   FontSizeExtraLargeIcon, | ||||
|   EdgeSharpIcon, | ||||
|   EdgeRoundIcon, | ||||
|   FreedrawIcon, | ||||
|   FontFamilyNormalIcon, | ||||
|   FontFamilyCodeIcon, | ||||
|   TextAlignLeftIcon, | ||||
|   TextAlignCenterIcon, | ||||
|   TextAlignRightIcon, | ||||
| @@ -50,12 +50,8 @@ import { | ||||
|   ArrowheadDiamondIcon, | ||||
|   ArrowheadDiamondOutlineIcon, | ||||
|   fontSizeIcon, | ||||
|   sharpArrowIcon, | ||||
|   roundArrowIcon, | ||||
|   elbowArrowIcon, | ||||
| } from "../components/icons"; | ||||
| import { | ||||
|   ARROW_TYPE, | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_FONT_SIZE, | ||||
|   FONT_FAMILY, | ||||
| @@ -69,17 +65,17 @@ import { | ||||
|   redrawTextBoundingBox, | ||||
| } from "../element"; | ||||
| import { mutateElement, newElementWith } from "../element/mutateElement"; | ||||
| import { getBoundTextElement } from "../element/textElement"; | ||||
| import { | ||||
|   isArrowElement, | ||||
|   getBoundTextElement, | ||||
|   getDefaultLineHeight, | ||||
| } from "../element/textElement"; | ||||
| import { | ||||
|   isBoundToContainer, | ||||
|   isElbowArrow, | ||||
|   isLinearElement, | ||||
|   isUsingAdaptiveRadius, | ||||
| } from "../element/typeChecks"; | ||||
| import type { | ||||
| import { | ||||
|   Arrowhead, | ||||
|   ExcalidrawBindableElement, | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawLinearElement, | ||||
|   ExcalidrawTextElement, | ||||
| @@ -98,23 +94,9 @@ import { | ||||
|   isSomeElementSelected, | ||||
| } from "../scene"; | ||||
| import { hasStrokeColor } from "../scene/comparisons"; | ||||
| import { | ||||
|   arrayToMap, | ||||
|   getFontFamilyString, | ||||
|   getShortcutKey, | ||||
|   tupleToCoors, | ||||
| } from "../utils"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
| import { StoreAction } from "../store"; | ||||
| import { Fonts, getLineHeight } from "../fonts"; | ||||
| import { | ||||
|   bindLinearElement, | ||||
|   bindPointToSnapToElementOutline, | ||||
|   calculateFixedPointForElbowArrowBinding, | ||||
|   getHoveredElementForBinding, | ||||
| } from "../element/binding"; | ||||
| import { mutateElbowArrow } from "../element/routing"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
|  | ||||
| const FONT_SIZE_RELATIVE_INCREASE_STEP = 0.1; | ||||
|  | ||||
| @@ -185,7 +167,7 @@ const offsetElementAfterFontResize = ( | ||||
|   prevElement: ExcalidrawTextElement, | ||||
|   nextElement: ExcalidrawTextElement, | ||||
| ) => { | ||||
|   if (isBoundToContainer(nextElement) || !nextElement.autoResize) { | ||||
|   if (isBoundToContainer(nextElement)) { | ||||
|     return nextElement; | ||||
|   } | ||||
|   return mutateElement( | ||||
| @@ -747,388 +729,104 @@ export const actionIncreaseFontSize = register({ | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| type ChangeFontFamilyData = Partial< | ||||
|   Pick< | ||||
|     AppState, | ||||
|     "openPopup" | "currentItemFontFamily" | "currentHoveredFontFamily" | ||||
|   > | ||||
| > & { | ||||
|   /** cache of selected & editing elements populated on opened popup */ | ||||
|   cachedElements?: Map<string, ExcalidrawElement>; | ||||
|   /** flag to reset all elements to their cached versions  */ | ||||
|   resetAll?: true; | ||||
|   /** flag to reset all containers to their cached versions */ | ||||
|   resetContainers?: true; | ||||
| }; | ||||
|  | ||||
| export const actionChangeFontFamily = register({ | ||||
|   name: "changeFontFamily", | ||||
|   label: "labels.fontFamily", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value, app) => { | ||||
|     const { cachedElements, resetAll, resetContainers, ...nextAppState } = | ||||
|       value as ChangeFontFamilyData; | ||||
|  | ||||
|     if (resetAll) { | ||||
|       const nextElements = changeProperty( | ||||
|     return { | ||||
|       elements: changeProperty( | ||||
|         elements, | ||||
|         appState, | ||||
|         (element) => { | ||||
|           const cachedElement = cachedElements?.get(element.id); | ||||
|           if (cachedElement) { | ||||
|             const newElement = newElementWith(element, { | ||||
|               ...cachedElement, | ||||
|             }); | ||||
|  | ||||
|         (oldElement) => { | ||||
|           if (isTextElement(oldElement)) { | ||||
|             const newElement: ExcalidrawTextElement = newElementWith( | ||||
|               oldElement, | ||||
|               { | ||||
|                 fontFamily: value, | ||||
|                 lineHeight: getDefaultLineHeight(value), | ||||
|               }, | ||||
|             ); | ||||
|             redrawTextBoundingBox( | ||||
|               newElement, | ||||
|               app.scene.getContainerElement(oldElement), | ||||
|               app.scene.getNonDeletedElementsMap(), | ||||
|             ); | ||||
|             return newElement; | ||||
|           } | ||||
|  | ||||
|           return element; | ||||
|           return oldElement; | ||||
|         }, | ||||
|         true, | ||||
|       ); | ||||
|  | ||||
|       return { | ||||
|         elements: nextElements, | ||||
|         appState: { | ||||
|           ...appState, | ||||
|           ...nextAppState, | ||||
|         }, | ||||
|         storeAction: StoreAction.UPDATE, | ||||
|       }; | ||||
|     } | ||||
|  | ||||
|     const { currentItemFontFamily, currentHoveredFontFamily } = value; | ||||
|  | ||||
|     let nexStoreAction: StoreActionType = StoreAction.NONE; | ||||
|     let nextFontFamily: FontFamilyValues | undefined; | ||||
|     let skipOnHoverRender = false; | ||||
|  | ||||
|     if (currentItemFontFamily) { | ||||
|       nextFontFamily = currentItemFontFamily; | ||||
|       nexStoreAction = StoreAction.CAPTURE; | ||||
|     } else if (currentHoveredFontFamily) { | ||||
|       nextFontFamily = currentHoveredFontFamily; | ||||
|       nexStoreAction = StoreAction.NONE; | ||||
|  | ||||
|       const selectedTextElements = getSelectedElements(elements, appState, { | ||||
|         includeBoundTextElement: true, | ||||
|       }).filter((element) => isTextElement(element)); | ||||
|  | ||||
|       // skip on hover re-render for more than 200 text elements or for text element with more than 5000 chars combined | ||||
|       if (selectedTextElements.length > 200) { | ||||
|         skipOnHoverRender = true; | ||||
|       } else { | ||||
|         let i = 0; | ||||
|         let textLengthAccumulator = 0; | ||||
|  | ||||
|         while ( | ||||
|           i < selectedTextElements.length && | ||||
|           textLengthAccumulator < 5000 | ||||
|         ) { | ||||
|           const textElement = selectedTextElements[i] as ExcalidrawTextElement; | ||||
|           textLengthAccumulator += textElement?.originalText.length || 0; | ||||
|           i++; | ||||
|         } | ||||
|  | ||||
|         if (textLengthAccumulator > 5000) { | ||||
|           skipOnHoverRender = true; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     const result = { | ||||
|       ), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         ...nextAppState, | ||||
|         currentItemFontFamily: value, | ||||
|       }, | ||||
|       storeAction: nexStoreAction, | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|     }; | ||||
|  | ||||
|     if (nextFontFamily && !skipOnHoverRender) { | ||||
|       const elementContainerMapping = new Map< | ||||
|         ExcalidrawTextElement, | ||||
|         ExcalidrawElement | null | ||||
|       >(); | ||||
|       let uniqueChars = new Set<string>(); | ||||
|       let skipFontFaceCheck = false; | ||||
|  | ||||
|       const fontsCache = Array.from(Fonts.loadedFontsCache.values()); | ||||
|       const fontFamily = Object.entries(FONT_FAMILY).find( | ||||
|         ([_, value]) => value === nextFontFamily, | ||||
|       )?.[0]; | ||||
|  | ||||
|       // skip `document.font.check` check on hover, if at least one font family has loaded as it's super slow (could result in slightly different bbox, which is fine) | ||||
|       if ( | ||||
|         currentHoveredFontFamily && | ||||
|         fontFamily && | ||||
|         fontsCache.some((sig) => sig.startsWith(fontFamily)) | ||||
|       ) { | ||||
|         skipFontFaceCheck = true; | ||||
|       } | ||||
|  | ||||
|       // following causes re-render so make sure we changed the family | ||||
|       // otherwise it could cause unexpected issues, such as preventing opening the popover when in wysiwyg | ||||
|       Object.assign(result, { | ||||
|         elements: changeProperty( | ||||
|           elements, | ||||
|           appState, | ||||
|           (oldElement) => { | ||||
|             if ( | ||||
|               isTextElement(oldElement) && | ||||
|               (oldElement.fontFamily !== nextFontFamily || | ||||
|                 currentItemFontFamily) // force update on selection | ||||
|             ) { | ||||
|               const newElement: ExcalidrawTextElement = newElementWith( | ||||
|                 oldElement, | ||||
|                 { | ||||
|                   fontFamily: nextFontFamily, | ||||
|                   lineHeight: getLineHeight(nextFontFamily!), | ||||
|                 }, | ||||
|               ); | ||||
|  | ||||
|               const cachedContainer = | ||||
|                 cachedElements?.get(oldElement.containerId || "") || {}; | ||||
|  | ||||
|               const container = app.scene.getContainerElement(oldElement); | ||||
|  | ||||
|               if (resetContainers && container && cachedContainer) { | ||||
|                 // reset the container back to it's cached version | ||||
|                 mutateElement(container, { ...cachedContainer }, false); | ||||
|               } | ||||
|  | ||||
|               if (!skipFontFaceCheck) { | ||||
|                 uniqueChars = new Set([ | ||||
|                   ...uniqueChars, | ||||
|                   ...Array.from(newElement.originalText), | ||||
|                 ]); | ||||
|               } | ||||
|  | ||||
|               elementContainerMapping.set(newElement, container); | ||||
|  | ||||
|               return newElement; | ||||
|             } | ||||
|  | ||||
|             return oldElement; | ||||
|           }, | ||||
|           true, | ||||
|         ), | ||||
|       }); | ||||
|  | ||||
|       // size is irrelevant, but necessary | ||||
|       const fontString = `10px ${getFontFamilyString({ | ||||
|         fontFamily: nextFontFamily, | ||||
|       })}`; | ||||
|       const chars = Array.from(uniqueChars.values()).join(); | ||||
|  | ||||
|       if (skipFontFaceCheck || window.document.fonts.check(fontString, chars)) { | ||||
|         // we either skip the check (have at least one font face loaded) or do the check and find out all the font faces have loaded | ||||
|         for (const [element, container] of elementContainerMapping) { | ||||
|           // trigger synchronous redraw | ||||
|           redrawTextBoundingBox( | ||||
|             element, | ||||
|             container, | ||||
|             app.scene.getNonDeletedElementsMap(), | ||||
|             false, | ||||
|           ); | ||||
|         } | ||||
|       } else { | ||||
|         // otherwise try to load all font faces for the given chars and redraw elements once our font faces loaded | ||||
|         window.document.fonts.load(fontString, chars).then((fontFaces) => { | ||||
|           for (const [element, container] of elementContainerMapping) { | ||||
|             // use latest element state to ensure we don't have closure over an old instance in order to avoid possible race conditions (i.e. font faces load out-of-order while rapidly switching fonts) | ||||
|             const latestElement = app.scene.getElement(element.id); | ||||
|             const latestContainer = container | ||||
|               ? app.scene.getElement(container.id) | ||||
|               : null; | ||||
|  | ||||
|             if (latestElement) { | ||||
|               // trigger async redraw | ||||
|               redrawTextBoundingBox( | ||||
|                 latestElement as ExcalidrawTextElement, | ||||
|                 latestContainer, | ||||
|                 app.scene.getNonDeletedElementsMap(), | ||||
|                 false, | ||||
|               ); | ||||
|             } | ||||
|           } | ||||
|  | ||||
|           // trigger update once we've mutated all the elements, which also updates our cache | ||||
|           app.fonts.onLoaded(fontFaces); | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, app, updateData }) => { | ||||
|     const cachedElementsRef = useRef<Map<string, ExcalidrawElement>>(new Map()); | ||||
|     const prevSelectedFontFamilyRef = useRef<number | null>(null); | ||||
|     // relying on state batching as multiple `FontPicker` handlers could be called in rapid succession and we want to combine them | ||||
|     const [batchedData, setBatchedData] = useState<ChangeFontFamilyData>({}); | ||||
|     const isUnmounted = useRef(true); | ||||
|  | ||||
|     const selectedFontFamily = useMemo(() => { | ||||
|       const getFontFamily = ( | ||||
|         elementsArray: readonly ExcalidrawElement[], | ||||
|         elementsMap: Map<string, ExcalidrawElement>, | ||||
|       ) => | ||||
|         getFormValue( | ||||
|           elementsArray, | ||||
|           appState, | ||||
|           (element) => { | ||||
|             if (isTextElement(element)) { | ||||
|               return element.fontFamily; | ||||
|             } | ||||
|             const boundTextElement = getBoundTextElement(element, elementsMap); | ||||
|             if (boundTextElement) { | ||||
|               return boundTextElement.fontFamily; | ||||
|             } | ||||
|             return null; | ||||
|           }, | ||||
|           (element) => | ||||
|             isTextElement(element) || | ||||
|             getBoundTextElement(element, elementsMap) !== null, | ||||
|           (hasSelection) => | ||||
|             hasSelection | ||||
|               ? null | ||||
|               : appState.currentItemFontFamily || DEFAULT_FONT_FAMILY, | ||||
|         ); | ||||
|  | ||||
|       // popup opened, use cached elements | ||||
|       if ( | ||||
|         batchedData.openPopup === "fontFamily" && | ||||
|         appState.openPopup === "fontFamily" | ||||
|       ) { | ||||
|         return getFontFamily( | ||||
|           Array.from(cachedElementsRef.current?.values() ?? []), | ||||
|           cachedElementsRef.current, | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       // popup closed, use all elements | ||||
|       if (!batchedData.openPopup && appState.openPopup !== "fontFamily") { | ||||
|         return getFontFamily(elements, app.scene.getNonDeletedElementsMap()); | ||||
|       } | ||||
|  | ||||
|       // popup props are not in sync, hence we are in the middle of an update, so keeping the previous value we've had | ||||
|       return prevSelectedFontFamilyRef.current; | ||||
|     }, [batchedData.openPopup, appState, elements, app.scene]); | ||||
|  | ||||
|     useEffect(() => { | ||||
|       prevSelectedFontFamilyRef.current = selectedFontFamily; | ||||
|     }, [selectedFontFamily]); | ||||
|  | ||||
|     useEffect(() => { | ||||
|       if (Object.keys(batchedData).length) { | ||||
|         updateData(batchedData); | ||||
|         // reset the data after we've used the data | ||||
|         setBatchedData({}); | ||||
|       } | ||||
|       // call update only on internal state changes | ||||
|       // eslint-disable-next-line react-hooks/exhaustive-deps | ||||
|     }, [batchedData]); | ||||
|  | ||||
|     useEffect(() => { | ||||
|       isUnmounted.current = false; | ||||
|  | ||||
|       return () => { | ||||
|         isUnmounted.current = true; | ||||
|       }; | ||||
|     }, []); | ||||
|   PanelComponent: ({ elements, appState, updateData, app }) => { | ||||
|     const options: { | ||||
|       value: FontFamilyValues; | ||||
|       text: string; | ||||
|       icon: JSX.Element; | ||||
|       testId: string; | ||||
|     }[] = [ | ||||
|       { | ||||
|         value: FONT_FAMILY.Virgil, | ||||
|         text: t("labels.handDrawn"), | ||||
|         icon: FreedrawIcon, | ||||
|         testId: "font-family-virgil", | ||||
|       }, | ||||
|       { | ||||
|         value: FONT_FAMILY.Helvetica, | ||||
|         text: t("labels.normal"), | ||||
|         icon: FontFamilyNormalIcon, | ||||
|         testId: "font-family-normal", | ||||
|       }, | ||||
|       { | ||||
|         value: FONT_FAMILY.Cascadia, | ||||
|         text: t("labels.code"), | ||||
|         icon: FontFamilyCodeIcon, | ||||
|         testId: "font-family-code", | ||||
|       }, | ||||
|     ]; | ||||
|  | ||||
|     return ( | ||||
|       <fieldset> | ||||
|         <legend>{t("labels.fontFamily")}</legend> | ||||
|         <FontPicker | ||||
|           isOpened={appState.openPopup === "fontFamily"} | ||||
|           selectedFontFamily={selectedFontFamily} | ||||
|           hoveredFontFamily={appState.currentHoveredFontFamily} | ||||
|           onSelect={(fontFamily) => { | ||||
|             setBatchedData({ | ||||
|               openPopup: null, | ||||
|               currentHoveredFontFamily: null, | ||||
|               currentItemFontFamily: fontFamily, | ||||
|             }); | ||||
|  | ||||
|             // defensive clear so immediate close won't abuse the cached elements | ||||
|             cachedElementsRef.current.clear(); | ||||
|           }} | ||||
|           onHover={(fontFamily) => { | ||||
|             setBatchedData({ | ||||
|               currentHoveredFontFamily: fontFamily, | ||||
|               cachedElements: new Map(cachedElementsRef.current), | ||||
|               resetContainers: true, | ||||
|             }); | ||||
|           }} | ||||
|           onLeave={() => { | ||||
|             setBatchedData({ | ||||
|               currentHoveredFontFamily: null, | ||||
|               cachedElements: new Map(cachedElementsRef.current), | ||||
|               resetAll: true, | ||||
|             }); | ||||
|           }} | ||||
|           onPopupChange={(open) => { | ||||
|             if (open) { | ||||
|               // open, populate the cache from scratch | ||||
|               cachedElementsRef.current.clear(); | ||||
|  | ||||
|               const { editingElement } = appState; | ||||
|  | ||||
|               if (editingElement?.type === "text") { | ||||
|                 // retrieve the latest version from the scene, as `editingElement` isn't mutated | ||||
|                 const latestEditingElement = app.scene.getElement( | ||||
|                   editingElement.id, | ||||
|                 ); | ||||
|  | ||||
|                 // inside the wysiwyg editor | ||||
|                 cachedElementsRef.current.set( | ||||
|                   editingElement.id, | ||||
|                   newElementWith( | ||||
|                     latestEditingElement || editingElement, | ||||
|                     {}, | ||||
|                     true, | ||||
|                   ), | ||||
|                 ); | ||||
|               } else { | ||||
|                 const selectedElements = getSelectedElements( | ||||
|                   elements, | ||||
|                   appState, | ||||
|                   { | ||||
|                     includeBoundTextElement: true, | ||||
|                   }, | ||||
|                 ); | ||||
|  | ||||
|                 for (const element of selectedElements) { | ||||
|                   cachedElementsRef.current.set( | ||||
|                     element.id, | ||||
|                     newElementWith(element, {}, true), | ||||
|                   ); | ||||
|                 } | ||||
|         <ButtonIconSelect<FontFamilyValues | false> | ||||
|           group="font-family" | ||||
|           options={options} | ||||
|           value={getFormValue( | ||||
|             elements, | ||||
|             appState, | ||||
|             (element) => { | ||||
|               if (isTextElement(element)) { | ||||
|                 return element.fontFamily; | ||||
|               } | ||||
|  | ||||
|               setBatchedData({ | ||||
|                 openPopup: "fontFamily", | ||||
|               }); | ||||
|             } else { | ||||
|               // close, use the cache and clear it afterwards | ||||
|               const data = { | ||||
|                 openPopup: null, | ||||
|                 currentHoveredFontFamily: null, | ||||
|                 cachedElements: new Map(cachedElementsRef.current), | ||||
|                 resetAll: true, | ||||
|               } as ChangeFontFamilyData; | ||||
|  | ||||
|               if (isUnmounted.current) { | ||||
|                 // in case the component was unmounted by the parent, trigger the update directly | ||||
|                 updateData({ ...batchedData, ...data }); | ||||
|               } else { | ||||
|                 setBatchedData(data); | ||||
|               const boundTextElement = getBoundTextElement( | ||||
|                 element, | ||||
|                 app.scene.getNonDeletedElementsMap(), | ||||
|               ); | ||||
|               if (boundTextElement) { | ||||
|                 return boundTextElement.fontFamily; | ||||
|               } | ||||
|  | ||||
|               cachedElementsRef.current.clear(); | ||||
|             } | ||||
|           }} | ||||
|               return null; | ||||
|             }, | ||||
|             (element) => | ||||
|               isTextElement(element) || | ||||
|               getBoundTextElement( | ||||
|                 element, | ||||
|                 app.scene.getNonDeletedElementsMap(), | ||||
|               ) !== null, | ||||
|             (hasSelection) => | ||||
|               hasSelection | ||||
|                 ? null | ||||
|                 : appState.currentItemFontFamily || DEFAULT_FONT_FAMILY, | ||||
|           )} | ||||
|           onChange={(value) => updateData(value)} | ||||
|         /> | ||||
|       </fieldset> | ||||
|     ); | ||||
| @@ -1321,12 +1019,8 @@ export const actionChangeRoundness = register({ | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => { | ||||
|         if (isElbowArrow(el)) { | ||||
|           return el; | ||||
|         } | ||||
|  | ||||
|         return newElementWith(el, { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
|         newElementWith(el, { | ||||
|           roundness: | ||||
|             value === "round" | ||||
|               ? { | ||||
| @@ -1335,8 +1029,8 @@ export const actionChangeRoundness = register({ | ||||
|                     : ROUNDNESS.PROPORTIONAL_RADIUS, | ||||
|                 } | ||||
|               : null, | ||||
|         }); | ||||
|       }), | ||||
|         }), | ||||
|       ), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         currentItemRoundness: value, | ||||
| @@ -1376,8 +1070,7 @@ export const actionChangeRoundness = register({ | ||||
|             appState, | ||||
|             (element) => | ||||
|               hasLegacyRoundness ? null : element.roundness ? "round" : "sharp", | ||||
|             (element) => | ||||
|               !isArrowElement(element) && element.hasOwnProperty("roundness"), | ||||
|             (element) => element.hasOwnProperty("roundness"), | ||||
|             (hasSelection) => | ||||
|               hasSelection ? null : appState.currentItemRoundness, | ||||
|           )} | ||||
| @@ -1540,219 +1233,3 @@ export const actionChangeArrowhead = register({ | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| export const actionChangeArrowType = register({ | ||||
|   name: "changeArrowType", | ||||
|   label: "Change arrow types", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value, app) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => { | ||||
|         if (!isArrowElement(el)) { | ||||
|           return el; | ||||
|         } | ||||
|         const newElement = newElementWith(el, { | ||||
|           roundness: | ||||
|             value === ARROW_TYPE.round | ||||
|               ? { | ||||
|                   type: ROUNDNESS.PROPORTIONAL_RADIUS, | ||||
|                 } | ||||
|               : null, | ||||
|           elbowed: value === ARROW_TYPE.elbow, | ||||
|           points: | ||||
|             value === ARROW_TYPE.elbow || el.elbowed | ||||
|               ? [el.points[0], el.points[el.points.length - 1]] | ||||
|               : el.points, | ||||
|         }); | ||||
|  | ||||
|         if (isElbowArrow(newElement)) { | ||||
|           const elementsMap = app.scene.getNonDeletedElementsMap(); | ||||
|  | ||||
|           app.dismissLinearEditor(); | ||||
|  | ||||
|           const startGlobalPoint = | ||||
|             LinearElementEditor.getPointAtIndexGlobalCoordinates( | ||||
|               newElement, | ||||
|               0, | ||||
|               elementsMap, | ||||
|             ); | ||||
|           const endGlobalPoint = | ||||
|             LinearElementEditor.getPointAtIndexGlobalCoordinates( | ||||
|               newElement, | ||||
|               -1, | ||||
|               elementsMap, | ||||
|             ); | ||||
|           const startHoveredElement = | ||||
|             !newElement.startBinding && | ||||
|             getHoveredElementForBinding( | ||||
|               tupleToCoors(startGlobalPoint), | ||||
|               elements, | ||||
|               elementsMap, | ||||
|               true, | ||||
|             ); | ||||
|           const endHoveredElement = | ||||
|             !newElement.endBinding && | ||||
|             getHoveredElementForBinding( | ||||
|               tupleToCoors(endGlobalPoint), | ||||
|               elements, | ||||
|               elementsMap, | ||||
|               true, | ||||
|             ); | ||||
|           const startElement = startHoveredElement | ||||
|             ? startHoveredElement | ||||
|             : newElement.startBinding && | ||||
|               (elementsMap.get( | ||||
|                 newElement.startBinding.elementId, | ||||
|               ) as ExcalidrawBindableElement); | ||||
|           const endElement = endHoveredElement | ||||
|             ? endHoveredElement | ||||
|             : newElement.endBinding && | ||||
|               (elementsMap.get( | ||||
|                 newElement.endBinding.elementId, | ||||
|               ) as ExcalidrawBindableElement); | ||||
|  | ||||
|           const finalStartPoint = startHoveredElement | ||||
|             ? bindPointToSnapToElementOutline( | ||||
|                 startGlobalPoint, | ||||
|                 endGlobalPoint, | ||||
|                 startHoveredElement, | ||||
|                 elementsMap, | ||||
|               ) | ||||
|             : startGlobalPoint; | ||||
|           const finalEndPoint = endHoveredElement | ||||
|             ? bindPointToSnapToElementOutline( | ||||
|                 endGlobalPoint, | ||||
|                 startGlobalPoint, | ||||
|                 endHoveredElement, | ||||
|                 elementsMap, | ||||
|               ) | ||||
|             : endGlobalPoint; | ||||
|  | ||||
|           startHoveredElement && | ||||
|             bindLinearElement( | ||||
|               newElement, | ||||
|               startHoveredElement, | ||||
|               "start", | ||||
|               elementsMap, | ||||
|             ); | ||||
|           endHoveredElement && | ||||
|             bindLinearElement( | ||||
|               newElement, | ||||
|               endHoveredElement, | ||||
|               "end", | ||||
|               elementsMap, | ||||
|             ); | ||||
|  | ||||
|           mutateElbowArrow( | ||||
|             newElement, | ||||
|             elementsMap, | ||||
|             [finalStartPoint, finalEndPoint].map( | ||||
|               (point) => | ||||
|                 [point[0] - newElement.x, point[1] - newElement.y] as Point, | ||||
|             ), | ||||
|             [0, 0], | ||||
|             { | ||||
|               ...(startElement && newElement.startBinding | ||||
|                 ? { | ||||
|                     startBinding: { | ||||
|                       // @ts-ignore TS cannot discern check above | ||||
|                       ...newElement.startBinding!, | ||||
|                       ...calculateFixedPointForElbowArrowBinding( | ||||
|                         newElement, | ||||
|                         startElement, | ||||
|                         "start", | ||||
|                         elementsMap, | ||||
|                       ), | ||||
|                     }, | ||||
|                   } | ||||
|                 : {}), | ||||
|               ...(endElement && newElement.endBinding | ||||
|                 ? { | ||||
|                     endBinding: { | ||||
|                       // @ts-ignore TS cannot discern check above | ||||
|                       ...newElement.endBinding, | ||||
|                       ...calculateFixedPointForElbowArrowBinding( | ||||
|                         newElement, | ||||
|                         endElement, | ||||
|                         "end", | ||||
|                         elementsMap, | ||||
|                       ), | ||||
|                     }, | ||||
|                   } | ||||
|                 : {}), | ||||
|             }, | ||||
|           ); | ||||
|         } else { | ||||
|           mutateElement( | ||||
|             newElement, | ||||
|             { | ||||
|               startBinding: newElement.startBinding | ||||
|                 ? { ...newElement.startBinding, fixedPoint: null } | ||||
|                 : null, | ||||
|               endBinding: newElement.endBinding | ||||
|                 ? { ...newElement.endBinding, fixedPoint: null } | ||||
|                 : null, | ||||
|             }, | ||||
|             false, | ||||
|           ); | ||||
|         } | ||||
|  | ||||
|         return newElement; | ||||
|       }), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         currentItemArrowType: value, | ||||
|       }, | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => { | ||||
|     return ( | ||||
|       <fieldset> | ||||
|         <legend>{t("labels.arrowtypes")}</legend> | ||||
|         <ButtonIconSelect | ||||
|           group="arrowtypes" | ||||
|           options={[ | ||||
|             { | ||||
|               value: ARROW_TYPE.sharp, | ||||
|               text: t("labels.arrowtype_sharp"), | ||||
|               icon: sharpArrowIcon, | ||||
|               testId: "sharp-arrow", | ||||
|             }, | ||||
|             { | ||||
|               value: ARROW_TYPE.round, | ||||
|               text: t("labels.arrowtype_round"), | ||||
|               icon: roundArrowIcon, | ||||
|               testId: "round-arrow", | ||||
|             }, | ||||
|             { | ||||
|               value: ARROW_TYPE.elbow, | ||||
|               text: t("labels.arrowtype_elbowed"), | ||||
|               icon: elbowArrowIcon, | ||||
|               testId: "elbow-arrow", | ||||
|             }, | ||||
|           ]} | ||||
|           value={getFormValue( | ||||
|             elements, | ||||
|             appState, | ||||
|             (element) => { | ||||
|               if (isArrowElement(element)) { | ||||
|                 return element.elbowed | ||||
|                   ? ARROW_TYPE.elbow | ||||
|                   : element.roundness | ||||
|                   ? ARROW_TYPE.round | ||||
|                   : ARROW_TYPE.sharp; | ||||
|               } | ||||
|  | ||||
|               return null; | ||||
|             }, | ||||
|             (element) => isArrowElement(element), | ||||
|             (hasSelection) => | ||||
|               hasSelection ? null : appState.currentItemArrowType, | ||||
|           )} | ||||
|           onChange={(value) => updateData(value)} | ||||
|         /> | ||||
|       </fieldset> | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { selectGroupsForSelectedElements } from "../groups"; | ||||
| import { getNonDeletedElements, isTextElement } from "../element"; | ||||
| import type { ExcalidrawElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { isLinearElement } from "../element/typeChecks"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
| import { excludeElementsInFramesFromSelection } from "../scene/selection"; | ||||
|   | ||||
| @@ -12,7 +12,10 @@ import { | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_TEXT_ALIGN, | ||||
| } from "../constants"; | ||||
| import { getBoundTextElement } from "../element/textElement"; | ||||
| import { | ||||
|   getBoundTextElement, | ||||
|   getDefaultLineHeight, | ||||
| } from "../element/textElement"; | ||||
| import { | ||||
|   hasBoundTextElement, | ||||
|   canApplyRoundnessTypeToElement, | ||||
| @@ -21,10 +24,9 @@ import { | ||||
|   isArrowElement, | ||||
| } from "../element/typeChecks"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import type { ExcalidrawTextElement } from "../element/types"; | ||||
| import { ExcalidrawTextElement } from "../element/types"; | ||||
| import { paintIcon } from "../components/icons"; | ||||
| import { StoreAction } from "../store"; | ||||
| import { getLineHeight } from "../fonts"; | ||||
|  | ||||
| // `copiedStyles` is exported only for tests. | ||||
| export let copiedStyles: string = "{}"; | ||||
| @@ -120,7 +122,7 @@ export const actionPasteStyles = register({ | ||||
|                 DEFAULT_TEXT_ALIGN, | ||||
|               lineHeight: | ||||
|                 (elementStylesToCopyFrom as ExcalidrawTextElement).lineHeight || | ||||
|                 getLineHeight(fontFamily), | ||||
|                 getDefaultLineHeight(fontFamily), | ||||
|             }); | ||||
|             let container = null; | ||||
|             if (newElement.containerId) { | ||||
|   | ||||
| @@ -1,48 +0,0 @@ | ||||
| import { isTextElement } from "../element"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { measureText } from "../element/textElement"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { StoreAction } from "../store"; | ||||
| import type { AppClassProperties } from "../types"; | ||||
| import { getFontString } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
| export const actionTextAutoResize = register({ | ||||
|   name: "autoResize", | ||||
|   label: "labels.autoResize", | ||||
|   icon: null, | ||||
|   trackEvent: { category: "element" }, | ||||
|   predicate: (elements, appState, _: unknown, app: AppClassProperties) => { | ||||
|     const selectedElements = getSelectedElements(elements, appState); | ||||
|     return ( | ||||
|       selectedElements.length === 1 && | ||||
|       isTextElement(selectedElements[0]) && | ||||
|       !selectedElements[0].autoResize | ||||
|     ); | ||||
|   }, | ||||
|   perform: (elements, appState, _, app) => { | ||||
|     const selectedElements = getSelectedElements(elements, appState); | ||||
|  | ||||
|     return { | ||||
|       appState, | ||||
|       elements: elements.map((element) => { | ||||
|         if (element.id === selectedElements[0].id && isTextElement(element)) { | ||||
|           const metrics = measureText( | ||||
|             element.originalText, | ||||
|             getFontString(element), | ||||
|             element.lineHeight, | ||||
|           ); | ||||
|  | ||||
|           return newElementWith(element, { | ||||
|             autoResize: true, | ||||
|             width: metrics.width, | ||||
|             height: metrics.height, | ||||
|             text: element.originalText, | ||||
|           }); | ||||
|         } | ||||
|         return element; | ||||
|       }), | ||||
|       storeAction: StoreAction.CAPTURE, | ||||
|     }; | ||||
|   }, | ||||
| }); | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { GRID_SIZE } from "../constants"; | ||||
| import type { AppState } from "../types"; | ||||
| import { AppState } from "../types"; | ||||
| import { gridIcon } from "../components/icons"; | ||||
| import { StoreAction } from "../store"; | ||||
|  | ||||
|   | ||||
| @@ -5,22 +5,21 @@ import { StoreAction } from "../store"; | ||||
|  | ||||
| export const actionToggleStats = register({ | ||||
|   name: "stats", | ||||
|   label: "stats.fullTitle", | ||||
|   label: "stats.title", | ||||
|   icon: abacusIcon, | ||||
|   paletteName: "Toggle stats", | ||||
|   viewMode: true, | ||||
|   trackEvent: { category: "menu" }, | ||||
|   keywords: ["edit", "attributes", "customize"], | ||||
|   perform(elements, appState) { | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         stats: { ...appState.stats, open: !this.checked!(appState) }, | ||||
|         showStats: !this.checked!(appState), | ||||
|       }, | ||||
|       storeAction: StoreAction.NONE, | ||||
|     }; | ||||
|   }, | ||||
|   checked: (appState) => appState.stats.open, | ||||
|   checked: (appState) => appState.showStats, | ||||
|   keyTest: (event) => | ||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.SLASH, | ||||
| }); | ||||
|   | ||||
| @@ -20,7 +20,6 @@ import { StoreAction } from "../store"; | ||||
| export const actionSendBackward = register({ | ||||
|   name: "sendBackward", | ||||
|   label: "labels.sendBackward", | ||||
|   keywords: ["move down", "zindex", "layer"], | ||||
|   icon: SendBackwardIcon, | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
| @@ -50,7 +49,6 @@ export const actionSendBackward = register({ | ||||
| export const actionBringForward = register({ | ||||
|   name: "bringForward", | ||||
|   label: "labels.bringForward", | ||||
|   keywords: ["move up", "zindex", "layer"], | ||||
|   icon: BringForwardIcon, | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
| @@ -80,7 +78,6 @@ export const actionBringForward = register({ | ||||
| export const actionSendToBack = register({ | ||||
|   name: "sendToBack", | ||||
|   label: "labels.sendToBack", | ||||
|   keywords: ["move down", "zindex", "layer"], | ||||
|   icon: SendToBackIcon, | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
| @@ -117,7 +114,6 @@ export const actionSendToBack = register({ | ||||
| export const actionBringToFront = register({ | ||||
|   name: "bringToFront", | ||||
|   label: "labels.bringToFront", | ||||
|   keywords: ["move up", "zindex", "layer"], | ||||
|   icon: BringToFrontIcon, | ||||
|   trackEvent: { category: "element" }, | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from "react"; | ||||
| import type { | ||||
| import { | ||||
|   Action, | ||||
|   UpdaterFn, | ||||
|   ActionName, | ||||
| @@ -7,11 +7,8 @@ import type { | ||||
|   PanelComponentProps, | ||||
|   ActionSource, | ||||
| } from "./types"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../element/types"; | ||||
| import type { AppClassProperties, AppState } from "../types"; | ||||
| import { ExcalidrawElement, OrderedExcalidrawElement } from "../element/types"; | ||||
| import { AppClassProperties, AppState } from "../types"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { isPromiseLike } from "../utils"; | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import type { Action } from "./types"; | ||||
| import { Action } from "./types"; | ||||
|  | ||||
| export let actions: readonly Action[] = []; | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { isDarwin } from "../constants"; | ||||
| import { t } from "../i18n"; | ||||
| import type { SubtypeOf } from "../utility-types"; | ||||
| import { SubtypeOf } from "../utility-types"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import type { ActionName } from "./types"; | ||||
| import { ActionName } from "./types"; | ||||
|  | ||||
| export type ShortcutName = | ||||
|   | SubtypeOf< | ||||
|   | ||||
| @@ -1,17 +1,14 @@ | ||||
| import type React from "react"; | ||||
| import type { | ||||
|   ExcalidrawElement, | ||||
|   OrderedExcalidrawElement, | ||||
| } from "../element/types"; | ||||
| import type { | ||||
| import React from "react"; | ||||
| import { ExcalidrawElement, OrderedExcalidrawElement } from "../element/types"; | ||||
| import { | ||||
|   AppClassProperties, | ||||
|   AppState, | ||||
|   ExcalidrawProps, | ||||
|   BinaryFiles, | ||||
|   UIAppState, | ||||
| } from "../types"; | ||||
| import type { MarkOptional } from "../utility-types"; | ||||
| import type { StoreActionType } from "../store"; | ||||
| import { MarkOptional } from "../utility-types"; | ||||
| import { StoreAction } from "../store"; | ||||
|  | ||||
| export type ActionSource = | ||||
|   | "ui" | ||||
| @@ -29,7 +26,7 @@ export type ActionResult = | ||||
|         "offsetTop" | "offsetLeft" | "width" | "height" | ||||
|       > | null; | ||||
|       files?: BinaryFiles | null; | ||||
|       storeAction: StoreActionType; | ||||
|       storeAction: keyof typeof StoreAction; | ||||
|       replaceFiles?: boolean; | ||||
|     } | ||||
|   | false; | ||||
| @@ -70,7 +67,6 @@ export type ActionName = | ||||
|   | "changeSloppiness" | ||||
|   | "changeStrokeStyle" | ||||
|   | "changeArrowhead" | ||||
|   | "changeArrowType" | ||||
|   | "changeOpacity" | ||||
|   | "changeFontSize" | ||||
|   | "toggleCanvasMenu" | ||||
| @@ -135,9 +131,7 @@ export type ActionName = | ||||
|   | "setEmbeddableAsActiveTool" | ||||
|   | "createContainerFromText" | ||||
|   | "wrapTextInContainer" | ||||
|   | "commandPalette" | ||||
|   | "autoResize" | ||||
|   | "elementStats"; | ||||
|   | "commandPalette"; | ||||
|  | ||||
| export type PanelComponentProps = { | ||||
|   elements: readonly ExcalidrawElement[]; | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| import type { ElementsMap, ExcalidrawElement } from "./element/types"; | ||||
| import { ElementsMap, ExcalidrawElement } from "./element/types"; | ||||
| import { newElementWith } from "./element/mutateElement"; | ||||
| import type { BoundingBox } from "./element/bounds"; | ||||
| import { getCommonBoundingBox } from "./element/bounds"; | ||||
| import { BoundingBox, getCommonBoundingBox } from "./element/bounds"; | ||||
| import { getMaximumGroups } from "./groups"; | ||||
|  | ||||
| export interface Alignment { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| // place here categories that you want to track. We want to track just a | ||||
| // small subset of categories at a given time. | ||||
| const ALLOWED_CATEGORIES_TO_TRACK = new Set(["command_palette"]); | ||||
| const ALLOWED_CATEGORIES_TO_TRACK = ["ai", "command_palette"] as string[]; | ||||
|  | ||||
| export const trackEvent = ( | ||||
|   category: string, | ||||
| @@ -9,20 +9,17 @@ export const trackEvent = ( | ||||
|   value?: number, | ||||
| ) => { | ||||
|   try { | ||||
|     // prettier-ignore | ||||
|     if ( | ||||
|       typeof window === "undefined" || | ||||
|       import.meta.env.VITE_WORKER_ID || | ||||
|       import.meta.env.VITE_APP_ENABLE_TRACKING !== "true" | ||||
|       typeof window === "undefined" | ||||
|       || import.meta.env.VITE_WORKER_ID | ||||
|       // comment out to debug locally | ||||
|       || import.meta.env.PROD | ||||
|     ) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     if (!ALLOWED_CATEGORIES_TO_TRACK.has(category)) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     if (import.meta.env.DEV) { | ||||
|       // comment out to debug in dev | ||||
|     if (!ALLOWED_CATEGORIES_TO_TRACK.includes(category)) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| import type { LaserPointerOptions } from "@excalidraw/laser-pointer"; | ||||
| import { LaserPointer } from "@excalidraw/laser-pointer"; | ||||
| import type { AnimationFrameHandler } from "./animation-frame-handler"; | ||||
| import type { AppState } from "./types"; | ||||
| import { LaserPointer, LaserPointerOptions } from "@excalidraw/laser-pointer"; | ||||
| import { AnimationFrameHandler } from "./animation-frame-handler"; | ||||
| import { AppState } from "./types"; | ||||
| import { getSvgPathFromStroke, sceneCoordsToViewportCoords } from "./utils"; | ||||
| import type App from "./components/App"; | ||||
| import { SVG_NS } from "./constants"; | ||||
|   | ||||
| @@ -1,15 +1,13 @@ | ||||
| import { COLOR_PALETTE } from "./colors"; | ||||
| import { | ||||
|   ARROW_TYPE, | ||||
|   DEFAULT_ELEMENT_PROPS, | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_FONT_SIZE, | ||||
|   DEFAULT_TEXT_ALIGN, | ||||
|   EXPORT_SCALES, | ||||
|   STATS_PANELS, | ||||
|   THEME, | ||||
| } from "./constants"; | ||||
| import type { AppState, NormalizedZoomValue } from "./types"; | ||||
| import { AppState, NormalizedZoomValue } from "./types"; | ||||
|  | ||||
| const defaultExportScale = EXPORT_SCALES.includes(devicePixelRatio) | ||||
|   ? devicePixelRatio | ||||
| @@ -34,14 +32,12 @@ export const getDefaultAppState = (): Omit< | ||||
|     currentItemStartArrowhead: null, | ||||
|     currentItemStrokeColor: DEFAULT_ELEMENT_PROPS.strokeColor, | ||||
|     currentItemRoundness: "round", | ||||
|     currentItemArrowType: ARROW_TYPE.round, | ||||
|     currentItemStrokeStyle: DEFAULT_ELEMENT_PROPS.strokeStyle, | ||||
|     currentItemStrokeWidth: DEFAULT_ELEMENT_PROPS.strokeWidth, | ||||
|     currentItemTextAlign: DEFAULT_TEXT_ALIGN, | ||||
|     currentHoveredFontFamily: null, | ||||
|     cursorButton: "up", | ||||
|     activeEmbeddable: null, | ||||
|     newElement: null, | ||||
|     draggingElement: null, | ||||
|     editingElement: null, | ||||
|     editingGroupId: null, | ||||
|     editingLinearElement: null, | ||||
| @@ -84,10 +80,7 @@ export const getDefaultAppState = (): Omit< | ||||
|     selectedElementsAreBeingDragged: false, | ||||
|     selectionElement: null, | ||||
|     shouldCacheIgnoreZoom: false, | ||||
|     stats: { | ||||
|       open: false, | ||||
|       panels: STATS_PANELS.generalStats | STATS_PANELS.elementProperties, | ||||
|     }, | ||||
|     showStats: false, | ||||
|     startBoundElement: null, | ||||
|     suggestedBindings: [], | ||||
|     frameRendering: { enabled: true, clip: true, name: true, outline: true }, | ||||
| @@ -145,11 +138,6 @@ const APP_STATE_STORAGE_CONF = (< | ||||
|     export: false, | ||||
|     server: false, | ||||
|   }, | ||||
|   currentItemArrowType: { | ||||
|     browser: true, | ||||
|     export: false, | ||||
|     server: false, | ||||
|   }, | ||||
|   currentItemOpacity: { browser: true, export: false, server: false }, | ||||
|   currentItemRoughness: { browser: true, export: false, server: false }, | ||||
|   currentItemStartArrowhead: { browser: true, export: false, server: false }, | ||||
| @@ -157,10 +145,9 @@ const APP_STATE_STORAGE_CONF = (< | ||||
|   currentItemStrokeStyle: { browser: true, export: false, server: false }, | ||||
|   currentItemStrokeWidth: { browser: true, export: false, server: false }, | ||||
|   currentItemTextAlign: { browser: true, export: false, server: false }, | ||||
|   currentHoveredFontFamily: { browser: false, export: false, server: false }, | ||||
|   cursorButton: { browser: true, export: false, server: false }, | ||||
|   activeEmbeddable: { browser: false, export: false, server: false }, | ||||
|   newElement: { browser: false, export: false, server: false }, | ||||
|   draggingElement: { browser: false, export: false, server: false }, | ||||
|   editingElement: { browser: false, export: false, server: false }, | ||||
|   editingGroupId: { browser: true, export: false, server: false }, | ||||
|   editingLinearElement: { browser: false, export: false, server: false }, | ||||
| @@ -209,7 +196,7 @@ const APP_STATE_STORAGE_CONF = (< | ||||
|   }, | ||||
|   selectionElement: { browser: false, export: false, server: false }, | ||||
|   shouldCacheIgnoreZoom: { browser: true, export: false, server: false }, | ||||
|   stats: { browser: true, export: false, server: false }, | ||||
|   showStats: { browser: true, export: false, server: false }, | ||||
|   startBoundElement: { browser: false, export: false, server: false }, | ||||
|   suggestedBindings: { browser: false, export: false, server: false }, | ||||
|   frameRendering: { browser: false, export: false, server: false }, | ||||
|   | ||||
| @@ -1,105 +0,0 @@ | ||||
| export default class BinaryHeap<T> { | ||||
|   private content: T[] = []; | ||||
|  | ||||
|   constructor(private scoreFunction: (node: T) => number) {} | ||||
|  | ||||
|   sinkDown(idx: number) { | ||||
|     const node = this.content[idx]; | ||||
|     while (idx > 0) { | ||||
|       const parentN = ((idx + 1) >> 1) - 1; | ||||
|       const parent = this.content[parentN]; | ||||
|       if (this.scoreFunction(node) < this.scoreFunction(parent)) { | ||||
|         this.content[parentN] = node; | ||||
|         this.content[idx] = parent; | ||||
|         idx = parentN; // TODO: Optimize | ||||
|       } else { | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   bubbleUp(idx: number) { | ||||
|     const length = this.content.length; | ||||
|     const node = this.content[idx]; | ||||
|     const score = this.scoreFunction(node); | ||||
|  | ||||
|     while (true) { | ||||
|       const child2N = (idx + 1) << 1; | ||||
|       const child1N = child2N - 1; | ||||
|       let swap = null; | ||||
|       let child1Score = 0; | ||||
|  | ||||
|       if (child1N < length) { | ||||
|         const child1 = this.content[child1N]; | ||||
|         child1Score = this.scoreFunction(child1); | ||||
|         if (child1Score < score) { | ||||
|           swap = child1N; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       if (child2N < length) { | ||||
|         const child2 = this.content[child2N]; | ||||
|         const child2Score = this.scoreFunction(child2); | ||||
|         if (child2Score < (swap === null ? score : child1Score)) { | ||||
|           swap = child2N; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       if (swap !== null) { | ||||
|         this.content[idx] = this.content[swap]; | ||||
|         this.content[swap] = node; | ||||
|         idx = swap; // TODO: Optimize | ||||
|       } else { | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   push(node: T) { | ||||
|     this.content.push(node); | ||||
|     this.sinkDown(this.content.length - 1); | ||||
|   } | ||||
|  | ||||
|   pop(): T | null { | ||||
|     if (this.content.length === 0) { | ||||
|       return null; | ||||
|     } | ||||
|  | ||||
|     const result = this.content[0]; | ||||
|     const end = this.content.pop()!; | ||||
|  | ||||
|     if (this.content.length > 0) { | ||||
|       this.content[0] = end; | ||||
|       this.bubbleUp(0); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   remove(node: T) { | ||||
|     if (this.content.length === 0) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     const i = this.content.indexOf(node); | ||||
|     const end = this.content.pop()!; | ||||
|  | ||||
|     if (i < this.content.length) { | ||||
|       this.content[i] = end; | ||||
|  | ||||
|       if (this.scoreFunction(end) < this.scoreFunction(node)) { | ||||
|         this.sinkDown(i); | ||||
|       } else { | ||||
|         this.bubbleUp(i); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   size(): number { | ||||
|     return this.content.length; | ||||
|   } | ||||
|  | ||||
|   rescoreElement(node: T) { | ||||
|     this.sinkDown(this.content.indexOf(node)); | ||||
|   } | ||||
| } | ||||
| @@ -1,14 +1,18 @@ | ||||
| import { ENV } from "./constants"; | ||||
| import type { BindableProp, BindingProp } from "./element/binding"; | ||||
| import { | ||||
|   BoundElement, | ||||
|   BindableElement, | ||||
|   BindableProp, | ||||
|   BindingProp, | ||||
|   bindingProperties, | ||||
|   updateBoundElements, | ||||
| } from "./element/binding"; | ||||
| import { LinearElementEditor } from "./element/linearElementEditor"; | ||||
| import type { ElementUpdate } from "./element/mutateElement"; | ||||
| import { mutateElement, newElementWith } from "./element/mutateElement"; | ||||
| import { | ||||
|   ElementUpdate, | ||||
|   mutateElement, | ||||
|   newElementWith, | ||||
| } from "./element/mutateElement"; | ||||
| import { | ||||
|   getBoundTextElementId, | ||||
|   redrawTextBoundingBox, | ||||
| @@ -19,7 +23,7 @@ import { | ||||
|   isBoundToContainer, | ||||
|   isTextElement, | ||||
| } from "./element/typeChecks"; | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawLinearElement, | ||||
|   ExcalidrawTextElement, | ||||
| @@ -30,13 +34,13 @@ import type { | ||||
| import { orderByFractionalIndex, syncMovedIndices } from "./fractionalIndex"; | ||||
| import { getNonDeletedGroupIds } from "./groups"; | ||||
| import { getObservedAppState } from "./store"; | ||||
| import type { | ||||
| import { | ||||
|   AppState, | ||||
|   ObservedAppState, | ||||
|   ObservedElementsAppState, | ||||
|   ObservedStandaloneAppState, | ||||
| } from "./types"; | ||||
| import type { SubtypeOf, ValueOf } from "./utility-types"; | ||||
| import { SubtypeOf, ValueOf } from "./utility-types"; | ||||
| import { | ||||
|   arrayToMap, | ||||
|   arrayToObject, | ||||
| @@ -1100,6 +1104,7 @@ export class ElementsChange implements Change<SceneElementsMap> { | ||||
|     try { | ||||
|       // TODO: #7348 refactor away mutations below, so that we couldn't end up in an incosistent state | ||||
|       ElementsChange.redrawTextBoundingBoxes(nextElements, changedElements); | ||||
|       ElementsChange.redrawBoundArrows(nextElements, changedElements); | ||||
|  | ||||
|       // the following reorder performs also mutations, but only on new instances of changed elements | ||||
|       // (unless something goes really bad and it fallbacks to fixing all invalid indices) | ||||
| @@ -1108,9 +1113,6 @@ export class ElementsChange implements Change<SceneElementsMap> { | ||||
|         changedElements, | ||||
|         flags, | ||||
|       ); | ||||
|  | ||||
|       // Need ordered nextElements to avoid z-index binding issues | ||||
|       ElementsChange.redrawBoundArrows(nextElements, changedElements); | ||||
|     } catch (e) { | ||||
|       console.error( | ||||
|         `Couldn't mutate elements after applying elements change`, | ||||
| @@ -1462,9 +1464,7 @@ export class ElementsChange implements Change<SceneElementsMap> { | ||||
|   ) { | ||||
|     for (const element of changed.values()) { | ||||
|       if (!element.isDeleted && isBindableElement(element)) { | ||||
|         updateBoundElements(element, elements, { | ||||
|           changedElements: changed, | ||||
|         }); | ||||
|         updateBoundElements(element, elements); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| @@ -1481,28 +1481,19 @@ export class ElementsChange implements Change<SceneElementsMap> { | ||||
|       return elements; | ||||
|     } | ||||
|  | ||||
|     const unordered = Array.from(elements.values()); | ||||
|     const ordered = orderByFractionalIndex([...unordered]); | ||||
|     const moved = Delta.getRightDifferences(unordered, ordered, true).reduce( | ||||
|       (acc, arrayIndex) => { | ||||
|         const candidate = unordered[Number(arrayIndex)]; | ||||
|         if (candidate && changed.has(candidate.id)) { | ||||
|           acc.set(candidate.id, candidate); | ||||
|         } | ||||
|     const previous = Array.from(elements.values()); | ||||
|     const reordered = orderByFractionalIndex([...previous]); | ||||
|  | ||||
|         return acc; | ||||
|       }, | ||||
|       new Map(), | ||||
|     ); | ||||
|  | ||||
|     if (!flags.containsVisibleDifference && moved.size) { | ||||
|     if ( | ||||
|       !flags.containsVisibleDifference && | ||||
|       Delta.isRightDifferent(previous, reordered, true) | ||||
|     ) { | ||||
|       // we found a difference in order! | ||||
|       flags.containsVisibleDifference = true; | ||||
|     } | ||||
|  | ||||
|     // synchronize all elements that were actually moved | ||||
|     // could fallback to synchronizing all invalid indices | ||||
|     return arrayToMap(syncMovedIndices(ordered, moved)) as typeof elements; | ||||
|     // let's synchronize all invalid indices of moved elements | ||||
|     return arrayToMap(syncMovedIndices(reordered, changed)) as typeof elements; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|   | ||||
| @@ -1,5 +1,9 @@ | ||||
| import type { Spreadsheet } from "./charts"; | ||||
| import { tryParseCells, tryParseNumber, VALID_SPREADSHEET } from "./charts"; | ||||
| import { | ||||
|   Spreadsheet, | ||||
|   tryParseCells, | ||||
|   tryParseNumber, | ||||
|   VALID_SPREADSHEET, | ||||
| } from "./charts"; | ||||
|  | ||||
| describe("charts", () => { | ||||
|   describe("tryParseNumber", () => { | ||||
|   | ||||
| @@ -9,7 +9,7 @@ import { | ||||
|   VERTICAL_ALIGN, | ||||
| } from "./constants"; | ||||
| import { newElement, newLinearElement, newTextElement } from "./element"; | ||||
| import type { NonDeletedExcalidrawElement } from "./element/types"; | ||||
| import { NonDeletedExcalidrawElement } from "./element/types"; | ||||
| import { randomId } from "./random"; | ||||
|  | ||||
| export type ChartElements = readonly NonDeletedExcalidrawElement[]; | ||||
| @@ -257,6 +257,8 @@ const chartLines = ( | ||||
|     type: "line", | ||||
|     x, | ||||
|     y, | ||||
|     startArrowhead: null, | ||||
|     endArrowhead: null, | ||||
|     width: chartWidth, | ||||
|     points: [ | ||||
|       [0, 0], | ||||
| @@ -271,6 +273,8 @@ const chartLines = ( | ||||
|     type: "line", | ||||
|     x, | ||||
|     y, | ||||
|     startArrowhead: null, | ||||
|     endArrowhead: null, | ||||
|     height: chartHeight, | ||||
|     points: [ | ||||
|       [0, 0], | ||||
| @@ -285,6 +289,8 @@ const chartLines = ( | ||||
|     type: "line", | ||||
|     x, | ||||
|     y: y - BAR_HEIGHT - BAR_GAP, | ||||
|     startArrowhead: null, | ||||
|     endArrowhead: null, | ||||
|     strokeStyle: "dotted", | ||||
|     width: chartWidth, | ||||
|     opacity: GRID_OPACITY, | ||||
| @@ -412,6 +418,8 @@ const chartTypeLine = ( | ||||
|     type: "line", | ||||
|     x: x + BAR_GAP + BAR_WIDTH / 2, | ||||
|     y: y - BAR_GAP, | ||||
|     startArrowhead: null, | ||||
|     endArrowhead: null, | ||||
|     height: maxY - minY, | ||||
|     width: maxX - minX, | ||||
|     strokeWidth: 2, | ||||
| @@ -445,6 +453,8 @@ const chartTypeLine = ( | ||||
|       type: "line", | ||||
|       x: x + cx + BAR_WIDTH / 2 + BAR_GAP / 2, | ||||
|       y: y - cy, | ||||
|       startArrowhead: null, | ||||
|       endArrowhead: null, | ||||
|       height: cy, | ||||
|       strokeStyle: "dotted", | ||||
|       opacity: GRID_OPACITY, | ||||
|   | ||||
| @@ -5,13 +5,13 @@ import { | ||||
|   THEME, | ||||
| } from "./constants"; | ||||
| import { roundRect } from "./renderer/roundRect"; | ||||
| import type { InteractiveCanvasRenderConfig } from "./scene/types"; | ||||
| import type { | ||||
| import { InteractiveCanvasRenderConfig } from "./scene/types"; | ||||
| import { | ||||
|   Collaborator, | ||||
|   InteractiveCanvasAppState, | ||||
|   SocketId, | ||||
|   UserIdleState, | ||||
| } from "./types"; | ||||
| import { UserIdleState } from "./types"; | ||||
|  | ||||
| function hashToInteger(id: string) { | ||||
|   let hash = 0; | ||||
|   | ||||
| @@ -1,10 +1,9 @@ | ||||
| import type { | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   NonDeletedExcalidrawElement, | ||||
| } from "./element/types"; | ||||
| import type { BinaryFiles } from "./types"; | ||||
| import type { Spreadsheet } from "./charts"; | ||||
| import { tryParseSpreadsheet, VALID_SPREADSHEET } from "./charts"; | ||||
| import { BinaryFiles } from "./types"; | ||||
| import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts"; | ||||
| import { | ||||
|   ALLOWED_PASTE_MIME_TYPES, | ||||
|   EXPORT_DATA_TYPES, | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import oc from "open-color"; | ||||
| import type { Merge } from "./utility-types"; | ||||
| import { Merge } from "./utility-types"; | ||||
|  | ||||
| // FIXME can't put to utils.ts rn because of circular dependency | ||||
| const pick = <R extends Record<string, any>, K extends readonly (keyof R)[]>( | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { useState } from "react"; | ||||
| import type { ActionManager } from "../actions/manager"; | ||||
| import type { | ||||
| import { ActionManager } from "../actions/manager"; | ||||
| import { | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawElementType, | ||||
|   NonDeletedElementsMap, | ||||
| @@ -17,18 +17,13 @@ import { | ||||
|   hasStrokeWidth, | ||||
| } from "../scene"; | ||||
| import { SHAPES } from "../shapes"; | ||||
| import type { AppClassProperties, AppProps, UIAppState, Zoom } from "../types"; | ||||
| import { AppClassProperties, AppProps, UIAppState, Zoom } from "../types"; | ||||
| import { capitalizeString, isTransparent } from "../utils"; | ||||
| import Stack from "./Stack"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { hasStrokeColor, toolIsArrow } from "../scene/comparisons"; | ||||
| import { hasStrokeColor } from "../scene/comparisons"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { | ||||
|   hasBoundTextElement, | ||||
|   isElbowArrow, | ||||
|   isLinearElement, | ||||
|   isTextElement, | ||||
| } from "../element/typeChecks"; | ||||
| import { hasBoundTextElement, isTextElement } from "../element/typeChecks"; | ||||
| import clsx from "clsx"; | ||||
| import { actionToggleZenMode } from "../actions"; | ||||
| import { Tooltip } from "./Tooltip"; | ||||
| @@ -119,12 +114,6 @@ export const SelectedShapeActions = ({ | ||||
|   const showLinkIcon = | ||||
|     targetElements.length === 1 || isSingleElementBoundContainer; | ||||
|  | ||||
|   const showLineEditorAction = | ||||
|     !appState.editingLinearElement && | ||||
|     targetElements.length === 1 && | ||||
|     isLinearElement(targetElements[0]) && | ||||
|     !isElbowArrow(targetElements[0]); | ||||
|  | ||||
|   return ( | ||||
|     <div className="panelColumn"> | ||||
|       <div> | ||||
| @@ -157,16 +146,13 @@ export const SelectedShapeActions = ({ | ||||
|         <>{renderAction("changeRoundness")}</> | ||||
|       )} | ||||
|  | ||||
|       {(toolIsArrow(appState.activeTool.type) || | ||||
|         targetElements.some((element) => toolIsArrow(element.type))) && ( | ||||
|         <>{renderAction("changeArrowType")}</> | ||||
|       )} | ||||
|  | ||||
|       {(appState.activeTool.type === "text" || | ||||
|         targetElements.some(isTextElement)) && ( | ||||
|         <> | ||||
|           {renderAction("changeFontFamily")} | ||||
|           {renderAction("changeFontSize")} | ||||
|  | ||||
|           {renderAction("changeFontFamily")} | ||||
|  | ||||
|           {(appState.activeTool.type === "text" || | ||||
|             suppportsHorizontalAlign(targetElements, elementsMap)) && | ||||
|             renderAction("changeTextAlign")} | ||||
| @@ -187,8 +173,8 @@ export const SelectedShapeActions = ({ | ||||
|         <div className="buttonList"> | ||||
|           {renderAction("sendToBack")} | ||||
|           {renderAction("sendBackward")} | ||||
|           {renderAction("bringForward")} | ||||
|           {renderAction("bringToFront")} | ||||
|           {renderAction("bringForward")} | ||||
|         </div> | ||||
|       </fieldset> | ||||
|  | ||||
| @@ -243,7 +229,6 @@ export const SelectedShapeActions = ({ | ||||
|             {renderAction("group")} | ||||
|             {renderAction("ungroup")} | ||||
|             {showLinkIcon && renderAction("hyperlink")} | ||||
|             {showLineEditorAction && renderAction("toggleLinearEditor")} | ||||
|           </div> | ||||
|         </fieldset> | ||||
|       )} | ||||
| @@ -348,8 +333,8 @@ export const ShapesSwitcher = ({ | ||||
|                 fontSize: 8, | ||||
|                 fontFamily: "Cascadia, monospace", | ||||
|                 position: "absolute", | ||||
|                 background: "var(--color-promo)", | ||||
|                 color: "var(--color-surface-lowest)", | ||||
|                 background: "pink", | ||||
|                 color: "black", | ||||
|                 bottom: 3, | ||||
|                 right: 4, | ||||
|               }} | ||||
| @@ -473,7 +458,6 @@ export const ExitZenModeAction = ({ | ||||
|   showExitZenModeBtn: boolean; | ||||
| }) => ( | ||||
|   <button | ||||
|     type="button" | ||||
|     className={clsx("disable-zen-mode", { | ||||
|       "disable-zen-mode--visible": showExitZenModeBtn, | ||||
|     })} | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,12 +0,0 @@ | ||||
| @import "../css/theme"; | ||||
|  | ||||
| .excalidraw { | ||||
|   button.standalone { | ||||
|     @include outlineButtonIconStyles; | ||||
|  | ||||
|     & > * { | ||||
|       // dissalow pointer events on children, so we always have event.target on the button itself | ||||
|       pointer-events: none; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,36 +0,0 @@ | ||||
| import { forwardRef } from "react"; | ||||
| import clsx from "clsx"; | ||||
|  | ||||
| import "./ButtonIcon.scss"; | ||||
|  | ||||
| interface ButtonIconProps { | ||||
|   icon: JSX.Element; | ||||
|   title: string; | ||||
|   className?: string; | ||||
|   testId?: string; | ||||
|   /** if not supplied, defaults to value identity check */ | ||||
|   active?: boolean; | ||||
|   /** include standalone style (could interfere with parent styles) */ | ||||
|   standalone?: boolean; | ||||
|   onClick: (event: React.MouseEvent<HTMLButtonElement, MouseEvent>) => void; | ||||
| } | ||||
|  | ||||
| export const ButtonIcon = forwardRef<HTMLButtonElement, ButtonIconProps>( | ||||
|   (props, ref) => { | ||||
|     const { title, className, testId, active, standalone, icon, onClick } = | ||||
|       props; | ||||
|     return ( | ||||
|       <button | ||||
|         type="button" | ||||
|         ref={ref} | ||||
|         key={title} | ||||
|         title={title} | ||||
|         data-testid={testId} | ||||
|         className={clsx(className, { standalone, active })} | ||||
|         onClick={onClick} | ||||
|       > | ||||
|         {icon} | ||||
|       </button> | ||||
|     ); | ||||
|   }, | ||||
| ); | ||||
| @@ -1,5 +1,4 @@ | ||||
| import clsx from "clsx"; | ||||
| import { ButtonIcon } from "./ButtonIcon"; | ||||
|  | ||||
| // TODO: It might be "clever" to add option.icon to the existing component <ButtonSelect /> | ||||
| export const ButtonIconSelect = <T extends Object>( | ||||
| @@ -25,17 +24,20 @@ export const ButtonIconSelect = <T extends Object>( | ||||
|       } | ||||
|   ), | ||||
| ) => ( | ||||
|   <div className="buttonList"> | ||||
|   <div className="buttonList buttonListIcon"> | ||||
|     {props.options.map((option) => | ||||
|       props.type === "button" ? ( | ||||
|         <ButtonIcon | ||||
|         <button | ||||
|           key={option.text} | ||||
|           icon={option.icon} | ||||
|           title={option.text} | ||||
|           testId={option.testId} | ||||
|           active={option.active ?? props.value === option.value} | ||||
|           onClick={(event) => props.onClick(option.value, event)} | ||||
|         /> | ||||
|           className={clsx({ | ||||
|             active: option.active ?? props.value === option.value, | ||||
|           })} | ||||
|           data-testid={option.testId} | ||||
|           title={option.text} | ||||
|         > | ||||
|           {option.icon} | ||||
|         </button> | ||||
|       ) : ( | ||||
|         <label | ||||
|           key={option.text} | ||||
|   | ||||
| @@ -1,10 +0,0 @@ | ||||
| export const ButtonSeparator = () => ( | ||||
|   <div | ||||
|     style={{ | ||||
|       width: 1, | ||||
|       height: "1rem", | ||||
|       backgroundColor: "var(--default-border-color)", | ||||
|       margin: "0 auto", | ||||
|     }} | ||||
|   /> | ||||
| ); | ||||
| @@ -22,12 +22,7 @@ export const CheckboxItem: React.FC<{ | ||||
|         ).focus(); | ||||
|       }} | ||||
|     > | ||||
|       <button | ||||
|         type="button" | ||||
|         className="Checkbox-box" | ||||
|         role="checkbox" | ||||
|         aria-checked={checked} | ||||
|       > | ||||
|       <button className="Checkbox-box" role="checkbox" aria-checked={checked}> | ||||
|         {checkIcon} | ||||
|       </button> | ||||
|       <div className="Checkbox-label">{children}</div> | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user