You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
830 lines
25 KiB
830 lines
25 KiB
import type React from "react"; |
|
import type { |
|
PointerType, |
|
ExcalidrawLinearElement, |
|
NonDeletedExcalidrawElement, |
|
NonDeleted, |
|
TextAlign, |
|
ExcalidrawElement, |
|
GroupId, |
|
ExcalidrawBindableElement, |
|
Arrowhead, |
|
ChartType, |
|
FontFamilyValues, |
|
FileId, |
|
ExcalidrawImageElement, |
|
Theme, |
|
StrokeRoundness, |
|
ExcalidrawEmbeddableElement, |
|
ExcalidrawMagicFrameElement, |
|
ExcalidrawFrameLikeElement, |
|
ExcalidrawElementType, |
|
ExcalidrawIframeLikeElement, |
|
OrderedExcalidrawElement, |
|
ExcalidrawNonSelectionElement, |
|
} from "./element/types"; |
|
import type { Action } from "./actions/types"; |
|
import type { Point as RoughPoint } from "roughjs/bin/geometry"; |
|
import type { LinearElementEditor } from "./element/linearElementEditor"; |
|
import type { SuggestedBinding } from "./element/binding"; |
|
import type { ImportedDataState } from "./data/types"; |
|
import type App from "./components/App"; |
|
import type { throttleRAF } from "./utils"; |
|
import type { Spreadsheet } from "./charts"; |
|
import type { Language } from "./i18n"; |
|
import type { ClipboardData } from "./clipboard"; |
|
import type { isOverScrollBars } from "./scene/scrollbars"; |
|
import type { MaybeTransformHandleType } from "./element/transformHandles"; |
|
import type Library from "./data/library"; |
|
import type { FileSystemHandle } from "./data/filesystem"; |
|
import type { IMAGE_MIME_TYPES, MIME_TYPES } from "./constants"; |
|
import type { ContextMenuItems } from "./components/ContextMenu"; |
|
import type { SnapLine } from "./snapping"; |
|
import type { Merge, MaybePromise, ValueOf } from "./utility-types"; |
|
import type { StoreActionType } from "./store"; |
|
|
|
export type Point = Readonly<RoughPoint>; |
|
|
|
export type SocketId = string & { _brand: "SocketId" }; |
|
|
|
export type Collaborator = Readonly<{ |
|
pointer?: CollaboratorPointer; |
|
button?: "up" | "down"; |
|
selectedElementIds?: AppState["selectedElementIds"]; |
|
username?: string | null; |
|
userState?: UserIdleState; |
|
color?: { |
|
background: string; |
|
stroke: string; |
|
}; |
|
// The url of the collaborator's avatar, defaults to username initials |
|
// if not present |
|
avatarUrl?: string; |
|
// user id. If supplied, we'll filter out duplicates when rendering user avatars. |
|
id?: string; |
|
socketId?: SocketId; |
|
isCurrentUser?: boolean; |
|
isInCall?: boolean; |
|
isSpeaking?: boolean; |
|
isMuted?: boolean; |
|
}>; |
|
|
|
export type CollaboratorPointer = { |
|
x: number; |
|
y: number; |
|
tool: "pointer" | "laser"; |
|
/** |
|
* Whether to render cursor + username. Useful when you only want to render |
|
* laser trail. |
|
* |
|
* @default true |
|
*/ |
|
renderCursor?: boolean; |
|
/** |
|
* Explicit laser color. |
|
* |
|
* @default string collaborator's cursor color |
|
*/ |
|
laserColor?: string; |
|
}; |
|
|
|
export type DataURL = string & { _brand: "DataURL" }; |
|
|
|
export type BinaryFileData = { |
|
mimeType: |
|
| ValueOf<typeof IMAGE_MIME_TYPES> |
|
// future user or unknown file type |
|
| typeof MIME_TYPES.binary; |
|
id: FileId; |
|
dataURL: DataURL; |
|
/** |
|
* Epoch timestamp in milliseconds |
|
*/ |
|
created: number; |
|
/** |
|
* Indicates when the file was last retrieved from storage to be loaded |
|
* onto the scene. We use this flag to determine whether to delete unused |
|
* files from storage. |
|
* |
|
* Epoch timestamp in milliseconds. |
|
*/ |
|
lastRetrieved?: number; |
|
}; |
|
|
|
export type BinaryFileMetadata = Omit<BinaryFileData, "dataURL">; |
|
|
|
export type BinaryFiles = Record<ExcalidrawElement["id"], BinaryFileData>; |
|
|
|
export type ToolType = |
|
| "selection" |
|
| "rectangle" |
|
| "diamond" |
|
| "ellipse" |
|
| "arrow" |
|
| "line" |
|
| "freedraw" |
|
| "text" |
|
| "image" |
|
| "eraser" |
|
| "hand" |
|
| "frame" |
|
| "magicframe" |
|
| "embeddable" |
|
| "laser"; |
|
|
|
export type ElementOrToolType = ExcalidrawElementType | ToolType | "custom"; |
|
|
|
export type ActiveTool = |
|
| { |
|
type: ToolType; |
|
customType: null; |
|
} |
|
| { |
|
type: "custom"; |
|
customType: string; |
|
}; |
|
|
|
export type SidebarName = string; |
|
export type SidebarTabName = string; |
|
|
|
export type UserToFollow = { |
|
socketId: SocketId; |
|
username: string; |
|
}; |
|
|
|
type _CommonCanvasAppState = { |
|
zoom: AppState["zoom"]; |
|
scrollX: AppState["scrollX"]; |
|
scrollY: AppState["scrollY"]; |
|
width: AppState["width"]; |
|
height: AppState["height"]; |
|
viewModeEnabled: AppState["viewModeEnabled"]; |
|
editingGroupId: AppState["editingGroupId"]; // TODO: move to interactive canvas if possible |
|
selectedElementIds: AppState["selectedElementIds"]; // TODO: move to interactive canvas if possible |
|
frameToHighlight: AppState["frameToHighlight"]; // TODO: move to interactive canvas if possible |
|
offsetLeft: AppState["offsetLeft"]; |
|
offsetTop: AppState["offsetTop"]; |
|
theme: AppState["theme"]; |
|
pendingImageElementId: AppState["pendingImageElementId"]; |
|
}; |
|
|
|
export type StaticCanvasAppState = Readonly< |
|
_CommonCanvasAppState & { |
|
shouldCacheIgnoreZoom: AppState["shouldCacheIgnoreZoom"]; |
|
/** null indicates transparent bg */ |
|
viewBackgroundColor: AppState["viewBackgroundColor"] | null; |
|
exportScale: AppState["exportScale"]; |
|
selectedElementsAreBeingDragged: AppState["selectedElementsAreBeingDragged"]; |
|
gridSize: AppState["gridSize"]; |
|
frameRendering: AppState["frameRendering"]; |
|
currentHoveredFontFamily: AppState["currentHoveredFontFamily"]; |
|
} |
|
>; |
|
|
|
export type InteractiveCanvasAppState = Readonly< |
|
_CommonCanvasAppState & { |
|
// renderInteractiveScene |
|
activeEmbeddable: AppState["activeEmbeddable"]; |
|
editingLinearElement: AppState["editingLinearElement"]; |
|
selectionElement: AppState["selectionElement"]; |
|
selectedGroupIds: AppState["selectedGroupIds"]; |
|
selectedLinearElement: AppState["selectedLinearElement"]; |
|
multiElement: AppState["multiElement"]; |
|
isBindingEnabled: AppState["isBindingEnabled"]; |
|
suggestedBindings: AppState["suggestedBindings"]; |
|
isRotating: AppState["isRotating"]; |
|
elementsToHighlight: AppState["elementsToHighlight"]; |
|
// Collaborators |
|
collaborators: AppState["collaborators"]; |
|
// SnapLines |
|
snapLines: AppState["snapLines"]; |
|
zenModeEnabled: AppState["zenModeEnabled"]; |
|
editingElement: AppState["editingElement"]; |
|
} |
|
>; |
|
|
|
export type ObservedAppState = ObservedStandaloneAppState & |
|
ObservedElementsAppState; |
|
|
|
export type ObservedStandaloneAppState = { |
|
name: AppState["name"]; |
|
viewBackgroundColor: AppState["viewBackgroundColor"]; |
|
}; |
|
|
|
export type ObservedElementsAppState = { |
|
editingGroupId: AppState["editingGroupId"]; |
|
selectedElementIds: AppState["selectedElementIds"]; |
|
selectedGroupIds: AppState["selectedGroupIds"]; |
|
// Avoiding storing whole instance, as it could lead into state incosistencies, empty undos/redos and etc. |
|
editingLinearElementId: LinearElementEditor["elementId"] | null; |
|
// Right now it's coupled to `editingLinearElement`, ideally it should not be really needed as we already have selectedElementIds & editingLinearElementId |
|
selectedLinearElementId: LinearElementEditor["elementId"] | null; |
|
}; |
|
|
|
export interface AppState { |
|
contextMenu: { |
|
items: ContextMenuItems; |
|
top: number; |
|
left: number; |
|
} | null; |
|
showWelcomeScreen: boolean; |
|
isLoading: boolean; |
|
errorMessage: React.ReactNode; |
|
activeEmbeddable: { |
|
element: NonDeletedExcalidrawElement; |
|
state: "hover" | "active"; |
|
} | null; |
|
/** |
|
* for a newly created element |
|
* - set on pointer down, updated during pointer move, used on pointer up |
|
*/ |
|
newElement: NonDeleted<ExcalidrawNonSelectionElement> | null; |
|
/** |
|
* for a single element that's being resized |
|
* - set on pointer down when it's selected and the active tool is selection |
|
*/ |
|
resizingElement: NonDeletedExcalidrawElement | null; |
|
/** |
|
* multiElement is for multi-point linear element that's created by clicking as opposed to dragging |
|
* - when set and present, the editor will handle linear element creation logic accordingly |
|
*/ |
|
multiElement: NonDeleted<ExcalidrawLinearElement> | null; |
|
/** |
|
* decoupled from newElement, dragging selection only creates selectionElement |
|
* - set on pointer down, updated during pointer move |
|
*/ |
|
selectionElement: NonDeletedExcalidrawElement | null; |
|
isBindingEnabled: boolean; |
|
startBoundElement: NonDeleted<ExcalidrawBindableElement> | null; |
|
suggestedBindings: SuggestedBinding[]; |
|
frameToHighlight: NonDeleted<ExcalidrawFrameLikeElement> | null; |
|
frameRendering: { |
|
enabled: boolean; |
|
name: boolean; |
|
outline: boolean; |
|
clip: boolean; |
|
}; |
|
editingFrame: string | null; |
|
elementsToHighlight: NonDeleted<ExcalidrawElement>[] | null; |
|
/** |
|
* currently set for: |
|
* - text elements while in wysiwyg |
|
* - newly created linear elements while in line editor (not set for existing |
|
* elements in line editor) |
|
* - and new images while being placed on canvas |
|
*/ |
|
editingElement: NonDeletedExcalidrawElement | null; |
|
editingLinearElement: LinearElementEditor | null; |
|
activeTool: { |
|
/** |
|
* indicates a previous tool we should revert back to if we deselect the |
|
* currently active tool. At the moment applies to `eraser` and `hand` tool. |
|
*/ |
|
lastActiveTool: ActiveTool | null; |
|
locked: boolean; |
|
} & ActiveTool; |
|
penMode: boolean; |
|
penDetected: boolean; |
|
exportBackground: boolean; |
|
exportEmbedScene: boolean; |
|
exportWithDarkMode: boolean; |
|
exportScale: number; |
|
currentItemStrokeColor: string; |
|
currentItemBackgroundColor: string; |
|
currentItemFillStyle: ExcalidrawElement["fillStyle"]; |
|
currentItemStrokeWidth: number; |
|
currentItemStrokeStyle: ExcalidrawElement["strokeStyle"]; |
|
currentItemRoughness: number; |
|
currentItemOpacity: number; |
|
currentItemFontFamily: FontFamilyValues; |
|
currentItemFontSize: number; |
|
currentItemTextAlign: TextAlign; |
|
currentItemStartArrowhead: Arrowhead | null; |
|
currentItemEndArrowhead: Arrowhead | null; |
|
currentHoveredFontFamily: FontFamilyValues | null; |
|
currentItemRoundness: StrokeRoundness; |
|
currentItemArrowType: "sharp" | "round" | "elbow"; |
|
viewBackgroundColor: string; |
|
scrollX: number; |
|
scrollY: number; |
|
cursorButton: "up" | "down"; |
|
scrolledOutside: boolean; |
|
name: string | null; |
|
isResizing: boolean; |
|
isRotating: boolean; |
|
zoom: Zoom; |
|
openMenu: "canvas" | "shape" | null; |
|
openPopup: |
|
| "canvasBackground" |
|
| "elementBackground" |
|
| "elementStroke" |
|
| "fontFamily" |
|
| null; |
|
openSidebar: { name: SidebarName; tab?: SidebarTabName } | null; |
|
openDialog: |
|
| null |
|
| { name: "imageExport" | "help" | "jsonExport" } |
|
| { |
|
name: "settings"; |
|
source: |
|
| "tool" // when magicframe tool is selected |
|
| "generation" // when magicframe generate button is clicked |
|
| "settings"; // when AI settings dialog is explicitly invoked |
|
tab: "text-to-diagram" | "diagram-to-code"; |
|
} |
|
| { name: "ttd"; tab: "text-to-diagram" | "mermaid" } |
|
| { name: "commandPalette" }; |
|
/** |
|
* Reflects user preference for whether the default sidebar should be docked. |
|
* |
|
* NOTE this is only a user preference and does not reflect the actual docked |
|
* state of the sidebar, because the host apps can override this through |
|
* a DefaultSidebar prop, which is not reflected back to the appState. |
|
*/ |
|
defaultSidebarDockedPreference: boolean; |
|
|
|
lastPointerDownWith: PointerType; |
|
selectedElementIds: Readonly<{ [id: string]: true }>; |
|
previousSelectedElementIds: { [id: string]: true }; |
|
selectedElementsAreBeingDragged: boolean; |
|
shouldCacheIgnoreZoom: boolean; |
|
toast: { message: string; closable?: boolean; duration?: number } | null; |
|
zenModeEnabled: boolean; |
|
theme: Theme; |
|
gridSize: number | null; |
|
viewModeEnabled: boolean; |
|
|
|
/** top-most selected groups (i.e. does not include nested groups) */ |
|
selectedGroupIds: { [groupId: string]: boolean }; |
|
/** group being edited when you drill down to its constituent element |
|
(e.g. when you double-click on a group's element) */ |
|
editingGroupId: GroupId | null; |
|
width: number; |
|
height: number; |
|
offsetTop: number; |
|
offsetLeft: number; |
|
|
|
fileHandle: FileSystemHandle | null; |
|
collaborators: Map<SocketId, Collaborator>; |
|
stats: { |
|
open: boolean; |
|
/** bitmap. Use `STATS_PANELS` bit values */ |
|
panels: number; |
|
}; |
|
currentChartType: ChartType; |
|
pasteDialog: |
|
| { |
|
shown: false; |
|
data: null; |
|
} |
|
| { |
|
shown: true; |
|
data: Spreadsheet; |
|
}; |
|
/** imageElement waiting to be placed on canvas */ |
|
pendingImageElementId: ExcalidrawImageElement["id"] | null; |
|
showHyperlinkPopup: false | "info" | "editor"; |
|
selectedLinearElement: LinearElementEditor | null; |
|
snapLines: readonly SnapLine[]; |
|
originSnapOffset: { |
|
x: number; |
|
y: number; |
|
} | null; |
|
objectsSnapModeEnabled: boolean; |
|
/** the user's socket id & username who is being followed on the canvas */ |
|
userToFollow: UserToFollow | null; |
|
/** the socket ids of the users following the current user */ |
|
followedBy: Set<SocketId>; |
|
} |
|
|
|
export type UIAppState = Omit< |
|
AppState, |
|
| "suggestedBindings" |
|
| "startBoundElement" |
|
| "cursorButton" |
|
| "scrollX" |
|
| "scrollY" |
|
>; |
|
|
|
export type NormalizedZoomValue = number & { _brand: "normalizedZoom" }; |
|
|
|
export type Zoom = Readonly<{ |
|
value: NormalizedZoomValue; |
|
}>; |
|
|
|
export type PointerCoords = Readonly<{ |
|
x: number; |
|
y: number; |
|
}>; |
|
|
|
export type Gesture = { |
|
pointers: Map<number, PointerCoords>; |
|
lastCenter: { x: number; y: number } | null; |
|
initialDistance: number | null; |
|
initialScale: number | null; |
|
}; |
|
|
|
export declare class GestureEvent extends UIEvent { |
|
readonly rotation: number; |
|
readonly scale: number; |
|
} |
|
|
|
// libraries |
|
// ----------------------------------------------------------------------------- |
|
/** @deprecated legacy: do not use outside of migration paths */ |
|
export type LibraryItem_v1 = readonly NonDeleted<ExcalidrawElement>[]; |
|
/** @deprecated legacy: do not use outside of migration paths */ |
|
type LibraryItems_v1 = readonly LibraryItem_v1[]; |
|
|
|
/** v2 library item */ |
|
export type LibraryItem = { |
|
id: string; |
|
status: "published" | "unpublished"; |
|
elements: readonly NonDeleted<ExcalidrawElement>[]; |
|
/** timestamp in epoch (ms) */ |
|
created: number; |
|
name?: string; |
|
error?: string; |
|
}; |
|
export type LibraryItems = readonly LibraryItem[]; |
|
export type LibraryItems_anyVersion = LibraryItems | LibraryItems_v1; |
|
|
|
export type LibraryItemsSource = |
|
| (( |
|
currentLibraryItems: LibraryItems, |
|
) => MaybePromise<LibraryItems_anyVersion | Blob>) |
|
| MaybePromise<LibraryItems_anyVersion | Blob>; |
|
// ----------------------------------------------------------------------------- |
|
|
|
export type ExcalidrawInitialDataState = Merge< |
|
ImportedDataState, |
|
{ |
|
libraryItems?: MaybePromise<Required<ImportedDataState>["libraryItems"]>; |
|
} |
|
>; |
|
|
|
export type OnUserFollowedPayload = { |
|
userToFollow: UserToFollow; |
|
action: "FOLLOW" | "UNFOLLOW"; |
|
}; |
|
|
|
export interface ExcalidrawProps { |
|
onChange?: ( |
|
elements: readonly OrderedExcalidrawElement[], |
|
appState: AppState, |
|
files: BinaryFiles, |
|
) => void; |
|
initialData?: |
|
| (() => MaybePromise<ExcalidrawInitialDataState | null>) |
|
| MaybePromise<ExcalidrawInitialDataState | null>; |
|
excalidrawAPI?: (api: ExcalidrawImperativeAPI) => void; |
|
isCollaborating?: boolean; |
|
onPointerUpdate?: (payload: { |
|
pointer: { x: number; y: number; tool: "pointer" | "laser" }; |
|
button: "down" | "up"; |
|
pointersMap: Gesture["pointers"]; |
|
}) => void; |
|
onPaste?: ( |
|
data: ClipboardData, |
|
event: ClipboardEvent | null, |
|
) => Promise<boolean> | boolean; |
|
renderTopRightUI?: ( |
|
isMobile: boolean, |
|
appState: UIAppState, |
|
) => JSX.Element | null; |
|
langCode?: Language["code"]; |
|
viewModeEnabled?: boolean; |
|
zenModeEnabled?: boolean; |
|
gridModeEnabled?: boolean; |
|
objectsSnapModeEnabled?: boolean; |
|
libraryReturnUrl?: string; |
|
theme?: Theme; |
|
// @TODO come with better API before v0.18.0 |
|
name?: string; |
|
renderCustomStats?: ( |
|
elements: readonly NonDeletedExcalidrawElement[], |
|
appState: UIAppState, |
|
) => JSX.Element; |
|
UIOptions?: Partial<UIOptions>; |
|
detectScroll?: boolean; |
|
handleKeyboardGlobally?: boolean; |
|
onLibraryChange?: (libraryItems: LibraryItems) => void | Promise<any>; |
|
autoFocus?: boolean; |
|
generateIdForFile?: (file: File) => string | Promise<string>; |
|
onLinkOpen?: ( |
|
element: NonDeletedExcalidrawElement, |
|
event: CustomEvent<{ |
|
nativeEvent: MouseEvent | React.PointerEvent<HTMLCanvasElement>; |
|
}>, |
|
) => void; |
|
onPointerDown?: ( |
|
activeTool: AppState["activeTool"], |
|
pointerDownState: PointerDownState, |
|
) => void; |
|
onPointerUp?: ( |
|
activeTool: AppState["activeTool"], |
|
pointerDownState: PointerDownState, |
|
) => void; |
|
onScrollChange?: (scrollX: number, scrollY: number, zoom: Zoom) => void; |
|
onUserFollow?: (payload: OnUserFollowedPayload) => void; |
|
children?: React.ReactNode; |
|
validateEmbeddable?: |
|
| boolean |
|
| string[] |
|
| RegExp |
|
| RegExp[] |
|
| ((link: string) => boolean | undefined); |
|
renderEmbeddable?: ( |
|
element: NonDeleted<ExcalidrawEmbeddableElement>, |
|
appState: AppState, |
|
) => JSX.Element | null; |
|
aiEnabled?: boolean; |
|
showDeprecatedFonts?: boolean; |
|
} |
|
|
|
export type SceneData = { |
|
elements?: ImportedDataState["elements"]; |
|
appState?: ImportedDataState["appState"]; |
|
collaborators?: Map<SocketId, Collaborator>; |
|
storeAction?: StoreActionType; |
|
}; |
|
|
|
export enum UserIdleState { |
|
ACTIVE = "active", |
|
AWAY = "away", |
|
IDLE = "idle", |
|
} |
|
|
|
export type ExportOpts = { |
|
saveFileToDisk?: boolean; |
|
onExportToBackend?: ( |
|
exportedElements: readonly NonDeletedExcalidrawElement[], |
|
appState: UIAppState, |
|
files: BinaryFiles, |
|
) => void; |
|
renderCustomUI?: ( |
|
exportedElements: readonly NonDeletedExcalidrawElement[], |
|
appState: UIAppState, |
|
files: BinaryFiles, |
|
canvas: HTMLCanvasElement, |
|
) => JSX.Element; |
|
}; |
|
|
|
// NOTE at the moment, if action name corresponds to canvasAction prop, its |
|
// truthiness value will determine whether the action is rendered or not |
|
// (see manager renderAction). We also override canvasAction values in |
|
// Excalidraw package index.tsx. |
|
export type CanvasActions = Partial<{ |
|
changeViewBackgroundColor: boolean; |
|
clearCanvas: boolean; |
|
export: false | ExportOpts; |
|
loadScene: boolean; |
|
saveToActiveFile: boolean; |
|
toggleTheme: boolean | null; |
|
saveAsImage: boolean; |
|
}>; |
|
|
|
export type UIOptions = Partial<{ |
|
dockedSidebarBreakpoint: number; |
|
canvasActions: CanvasActions; |
|
tools: { |
|
image: boolean; |
|
}; |
|
/** @deprecated does nothing. Will be removed in 0.15 */ |
|
welcomeScreen?: boolean; |
|
}>; |
|
|
|
export type AppProps = Merge< |
|
ExcalidrawProps, |
|
{ |
|
UIOptions: Merge< |
|
UIOptions, |
|
{ |
|
canvasActions: Required<CanvasActions> & { export: ExportOpts }; |
|
} |
|
>; |
|
detectScroll: boolean; |
|
handleKeyboardGlobally: boolean; |
|
isCollaborating: boolean; |
|
children?: React.ReactNode; |
|
aiEnabled: boolean; |
|
} |
|
>; |
|
|
|
/** A subset of App class properties that we need to use elsewhere |
|
* in the app, eg Manager. Factored out into a separate type to keep DRY. */ |
|
export type AppClassProperties = { |
|
props: AppProps; |
|
interactiveCanvas: HTMLCanvasElement | null; |
|
/** static canvas */ |
|
canvas: HTMLCanvasElement; |
|
focusContainer(): void; |
|
library: Library; |
|
imageCache: Map< |
|
FileId, |
|
{ |
|
image: HTMLImageElement | Promise<HTMLImageElement>; |
|
mimeType: ValueOf<typeof IMAGE_MIME_TYPES>; |
|
} |
|
>; |
|
files: BinaryFiles; |
|
device: App["device"]; |
|
scene: App["scene"]; |
|
syncActionResult: App["syncActionResult"]; |
|
fonts: App["fonts"]; |
|
pasteFromClipboard: App["pasteFromClipboard"]; |
|
id: App["id"]; |
|
onInsertElements: App["onInsertElements"]; |
|
onExportImage: App["onExportImage"]; |
|
lastViewportPosition: App["lastViewportPosition"]; |
|
scrollToContent: App["scrollToContent"]; |
|
addFiles: App["addFiles"]; |
|
addElementsFromPasteOrLibrary: App["addElementsFromPasteOrLibrary"]; |
|
togglePenMode: App["togglePenMode"]; |
|
toggleLock: App["toggleLock"]; |
|
setActiveTool: App["setActiveTool"]; |
|
setOpenDialog: App["setOpenDialog"]; |
|
insertEmbeddableElement: App["insertEmbeddableElement"]; |
|
onMagicframeToolSelect: App["onMagicframeToolSelect"]; |
|
getName: App["getName"]; |
|
dismissLinearEditor: App["dismissLinearEditor"]; |
|
}; |
|
|
|
export type PointerDownState = Readonly<{ |
|
// The first position at which pointerDown happened |
|
origin: Readonly<{ x: number; y: number }>; |
|
// Same as "origin" but snapped to the grid, if grid is on |
|
originInGrid: Readonly<{ x: number; y: number }>; |
|
// Scrollbar checks |
|
scrollbars: ReturnType<typeof isOverScrollBars>; |
|
// The previous pointer position |
|
lastCoords: { x: number; y: number }; |
|
// map of original elements data |
|
originalElements: Map<string, NonDeleted<ExcalidrawElement>>; |
|
resize: { |
|
// Handle when resizing, might change during the pointer interaction |
|
handleType: MaybeTransformHandleType; |
|
// This is determined on the initial pointer down event |
|
isResizing: boolean; |
|
// This is determined on the initial pointer down event |
|
offset: { x: number; y: number }; |
|
// This is determined on the initial pointer down event |
|
arrowDirection: "origin" | "end"; |
|
// This is a center point of selected elements determined on the initial pointer down event (for rotation only) |
|
center: { x: number; y: number }; |
|
}; |
|
hit: { |
|
// The element the pointer is "hitting", is determined on the initial |
|
// pointer down event |
|
element: NonDeleted<ExcalidrawElement> | null; |
|
// The elements the pointer is "hitting", is determined on the initial |
|
// pointer down event |
|
allHitElements: NonDeleted<ExcalidrawElement>[]; |
|
// This is determined on the initial pointer down event |
|
wasAddedToSelection: boolean; |
|
// Whether selected element(s) were duplicated, might change during the |
|
// pointer interaction |
|
hasBeenDuplicated: boolean; |
|
hasHitCommonBoundingBoxOfSelectedElements: boolean; |
|
}; |
|
withCmdOrCtrl: boolean; |
|
drag: { |
|
// Might change during the pointer interaction |
|
hasOccurred: boolean; |
|
// Might change during the pointer interaction |
|
offset: { x: number; y: number } | null; |
|
}; |
|
// We need to have these in the state so that we can unsubscribe them |
|
eventListeners: { |
|
// It's defined on the initial pointer down event |
|
onMove: null | ReturnType<typeof throttleRAF>; |
|
// It's defined on the initial pointer down event |
|
onUp: null | ((event: PointerEvent) => void); |
|
// It's defined on the initial pointer down event |
|
onKeyDown: null | ((event: KeyboardEvent) => void); |
|
// It's defined on the initial pointer down event |
|
onKeyUp: null | ((event: KeyboardEvent) => void); |
|
}; |
|
boxSelection: { |
|
hasOccurred: boolean; |
|
}; |
|
}>; |
|
|
|
export type UnsubscribeCallback = () => void; |
|
|
|
export interface ExcalidrawImperativeAPI { |
|
updateScene: InstanceType<typeof App>["updateScene"]; |
|
updateLibrary: InstanceType<typeof Library>["updateLibrary"]; |
|
resetScene: InstanceType<typeof App>["resetScene"]; |
|
getSceneElementsIncludingDeleted: InstanceType< |
|
typeof App |
|
>["getSceneElementsIncludingDeleted"]; |
|
history: { |
|
clear: InstanceType<typeof App>["resetHistory"]; |
|
}; |
|
getSceneElements: InstanceType<typeof App>["getSceneElements"]; |
|
getAppState: () => InstanceType<typeof App>["state"]; |
|
getFiles: () => InstanceType<typeof App>["files"]; |
|
getName: InstanceType<typeof App>["getName"]; |
|
scrollToContent: InstanceType<typeof App>["scrollToContent"]; |
|
registerAction: (action: Action) => void; |
|
refresh: InstanceType<typeof App>["refresh"]; |
|
setToast: InstanceType<typeof App>["setToast"]; |
|
addFiles: (data: BinaryFileData[]) => void; |
|
id: string; |
|
setActiveTool: InstanceType<typeof App>["setActiveTool"]; |
|
setCursor: InstanceType<typeof App>["setCursor"]; |
|
resetCursor: InstanceType<typeof App>["resetCursor"]; |
|
toggleSidebar: InstanceType<typeof App>["toggleSidebar"]; |
|
/** |
|
* Disables rendering of frames (including element clipping), but currently |
|
* the frames are still interactive in edit mode. As such, this API should be |
|
* used in conjunction with view mode (props.viewModeEnabled). |
|
*/ |
|
updateFrameRendering: InstanceType<typeof App>["updateFrameRendering"]; |
|
onChange: ( |
|
callback: ( |
|
elements: readonly ExcalidrawElement[], |
|
appState: AppState, |
|
files: BinaryFiles, |
|
) => void, |
|
) => UnsubscribeCallback; |
|
onPointerDown: ( |
|
callback: ( |
|
activeTool: AppState["activeTool"], |
|
pointerDownState: PointerDownState, |
|
event: React.PointerEvent<HTMLElement>, |
|
) => void, |
|
) => UnsubscribeCallback; |
|
onPointerUp: ( |
|
callback: ( |
|
activeTool: AppState["activeTool"], |
|
pointerDownState: PointerDownState, |
|
event: PointerEvent, |
|
) => void, |
|
) => UnsubscribeCallback; |
|
onScrollChange: ( |
|
callback: (scrollX: number, scrollY: number, zoom: Zoom) => void, |
|
) => UnsubscribeCallback; |
|
onUserFollow: ( |
|
callback: (payload: OnUserFollowedPayload) => void, |
|
) => UnsubscribeCallback; |
|
} |
|
|
|
export type Device = Readonly<{ |
|
viewport: { |
|
isMobile: boolean; |
|
isLandscape: boolean; |
|
}; |
|
editor: { |
|
isMobile: boolean; |
|
canFitSidebar: boolean; |
|
}; |
|
isTouchScreen: boolean; |
|
}>; |
|
|
|
export type FrameNameBounds = { |
|
x: number; |
|
y: number; |
|
width: number; |
|
height: number; |
|
angle: number; |
|
}; |
|
|
|
export type FrameNameBoundsCache = { |
|
get: ( |
|
frameElement: ExcalidrawFrameLikeElement | ExcalidrawMagicFrameElement, |
|
) => FrameNameBounds | null; |
|
_cache: Map< |
|
string, |
|
FrameNameBounds & { |
|
zoom: AppState["zoom"]["value"]; |
|
versionNonce: ExcalidrawFrameLikeElement["versionNonce"]; |
|
} |
|
>; |
|
}; |
|
|
|
export type KeyboardModifiersObject = { |
|
ctrlKey: boolean; |
|
shiftKey: boolean; |
|
altKey: boolean; |
|
metaKey: boolean; |
|
}; |
|
|
|
export type Primitive = |
|
| number |
|
| string |
|
| boolean |
|
| bigint |
|
| symbol |
|
| null |
|
| undefined; |
|
|
|
export type JSONValue = string | number | boolean | null | object; |
|
|
|
export type EmbedsValidationStatus = Map< |
|
ExcalidrawIframeLikeElement["id"], |
|
boolean |
|
>; |
|
|
|
export type ElementsPendingErasure = Set<ExcalidrawElement["id"]>;
|
|
|