mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-26 08:24:20 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			606 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			606 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { clamp, roundToStep } from "@excalidraw/math";
 | |
| 
 | |
| import {
 | |
|   DEFAULT_CANVAS_BACKGROUND_PICKS,
 | |
|   CURSOR_TYPE,
 | |
|   MAX_ZOOM,
 | |
|   MIN_ZOOM,
 | |
|   THEME,
 | |
|   ZOOM_STEP,
 | |
|   getShortcutKey,
 | |
|   updateActiveTool,
 | |
|   CODES,
 | |
|   KEYS,
 | |
| } from "@excalidraw/common";
 | |
| 
 | |
| import { getNonDeletedElements } from "@excalidraw/element";
 | |
| import { newElementWith } from "@excalidraw/element";
 | |
| import { getCommonBounds, type SceneBounds } from "@excalidraw/element";
 | |
| 
 | |
| import { CaptureUpdateAction } from "@excalidraw/element";
 | |
| 
 | |
| import type { ExcalidrawElement } from "@excalidraw/element/types";
 | |
| 
 | |
| import {
 | |
|   getDefaultAppState,
 | |
|   isEraserActive,
 | |
|   isHandToolActive,
 | |
| } from "../appState";
 | |
| import { ColorPicker } from "../components/ColorPicker/ColorPicker";
 | |
| import { ToolButton } from "../components/ToolButton";
 | |
| import { Tooltip } from "../components/Tooltip";
 | |
| import {
 | |
|   handIcon,
 | |
|   LassoIcon,
 | |
|   MoonIcon,
 | |
|   SunIcon,
 | |
|   TrashIcon,
 | |
|   zoomAreaIcon,
 | |
|   ZoomInIcon,
 | |
|   ZoomOutIcon,
 | |
|   ZoomResetIcon,
 | |
| } from "../components/icons";
 | |
| import { setCursor } from "../cursor";
 | |
| 
 | |
| import { t } from "../i18n";
 | |
| import { getNormalizedZoom } from "../scene";
 | |
| import { centerScrollOn } from "../scene/scroll";
 | |
| import { getStateForZoom } from "../scene/zoom";
 | |
| 
 | |
| import { register } from "./register";
 | |
| 
 | |
| import type { AppState, Offsets } from "../types";
 | |
| 
 | |
| export const actionChangeViewBackgroundColor = register({
 | |
|   name: "changeViewBackgroundColor",
 | |
|   label: "labels.canvasBackground",
 | |
|   trackEvent: false,
 | |
|   predicate: (elements, appState, props, app) => {
 | |
|     return (
 | |
|       !!app.props.UIOptions.canvasActions.changeViewBackgroundColor &&
 | |
|       !appState.viewModeEnabled
 | |
|     );
 | |
|   },
 | |
|   perform: (_, appState, value) => {
 | |
|     return {
 | |
|       appState: { ...appState, ...value },
 | |
|       captureUpdate: !!value.viewBackgroundColor
 | |
|         ? CaptureUpdateAction.IMMEDIATELY
 | |
|         : CaptureUpdateAction.EVENTUALLY,
 | |
|     };
 | |
|   },
 | |
|   PanelComponent: ({ elements, appState, updateData, appProps, data }) => {
 | |
|     // FIXME move me to src/components/mainMenu/DefaultItems.tsx
 | |
|     return (
 | |
|       <ColorPicker
 | |
|         palette={null}
 | |
|         topPicks={DEFAULT_CANVAS_BACKGROUND_PICKS}
 | |
|         label={t("labels.canvasBackground")}
 | |
|         type="canvasBackground"
 | |
|         color={appState.viewBackgroundColor}
 | |
|         onChange={(color) => updateData({ viewBackgroundColor: color })}
 | |
|         data-testid="canvas-background-picker"
 | |
|         elements={elements}
 | |
|         appState={appState}
 | |
|         updateData={updateData}
 | |
|         compactMode={appState.stylesPanelMode === "compact"}
 | |
|       />
 | |
|     );
 | |
|   },
 | |
| });
 | |
| 
 | |
| export const actionClearCanvas = register({
 | |
|   name: "clearCanvas",
 | |
|   label: "labels.clearCanvas",
 | |
|   icon: TrashIcon,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   predicate: (elements, appState, props, app) => {
 | |
|     return (
 | |
|       !!app.props.UIOptions.canvasActions.clearCanvas &&
 | |
|       !appState.viewModeEnabled &&
 | |
|       appState.openDialog?.name !== "elementLinkSelector"
 | |
|     );
 | |
|   },
 | |
|   perform: (elements, appState, _, app) => {
 | |
|     app.imageCache.clear();
 | |
|     return {
 | |
|       elements: elements.map((element) =>
 | |
|         newElementWith(element, { isDeleted: true }),
 | |
|       ),
 | |
|       appState: {
 | |
|         ...getDefaultAppState(),
 | |
|         files: {},
 | |
|         theme: appState.theme,
 | |
|         penMode: appState.penMode,
 | |
|         penDetected: appState.penDetected,
 | |
|         exportBackground: appState.exportBackground,
 | |
|         exportEmbedScene: appState.exportEmbedScene,
 | |
|         gridSize: appState.gridSize,
 | |
|         gridStep: appState.gridStep,
 | |
|         gridModeEnabled: appState.gridModeEnabled,
 | |
|         stats: appState.stats,
 | |
|         pasteDialog: appState.pasteDialog,
 | |
|         activeTool:
 | |
|           appState.activeTool.type === "image"
 | |
|             ? {
 | |
|                 ...appState.activeTool,
 | |
|                 type: app.state.preferredSelectionTool.type,
 | |
|               }
 | |
|             : appState.activeTool,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.IMMEDIATELY,
 | |
|     };
 | |
|   },
 | |
| });
 | |
| 
 | |
| export const actionZoomIn = register({
 | |
|   name: "zoomIn",
 | |
|   label: "buttons.zoomIn",
 | |
|   viewMode: true,
 | |
|   icon: ZoomInIcon,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (_elements, appState, _, app) => {
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         ...getStateForZoom(
 | |
|           {
 | |
|             viewportX: appState.width / 2 + appState.offsetLeft,
 | |
|             viewportY: appState.height / 2 + appState.offsetTop,
 | |
|             nextZoom: getNormalizedZoom(appState.zoom.value + ZOOM_STEP),
 | |
|           },
 | |
|           appState,
 | |
|         ),
 | |
|         userToFollow: null,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.EVENTUALLY,
 | |
|     };
 | |
|   },
 | |
|   PanelComponent: ({ updateData, appState }) => (
 | |
|     <ToolButton
 | |
|       type="button"
 | |
|       className="zoom-in-button zoom-button"
 | |
|       icon={ZoomInIcon}
 | |
|       title={`${t("buttons.zoomIn")} — ${getShortcutKey("CtrlOrCmd++")}`}
 | |
|       aria-label={t("buttons.zoomIn")}
 | |
|       disabled={appState.zoom.value >= MAX_ZOOM}
 | |
|       onClick={() => {
 | |
|         updateData(null);
 | |
|       }}
 | |
|     />
 | |
|   ),
 | |
|   keyTest: (event) =>
 | |
|     (event.code === CODES.EQUAL || event.code === CODES.NUM_ADD) &&
 | |
|     (event[KEYS.CTRL_OR_CMD] || event.shiftKey),
 | |
| });
 | |
| 
 | |
| export const actionZoomOut = register({
 | |
|   name: "zoomOut",
 | |
|   label: "buttons.zoomOut",
 | |
|   icon: ZoomOutIcon,
 | |
|   viewMode: true,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (_elements, appState, _, app) => {
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         ...getStateForZoom(
 | |
|           {
 | |
|             viewportX: appState.width / 2 + appState.offsetLeft,
 | |
|             viewportY: appState.height / 2 + appState.offsetTop,
 | |
|             nextZoom: getNormalizedZoom(appState.zoom.value - ZOOM_STEP),
 | |
|           },
 | |
|           appState,
 | |
|         ),
 | |
|         userToFollow: null,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.EVENTUALLY,
 | |
|     };
 | |
|   },
 | |
|   PanelComponent: ({ updateData, appState }) => (
 | |
|     <ToolButton
 | |
|       type="button"
 | |
|       className="zoom-out-button zoom-button"
 | |
|       icon={ZoomOutIcon}
 | |
|       title={`${t("buttons.zoomOut")} — ${getShortcutKey("CtrlOrCmd+-")}`}
 | |
|       aria-label={t("buttons.zoomOut")}
 | |
|       disabled={appState.zoom.value <= MIN_ZOOM}
 | |
|       onClick={() => {
 | |
|         updateData(null);
 | |
|       }}
 | |
|     />
 | |
|   ),
 | |
|   keyTest: (event) =>
 | |
|     (event.code === CODES.MINUS || event.code === CODES.NUM_SUBTRACT) &&
 | |
|     (event[KEYS.CTRL_OR_CMD] || event.shiftKey),
 | |
| });
 | |
| 
 | |
| export const actionResetZoom = register({
 | |
|   name: "resetZoom",
 | |
|   label: "buttons.resetZoom",
 | |
|   icon: ZoomResetIcon,
 | |
|   viewMode: true,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (_elements, appState, _, app) => {
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         ...getStateForZoom(
 | |
|           {
 | |
|             viewportX: appState.width / 2 + appState.offsetLeft,
 | |
|             viewportY: appState.height / 2 + appState.offsetTop,
 | |
|             nextZoom: getNormalizedZoom(1),
 | |
|           },
 | |
|           appState,
 | |
|         ),
 | |
|         userToFollow: null,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.EVENTUALLY,
 | |
|     };
 | |
|   },
 | |
|   PanelComponent: ({ updateData, appState }) => (
 | |
|     <Tooltip label={t("buttons.resetZoom")} style={{ height: "100%" }}>
 | |
|       <ToolButton
 | |
|         type="button"
 | |
|         className="reset-zoom-button zoom-button"
 | |
|         title={t("buttons.resetZoom")}
 | |
|         aria-label={t("buttons.resetZoom")}
 | |
|         onClick={() => {
 | |
|           updateData(null);
 | |
|         }}
 | |
|       >
 | |
|         {(appState.zoom.value * 100).toFixed(0)}%
 | |
|       </ToolButton>
 | |
|     </Tooltip>
 | |
|   ),
 | |
|   keyTest: (event) =>
 | |
|     (event.code === CODES.ZERO || event.code === CODES.NUM_ZERO) &&
 | |
|     (event[KEYS.CTRL_OR_CMD] || event.shiftKey),
 | |
| });
 | |
| 
 | |
| const zoomValueToFitBoundsOnViewport = (
 | |
|   bounds: SceneBounds,
 | |
|   viewportDimensions: { width: number; height: number },
 | |
|   viewportZoomFactor: number = 1, // default to 1 if not provided
 | |
| ) => {
 | |
|   const [x1, y1, x2, y2] = bounds;
 | |
|   const commonBoundsWidth = x2 - x1;
 | |
|   const zoomValueForWidth = viewportDimensions.width / commonBoundsWidth;
 | |
|   const commonBoundsHeight = y2 - y1;
 | |
|   const zoomValueForHeight = viewportDimensions.height / commonBoundsHeight;
 | |
|   const smallestZoomValue = Math.min(zoomValueForWidth, zoomValueForHeight);
 | |
| 
 | |
|   const adjustedZoomValue =
 | |
|     smallestZoomValue * clamp(viewportZoomFactor, 0.1, 1);
 | |
| 
 | |
|   return Math.min(adjustedZoomValue, 1);
 | |
| };
 | |
| 
 | |
| export const zoomToFitBounds = ({
 | |
|   bounds,
 | |
|   appState,
 | |
|   canvasOffsets,
 | |
|   fitToViewport = false,
 | |
|   viewportZoomFactor = 1,
 | |
|   minZoom = -Infinity,
 | |
|   maxZoom = Infinity,
 | |
| }: {
 | |
|   bounds: SceneBounds;
 | |
|   canvasOffsets?: Offsets;
 | |
|   appState: Readonly<AppState>;
 | |
|   /** whether to fit content to viewport (beyond >100%) */
 | |
|   fitToViewport: boolean;
 | |
|   /** zoom content to cover X of the viewport, when fitToViewport=true */
 | |
|   viewportZoomFactor?: number;
 | |
|   minZoom?: number;
 | |
|   maxZoom?: number;
 | |
| }) => {
 | |
|   viewportZoomFactor = clamp(viewportZoomFactor, MIN_ZOOM, MAX_ZOOM);
 | |
| 
 | |
|   const [x1, y1, x2, y2] = bounds;
 | |
|   const centerX = (x1 + x2) / 2;
 | |
|   const centerY = (y1 + y2) / 2;
 | |
| 
 | |
|   const canvasOffsetLeft = canvasOffsets?.left ?? 0;
 | |
|   const canvasOffsetTop = canvasOffsets?.top ?? 0;
 | |
|   const canvasOffsetRight = canvasOffsets?.right ?? 0;
 | |
|   const canvasOffsetBottom = canvasOffsets?.bottom ?? 0;
 | |
| 
 | |
|   const effectiveCanvasWidth =
 | |
|     appState.width - canvasOffsetLeft - canvasOffsetRight;
 | |
|   const effectiveCanvasHeight =
 | |
|     appState.height - canvasOffsetTop - canvasOffsetBottom;
 | |
| 
 | |
|   let adjustedZoomValue;
 | |
| 
 | |
|   if (fitToViewport) {
 | |
|     const commonBoundsWidth = x2 - x1;
 | |
|     const commonBoundsHeight = y2 - y1;
 | |
| 
 | |
|     adjustedZoomValue =
 | |
|       Math.min(
 | |
|         effectiveCanvasWidth / commonBoundsWidth,
 | |
|         effectiveCanvasHeight / commonBoundsHeight,
 | |
|       ) * viewportZoomFactor;
 | |
|   } else {
 | |
|     adjustedZoomValue = zoomValueToFitBoundsOnViewport(
 | |
|       bounds,
 | |
|       {
 | |
|         width: effectiveCanvasWidth,
 | |
|         height: effectiveCanvasHeight,
 | |
|       },
 | |
|       viewportZoomFactor,
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   const newZoomValue = getNormalizedZoom(
 | |
|     clamp(roundToStep(adjustedZoomValue, ZOOM_STEP, "floor"), minZoom, maxZoom),
 | |
|   );
 | |
| 
 | |
|   const centerScroll = centerScrollOn({
 | |
|     scenePoint: { x: centerX, y: centerY },
 | |
|     viewportDimensions: {
 | |
|       width: appState.width,
 | |
|       height: appState.height,
 | |
|     },
 | |
|     offsets: canvasOffsets,
 | |
|     zoom: { value: newZoomValue },
 | |
|   });
 | |
| 
 | |
|   return {
 | |
|     appState: {
 | |
|       ...appState,
 | |
|       scrollX: centerScroll.scrollX,
 | |
|       scrollY: centerScroll.scrollY,
 | |
|       zoom: { value: newZoomValue },
 | |
|     },
 | |
|     captureUpdate: CaptureUpdateAction.EVENTUALLY,
 | |
|   };
 | |
| };
 | |
| 
 | |
| export const zoomToFit = ({
 | |
|   canvasOffsets,
 | |
|   targetElements,
 | |
|   appState,
 | |
|   fitToViewport,
 | |
|   viewportZoomFactor,
 | |
|   minZoom,
 | |
|   maxZoom,
 | |
| }: {
 | |
|   canvasOffsets?: Offsets;
 | |
|   targetElements: readonly ExcalidrawElement[];
 | |
|   appState: Readonly<AppState>;
 | |
|   /** whether to fit content to viewport (beyond >100%) */
 | |
|   fitToViewport: boolean;
 | |
|   /** zoom content to cover X of the viewport, when fitToViewport=true */
 | |
|   viewportZoomFactor?: number;
 | |
|   minZoom?: number;
 | |
|   maxZoom?: number;
 | |
| }) => {
 | |
|   const commonBounds = getCommonBounds(getNonDeletedElements(targetElements));
 | |
| 
 | |
|   return zoomToFitBounds({
 | |
|     canvasOffsets,
 | |
|     bounds: commonBounds,
 | |
|     appState,
 | |
|     fitToViewport,
 | |
|     viewportZoomFactor,
 | |
|     minZoom,
 | |
|     maxZoom,
 | |
|   });
 | |
| };
 | |
| 
 | |
| // Note, this action differs from actionZoomToFitSelection in that it doesn't
 | |
| // zoom beyond 100%. In other words, if the content is smaller than viewport
 | |
| // size, it won't be zoomed in.
 | |
| export const actionZoomToFitSelectionInViewport = register({
 | |
|   name: "zoomToFitSelectionInViewport",
 | |
|   label: "labels.zoomToFitViewport",
 | |
|   icon: zoomAreaIcon,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (elements, appState, _, app) => {
 | |
|     const selectedElements = app.scene.getSelectedElements(appState);
 | |
|     return zoomToFit({
 | |
|       targetElements: selectedElements.length ? selectedElements : elements,
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         userToFollow: null,
 | |
|       },
 | |
|       fitToViewport: false,
 | |
|       canvasOffsets: app.getEditorUIOffsets(),
 | |
|     });
 | |
|   },
 | |
|   // NOTE shift-2 should have been assigned actionZoomToFitSelection.
 | |
|   // TBD on how proceed
 | |
|   keyTest: (event) =>
 | |
|     event.code === CODES.TWO &&
 | |
|     event.shiftKey &&
 | |
|     !event.altKey &&
 | |
|     !event[KEYS.CTRL_OR_CMD],
 | |
| });
 | |
| 
 | |
| export const actionZoomToFitSelection = register({
 | |
|   name: "zoomToFitSelection",
 | |
|   label: "helpDialog.zoomToSelection",
 | |
|   icon: zoomAreaIcon,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (elements, appState, _, app) => {
 | |
|     const selectedElements = app.scene.getSelectedElements(appState);
 | |
|     return zoomToFit({
 | |
|       targetElements: selectedElements.length ? selectedElements : elements,
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         userToFollow: null,
 | |
|       },
 | |
|       fitToViewport: true,
 | |
|       canvasOffsets: app.getEditorUIOffsets(),
 | |
|     });
 | |
|   },
 | |
|   // NOTE this action should use shift-2 per figma, alas
 | |
|   keyTest: (event) =>
 | |
|     event.code === CODES.THREE &&
 | |
|     event.shiftKey &&
 | |
|     !event.altKey &&
 | |
|     !event[KEYS.CTRL_OR_CMD],
 | |
| });
 | |
| 
 | |
| export const actionZoomToFit = register({
 | |
|   name: "zoomToFit",
 | |
|   label: "helpDialog.zoomToFit",
 | |
|   icon: zoomAreaIcon,
 | |
|   viewMode: true,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (elements, appState, _, app) =>
 | |
|     zoomToFit({
 | |
|       targetElements: elements,
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         userToFollow: null,
 | |
|       },
 | |
|       fitToViewport: false,
 | |
|       canvasOffsets: app.getEditorUIOffsets(),
 | |
|     }),
 | |
|   keyTest: (event) =>
 | |
|     event.code === CODES.ONE &&
 | |
|     event.shiftKey &&
 | |
|     !event.altKey &&
 | |
|     !event[KEYS.CTRL_OR_CMD],
 | |
| });
 | |
| 
 | |
| export const actionToggleTheme = register({
 | |
|   name: "toggleTheme",
 | |
|   label: (_, appState) => {
 | |
|     return appState.theme === THEME.DARK
 | |
|       ? "buttons.lightMode"
 | |
|       : "buttons.darkMode";
 | |
|   },
 | |
|   keywords: ["toggle", "dark", "light", "mode", "theme"],
 | |
|   icon: (appState) => (appState.theme === THEME.LIGHT ? MoonIcon : SunIcon),
 | |
|   viewMode: true,
 | |
|   trackEvent: { category: "canvas" },
 | |
|   perform: (_, appState, value) => {
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         theme:
 | |
|           value || (appState.theme === THEME.LIGHT ? THEME.DARK : THEME.LIGHT),
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.EVENTUALLY,
 | |
|     };
 | |
|   },
 | |
|   keyTest: (event) => event.altKey && event.shiftKey && event.code === CODES.D,
 | |
|   predicate: (elements, appState, props, app) => {
 | |
|     return !!app.props.UIOptions.canvasActions.toggleTheme;
 | |
|   },
 | |
| });
 | |
| 
 | |
| export const actionToggleEraserTool = register({
 | |
|   name: "toggleEraserTool",
 | |
|   label: "toolBar.eraser",
 | |
|   trackEvent: { category: "toolbar" },
 | |
|   perform: (elements, appState, _, app) => {
 | |
|     let activeTool: AppState["activeTool"];
 | |
| 
 | |
|     if (isEraserActive(appState)) {
 | |
|       activeTool = updateActiveTool(appState, {
 | |
|         ...(appState.activeTool.lastActiveTool || {
 | |
|           type: app.state.preferredSelectionTool.type,
 | |
|         }),
 | |
|         lastActiveToolBeforeEraser: null,
 | |
|       });
 | |
|     } else {
 | |
|       activeTool = updateActiveTool(appState, {
 | |
|         type: "eraser",
 | |
|         lastActiveToolBeforeEraser: appState.activeTool,
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         selectedElementIds: {},
 | |
|         selectedGroupIds: {},
 | |
|         activeEmbeddable: null,
 | |
|         activeTool,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.IMMEDIATELY,
 | |
|     };
 | |
|   },
 | |
|   keyTest: (event) => event.key === KEYS.E,
 | |
| });
 | |
| 
 | |
| export const actionToggleLassoTool = register({
 | |
|   name: "toggleLassoTool",
 | |
|   label: "toolBar.lasso",
 | |
|   icon: LassoIcon,
 | |
|   trackEvent: { category: "toolbar" },
 | |
|   predicate: (elements, appState, props, app) => {
 | |
|     return app.state.preferredSelectionTool.type !== "lasso";
 | |
|   },
 | |
|   perform: (elements, appState, _, app) => {
 | |
|     let activeTool: AppState["activeTool"];
 | |
| 
 | |
|     if (appState.activeTool.type !== "lasso") {
 | |
|       activeTool = updateActiveTool(appState, {
 | |
|         type: "lasso",
 | |
|         fromSelection: false,
 | |
|       });
 | |
|       setCursor(app.interactiveCanvas, CURSOR_TYPE.CROSSHAIR);
 | |
|     } else {
 | |
|       activeTool = updateActiveTool(appState, {
 | |
|         type: "selection",
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         selectedElementIds: {},
 | |
|         selectedGroupIds: {},
 | |
|         activeEmbeddable: null,
 | |
|         activeTool,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.NEVER,
 | |
|     };
 | |
|   },
 | |
| });
 | |
| 
 | |
| export const actionToggleHandTool = register({
 | |
|   name: "toggleHandTool",
 | |
|   label: "toolBar.hand",
 | |
|   trackEvent: { category: "toolbar" },
 | |
|   icon: handIcon,
 | |
|   viewMode: false,
 | |
|   perform: (elements, appState, _, app) => {
 | |
|     let activeTool: AppState["activeTool"];
 | |
| 
 | |
|     if (isHandToolActive(appState)) {
 | |
|       activeTool = updateActiveTool(appState, {
 | |
|         ...(appState.activeTool.lastActiveTool || {
 | |
|           type: "selection",
 | |
|         }),
 | |
|         lastActiveToolBeforeEraser: null,
 | |
|       });
 | |
|     } else {
 | |
|       activeTool = updateActiveTool(appState, {
 | |
|         type: "hand",
 | |
|         lastActiveToolBeforeEraser: appState.activeTool,
 | |
|       });
 | |
|       setCursor(app.interactiveCanvas, CURSOR_TYPE.GRAB);
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|       appState: {
 | |
|         ...appState,
 | |
|         selectedElementIds: {},
 | |
|         selectedGroupIds: {},
 | |
|         activeEmbeddable: null,
 | |
|         activeTool,
 | |
|       },
 | |
|       captureUpdate: CaptureUpdateAction.IMMEDIATELY,
 | |
|     };
 | |
|   },
 | |
|   keyTest: (event) =>
 | |
|     !event.altKey && !event[KEYS.CTRL_OR_CMD] && event.key === KEYS.H,
 | |
| });
 | 
