123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659 |
- import React, {
- useRef,
- useState,
- RefObject,
- useEffect,
- useCallback,
- } from "react";
- import { showSelectedShapeActions } from "../element";
- import { calculateScrollCenter, getSelectedElements } from "../scene";
- import { exportCanvas } from "../data";
- import { AppState, LibraryItems, LibraryItem } from "../types";
- import { NonDeletedExcalidrawElement } from "../element/types";
- import { ActionManager } from "../actions/manager";
- import { Island } from "./Island";
- import Stack from "./Stack";
- import { FixedSideContainer } from "./FixedSideContainer";
- import { UserList } from "./UserList";
- import { LockIcon } from "./LockIcon";
- import { ExportDialog, ExportCB } from "./ExportDialog";
- import { LanguageList } from "./LanguageList";
- import { t, languages, setLanguage } from "../i18n";
- import { HintViewer } from "./HintViewer";
- import useIsMobile from "../is-mobile";
- import { ExportType } from "../scene/types";
- import { MobileMenu } from "./MobileMenu";
- import { ZoomActions, SelectedShapeActions, ShapesSwitcher } from "./Actions";
- import { Section } from "./Section";
- import { RoomDialog } from "./RoomDialog";
- import { ErrorDialog } from "./ErrorDialog";
- import { ShortcutsDialog } from "./ShortcutsDialog";
- import { LoadingMessage } from "./LoadingMessage";
- import { CLASSES } from "../constants";
- import { shield, exportFile, load } from "./icons";
- import { GitHubCorner } from "./GitHubCorner";
- import { Tooltip } from "./Tooltip";
- import "./LayerUI.scss";
- import { LibraryUnit } from "./LibraryUnit";
- import { ToolButton } from "./ToolButton";
- import { saveLibraryAsJSON, importLibraryFromJSON } from "../data/json";
- import { muteFSAbortError } from "../utils";
- import { BackgroundPickerAndDarkModeToggle } from "./BackgroundPickerAndDarkModeToggle";
- import clsx from "clsx";
- import { Library } from "../data/library";
- import { EVENT_EXIT, trackEvent } from "../analytics";
- interface LayerUIProps {
- actionManager: ActionManager;
- appState: AppState;
- canvas: HTMLCanvasElement | null;
- setAppState: React.Component<any, AppState>["setState"];
- elements: readonly NonDeletedExcalidrawElement[];
- onRoomCreate: () => void;
- onUsernameChange: (username: string) => void;
- onRoomDestroy: () => void;
- onLockToggle: () => void;
- onInsertShape: (elements: LibraryItem) => void;
- zenModeEnabled: boolean;
- toggleZenMode: () => void;
- lng: string;
- }
- const useOnClickOutside = (
- ref: RefObject<HTMLElement>,
- cb: (event: MouseEvent) => void,
- ) => {
- useEffect(() => {
- const listener = (event: MouseEvent) => {
- if (!ref.current) {
- return;
- }
- if (
- event.target instanceof Element &&
- (ref.current.contains(event.target) ||
- !document.body.contains(event.target))
- ) {
- return;
- }
- cb(event);
- };
- document.addEventListener("pointerdown", listener, false);
- return () => {
- document.removeEventListener("pointerdown", listener);
- };
- }, [ref, cb]);
- };
- const LibraryMenuItems = ({
- library,
- onRemoveFromLibrary,
- onAddToLibrary,
- onInsertShape,
- pendingElements,
- setAppState,
- }: {
- library: LibraryItems;
- pendingElements: LibraryItem;
- onRemoveFromLibrary: (index: number) => void;
- onInsertShape: (elements: LibraryItem) => void;
- onAddToLibrary: (elements: LibraryItem) => void;
- setAppState: React.Component<any, AppState>["setState"];
- }) => {
- const isMobile = useIsMobile();
- const numCells = library.length + (pendingElements.length > 0 ? 1 : 0);
- const CELLS_PER_ROW = isMobile ? 4 : 6;
- const numRows = Math.max(1, Math.ceil(numCells / CELLS_PER_ROW));
- const rows = [];
- let addedPendingElements = false;
- rows.push(
- <Stack.Row
- align="center"
- gap={1}
- key={"actions"}
- style={{ padding: "2px 0" }}
- >
- <ToolButton
- key="import"
- type="button"
- title={t("buttons.load")}
- aria-label={t("buttons.load")}
- icon={load}
- onClick={() => {
- importLibraryFromJSON()
- .then(() => {
- // Maybe we should close and open the menu so that the items get updated.
- // But for now we just close the menu.
- setAppState({ isLibraryOpen: false });
- })
- .catch(muteFSAbortError)
- .catch((error) => {
- setAppState({ errorMessage: error.message });
- });
- }}
- />
- <ToolButton
- key="export"
- type="button"
- title={t("buttons.export")}
- aria-label={t("buttons.export")}
- icon={exportFile}
- onClick={() => {
- saveLibraryAsJSON()
- .catch(muteFSAbortError)
- .catch((error) => {
- setAppState({ errorMessage: error.message });
- });
- }}
- />
- </Stack.Row>,
- );
- for (let row = 0; row < numRows; row++) {
- const i = CELLS_PER_ROW * row;
- const children = [];
- for (let j = 0; j < CELLS_PER_ROW; j++) {
- const shouldAddPendingElements: boolean =
- pendingElements.length > 0 &&
- !addedPendingElements &&
- i + j >= library.length;
- addedPendingElements = addedPendingElements || shouldAddPendingElements;
- children.push(
- <Stack.Col key={j}>
- <LibraryUnit
- elements={library[i + j]}
- pendingElements={
- shouldAddPendingElements ? pendingElements : undefined
- }
- onRemoveFromLibrary={onRemoveFromLibrary.bind(null, i + j)}
- onClick={
- shouldAddPendingElements
- ? onAddToLibrary.bind(null, pendingElements)
- : onInsertShape.bind(null, library[i + j])
- }
- />
- </Stack.Col>,
- );
- }
- rows.push(
- <Stack.Row align="center" gap={1} key={row}>
- {children}
- </Stack.Row>,
- );
- }
- return (
- <Stack.Col align="center" gap={1} className="layer-ui__library-items">
- {rows}
- </Stack.Col>
- );
- };
- const LibraryMenu = ({
- onClickOutside,
- onInsertShape,
- pendingElements,
- onAddToLibrary,
- setAppState,
- }: {
- pendingElements: LibraryItem;
- onClickOutside: (event: MouseEvent) => void;
- onInsertShape: (elements: LibraryItem) => void;
- onAddToLibrary: () => void;
- setAppState: React.Component<any, AppState>["setState"];
- }) => {
- const ref = useRef<HTMLDivElement | null>(null);
- useOnClickOutside(ref, (event) => {
- // If click on the library icon, do nothing.
- if ((event.target as Element).closest(".ToolIcon_type_button__library")) {
- return;
- }
- onClickOutside(event);
- });
- const [libraryItems, setLibraryItems] = useState<LibraryItems>([]);
- const [loadingState, setIsLoading] = useState<
- "preloading" | "loading" | "ready"
- >("preloading");
- const loadingTimerRef = useRef<NodeJS.Timeout | null>(null);
- useEffect(() => {
- Promise.race([
- new Promise((resolve) => {
- loadingTimerRef.current = setTimeout(() => {
- resolve("loading");
- }, 100);
- }),
- Library.loadLibrary().then((items) => {
- setLibraryItems(items);
- setIsLoading("ready");
- }),
- ]).then((data) => {
- if (data === "loading") {
- setIsLoading("loading");
- }
- });
- return () => {
- clearTimeout(loadingTimerRef.current!);
- };
- }, []);
- const removeFromLibrary = useCallback(async (indexToRemove) => {
- const items = await Library.loadLibrary();
- const nextItems = items.filter((_, index) => index !== indexToRemove);
- Library.saveLibrary(nextItems);
- setLibraryItems(nextItems);
- }, []);
- const addToLibrary = useCallback(
- async (elements: LibraryItem) => {
- const items = await Library.loadLibrary();
- const nextItems = [...items, elements];
- onAddToLibrary();
- Library.saveLibrary(nextItems);
- setLibraryItems(nextItems);
- },
- [onAddToLibrary],
- );
- return loadingState === "preloading" ? null : (
- <Island padding={1} ref={ref} className="layer-ui__library">
- {loadingState === "loading" ? (
- <div className="layer-ui__library-message">
- {t("labels.libraryLoadingMessage")}
- </div>
- ) : (
- <LibraryMenuItems
- library={libraryItems}
- onRemoveFromLibrary={removeFromLibrary}
- onAddToLibrary={addToLibrary}
- onInsertShape={onInsertShape}
- pendingElements={pendingElements}
- setAppState={setAppState}
- />
- )}
- </Island>
- );
- };
- const LayerUI = ({
- actionManager,
- appState,
- setAppState,
- canvas,
- elements,
- onRoomCreate,
- onUsernameChange,
- onRoomDestroy,
- onLockToggle,
- onInsertShape,
- zenModeEnabled,
- toggleZenMode,
- }: LayerUIProps) => {
- const isMobile = useIsMobile();
- // TODO: Extend tooltip component and use here.
- const renderEncryptedIcon = () => (
- <a
- className={clsx("encrypted-icon tooltip zen-mode-visibility", {
- "zen-mode-visibility--hidden": zenModeEnabled,
- })}
- href="https://blog.excalidraw.com/end-to-end-encryption/"
- target="_blank"
- rel="noopener noreferrer"
- onClick={() => {
- trackEvent(EVENT_EXIT, "e2ee shield");
- }}
- >
- <span className="tooltip-text" dir="auto">
- {t("encrypted.tooltip")}
- </span>
- {shield}
- </a>
- );
- const renderExportDialog = () => {
- const createExporter = (type: ExportType): ExportCB => async (
- exportedElements,
- scale,
- ) => {
- if (canvas) {
- await exportCanvas(type, exportedElements, appState, canvas, {
- exportBackground: appState.exportBackground,
- name: appState.name,
- viewBackgroundColor: appState.viewBackgroundColor,
- scale,
- shouldAddWatermark: appState.shouldAddWatermark,
- })
- .catch(muteFSAbortError)
- .catch((error) => {
- console.error(error);
- setAppState({ errorMessage: error.message });
- });
- }
- };
- return (
- <ExportDialog
- elements={elements}
- appState={appState}
- actionManager={actionManager}
- onExportToPng={createExporter("png")}
- onExportToSvg={createExporter("svg")}
- onExportToClipboard={createExporter("clipboard")}
- onExportToBackend={async (exportedElements) => {
- if (canvas) {
- try {
- await exportCanvas(
- "backend",
- exportedElements,
- {
- ...appState,
- selectedElementIds: {},
- },
- canvas,
- appState,
- );
- } catch (error) {
- if (error.name !== "AbortError") {
- const { width, height } = canvas;
- console.error(error, { width, height });
- setAppState({ errorMessage: error.message });
- }
- }
- }
- }}
- />
- );
- };
- const renderCanvasActions = () => (
- <Section
- heading="canvasActions"
- className={clsx("zen-mode-transition", {
- "transition-left": zenModeEnabled,
- })}
- >
- {/* the zIndex ensures this menu has higher stacking order,
- see https://github.com/excalidraw/excalidraw/pull/1445 */}
- <Island padding={2} style={{ zIndex: 1 }}>
- <Stack.Col gap={4}>
- <Stack.Row gap={1} justifyContent="space-between">
- {actionManager.renderAction("loadScene")}
- {actionManager.renderAction("saveScene")}
- {actionManager.renderAction("saveAsScene")}
- {renderExportDialog()}
- {actionManager.renderAction("clearCanvas")}
- <RoomDialog
- isCollaborating={appState.isCollaborating}
- collaboratorCount={appState.collaborators.size}
- username={appState.username}
- onUsernameChange={onUsernameChange}
- onRoomCreate={onRoomCreate}
- onRoomDestroy={onRoomDestroy}
- setErrorMessage={(message: string) =>
- setAppState({ errorMessage: message })
- }
- />
- </Stack.Row>
- <BackgroundPickerAndDarkModeToggle
- actionManager={actionManager}
- appState={appState}
- setAppState={setAppState}
- />
- </Stack.Col>
- </Island>
- </Section>
- );
- const renderSelectedShapeActions = () => (
- <Section
- heading="selectedShapeActions"
- className={clsx("zen-mode-transition", {
- "transition-left": zenModeEnabled,
- })}
- >
- <Island className={CLASSES.SHAPE_ACTIONS_MENU} padding={2}>
- <SelectedShapeActions
- appState={appState}
- elements={elements}
- renderAction={actionManager.renderAction}
- elementType={appState.elementType}
- />
- </Island>
- </Section>
- );
- const closeLibrary = useCallback(
- (event) => {
- setAppState({ isLibraryOpen: false });
- },
- [setAppState],
- );
- const deselectItems = useCallback(() => {
- setAppState({
- selectedElementIds: {},
- selectedGroupIds: {},
- });
- }, [setAppState]);
- const libraryMenu = appState.isLibraryOpen ? (
- <LibraryMenu
- pendingElements={getSelectedElements(elements, appState)}
- onClickOutside={closeLibrary}
- onInsertShape={onInsertShape}
- onAddToLibrary={deselectItems}
- setAppState={setAppState}
- />
- ) : null;
- const renderFixedSideContainer = () => {
- const shouldRenderSelectedShapeActions = showSelectedShapeActions(
- appState,
- elements,
- );
- return (
- <FixedSideContainer side="top">
- <div className="App-menu App-menu_top">
- <Stack.Col
- gap={4}
- className={clsx({ "disable-pointerEvents": zenModeEnabled })}
- >
- {renderCanvasActions()}
- {shouldRenderSelectedShapeActions && renderSelectedShapeActions()}
- </Stack.Col>
- <Section heading="shapes">
- {(heading) => (
- <Stack.Col gap={4} align="start">
- <Stack.Row gap={1}>
- <Island
- padding={1}
- className={clsx({ "zen-mode": zenModeEnabled })}
- >
- <HintViewer appState={appState} elements={elements} />
- {heading}
- <Stack.Row gap={1}>
- <ShapesSwitcher
- elementType={appState.elementType}
- setAppState={setAppState}
- isLibraryOpen={appState.isLibraryOpen}
- />
- </Stack.Row>
- </Island>
- <LockIcon
- zenModeEnabled={zenModeEnabled}
- checked={appState.elementLocked}
- onChange={onLockToggle}
- title={t("toolBar.lock")}
- />
- </Stack.Row>
- {libraryMenu}
- </Stack.Col>
- )}
- </Section>
- <UserList
- className={clsx("zen-mode-transition", {
- "transition-right": zenModeEnabled,
- })}
- >
- {Array.from(appState.collaborators)
- // Collaborator is either not initialized or is actually the current user.
- .filter(([_, client]) => Object.keys(client).length !== 0)
- .map(([clientId, client]) => (
- <Tooltip
- label={client.username || "Unknown user"}
- key={clientId}
- >
- {actionManager.renderAction("goToCollaborator", clientId)}
- </Tooltip>
- ))}
- </UserList>
- </div>
- </FixedSideContainer>
- );
- };
- const renderBottomAppMenu = () => {
- return (
- <div
- className={clsx("App-menu App-menu_bottom zen-mode-transition", {
- "App-menu_bottom--transition-left": zenModeEnabled,
- })}
- >
- <Stack.Col gap={2}>
- <Section heading="canvasActions">
- <Island padding={1}>
- <ZoomActions
- renderAction={actionManager.renderAction}
- zoom={appState.zoom}
- />
- </Island>
- {renderEncryptedIcon()}
- </Section>
- </Stack.Col>
- </div>
- );
- };
- const renderFooter = () => (
- <footer role="contentinfo" className="layer-ui__wrapper__footer">
- <div
- className={clsx("zen-mode-transition", {
- "transition-right disable-pointerEvents": zenModeEnabled,
- })}
- >
- <LanguageList
- onChange={async (lng) => {
- await setLanguage(lng);
- setAppState({});
- }}
- languages={languages}
- floating
- />
- {actionManager.renderAction("toggleShortcuts")}
- </div>
- <button
- className={clsx("disable-zen-mode", {
- "disable-zen-mode--visible": zenModeEnabled,
- })}
- onClick={toggleZenMode}
- >
- {t("buttons.exitZenMode")}
- </button>
- {appState.scrolledOutside && (
- <button
- className="scroll-back-to-content"
- onClick={() => {
- setAppState({
- ...calculateScrollCenter(elements, appState, canvas),
- });
- }}
- >
- {t("buttons.scrollBackToContent")}
- </button>
- )}
- </footer>
- );
- return isMobile ? (
- <MobileMenu
- appState={appState}
- elements={elements}
- actionManager={actionManager}
- libraryMenu={libraryMenu}
- exportButton={renderExportDialog()}
- setAppState={setAppState}
- onUsernameChange={onUsernameChange}
- onRoomCreate={onRoomCreate}
- onRoomDestroy={onRoomDestroy}
- onLockToggle={onLockToggle}
- canvas={canvas}
- />
- ) : (
- <div className="layer-ui__wrapper">
- {appState.isLoading && <LoadingMessage />}
- {appState.errorMessage && (
- <ErrorDialog
- message={appState.errorMessage}
- onClose={() => setAppState({ errorMessage: null })}
- />
- )}
- {appState.showShortcutsDialog && (
- <ShortcutsDialog
- onClose={() => setAppState({ showShortcutsDialog: false })}
- />
- )}
- {renderFixedSideContainer()}
- {renderBottomAppMenu()}
- {
- <aside
- className={clsx(
- "layer-ui__wrapper__github-corner zen-mode-transition",
- {
- "transition-right": zenModeEnabled,
- },
- )}
- >
- <GitHubCorner appearance={appState.appearance} />
- </aside>
- }
- {renderFooter()}
- </div>
- );
- };
- const areEqual = (prev: LayerUIProps, next: LayerUIProps) => {
- const getNecessaryObj = (appState: AppState): Partial<AppState> => {
- const {
- cursorX,
- cursorY,
- suggestedBindings,
- startBoundElement: boundElement,
- ...ret
- } = appState;
- return ret;
- };
- const prevAppState = getNecessaryObj(prev.appState);
- const nextAppState = getNecessaryObj(next.appState);
- const keys = Object.keys(prevAppState) as (keyof Partial<AppState>)[];
- return (
- prev.lng === next.lng &&
- prev.elements === next.elements &&
- keys.every((key) => prevAppState[key] === nextAppState[key])
- );
- };
- export default React.memo(LayerUI, areEqual);
|