123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- import LanguageDetector from "i18next-browser-languagedetector";
- import React, {
- useCallback,
- useContext,
- useEffect,
- useRef,
- useState,
- } from "react";
- import { trackEvent } from "../analytics";
- import { getDefaultAppState } from "../appState";
- import { ExcalidrawImperativeAPI } from "../components/App";
- import { ErrorDialog } from "../components/ErrorDialog";
- import { TopErrorBoundary } from "../components/TopErrorBoundary";
- import {
- APP_NAME,
- EVENT,
- TITLE_TIMEOUT,
- URL_HASH_KEYS,
- VERSION_TIMEOUT,
- } from "../constants";
- import { loadFromBlob } from "../data/blob";
- import { DataState, ImportedDataState } from "../data/types";
- import {
- ExcalidrawElement,
- NonDeletedExcalidrawElement,
- } from "../element/types";
- import { useCallbackRefState } from "../hooks/useCallbackRefState";
- import { Language, t } from "../i18n";
- import Excalidraw, {
- defaultLang,
- languages,
- } from "../packages/excalidraw/index";
- import { AppState } from "../types";
- import {
- debounce,
- getVersion,
- ResolvablePromise,
- resolvablePromise,
- } from "../utils";
- import { SAVE_TO_LOCAL_STORAGE_TIMEOUT } from "./app_constants";
- import CollabWrapper, {
- CollabAPI,
- CollabContext,
- CollabContextConsumer,
- } from "./collab/CollabWrapper";
- import { LanguageList } from "./components/LanguageList";
- import { exportToBackend, getCollaborationLinkData, loadScene } from "./data";
- import {
- importFromLocalStorage,
- saveToLocalStorage,
- } from "./data/localStorage";
- import CustomStats from "./CustomStats";
- const languageDetector = new LanguageDetector();
- languageDetector.init({
- languageUtils: {
- formatLanguageCode: (langCode: Language["code"]) => langCode,
- isWhitelisted: () => true,
- },
- checkWhitelist: false,
- });
- const saveDebounced = debounce(
- (elements: readonly ExcalidrawElement[], state: AppState) => {
- saveToLocalStorage(elements, state);
- },
- SAVE_TO_LOCAL_STORAGE_TIMEOUT,
- );
- const onBlur = () => {
- saveDebounced.flush();
- };
- const initializeScene = async (opts: {
- collabAPI: CollabAPI;
- }): Promise<ImportedDataState | null> => {
- const searchParams = new URLSearchParams(window.location.search);
- const id = searchParams.get("id");
- const jsonBackendMatch = window.location.hash.match(
- /^#json=([0-9]+),([a-zA-Z0-9_-]+)$/,
- );
- const externalUrlMatch = window.location.hash.match(/^#url=(.*)$/);
- const initialData = importFromLocalStorage();
- let scene: DataState & { scrollToContent?: boolean } = await loadScene(
- null,
- null,
- initialData,
- );
- let roomLinkData = getCollaborationLinkData(window.location.href);
- const isExternalScene = !!(id || jsonBackendMatch || roomLinkData);
- if (isExternalScene) {
- if (
- // don't prompt if scene is empty
- !scene.elements.length ||
- // don't prompt for collab scenes because we don't override local storage
- roomLinkData ||
- // otherwise, prompt whether user wants to override current scene
- window.confirm(t("alerts.loadSceneOverridePrompt"))
- ) {
- // Backwards compatibility with legacy url format
- if (id) {
- scene = await loadScene(id, null, initialData);
- } else if (jsonBackendMatch) {
- scene = await loadScene(
- jsonBackendMatch[1],
- jsonBackendMatch[2],
- initialData,
- );
- }
- scene.scrollToContent = true;
- if (!roomLinkData) {
- window.history.replaceState({}, APP_NAME, window.location.origin);
- }
- } else {
- // https://github.com/excalidraw/excalidraw/issues/1919
- if (document.hidden) {
- return new Promise((resolve, reject) => {
- window.addEventListener(
- "focus",
- () => initializeScene(opts).then(resolve).catch(reject),
- {
- once: true,
- },
- );
- });
- }
- roomLinkData = null;
- window.history.replaceState({}, APP_NAME, window.location.origin);
- }
- } else if (externalUrlMatch) {
- window.history.replaceState({}, APP_NAME, window.location.origin);
- const url = externalUrlMatch[1];
- try {
- const request = await fetch(window.decodeURIComponent(url));
- const data = await loadFromBlob(await request.blob(), null);
- if (
- !scene.elements.length ||
- window.confirm(t("alerts.loadSceneOverridePrompt"))
- ) {
- return data;
- }
- } catch (error) {
- return {
- appState: {
- errorMessage: t("alerts.invalidSceneUrl"),
- },
- };
- }
- }
- if (roomLinkData) {
- return opts.collabAPI.initializeSocketClient(roomLinkData);
- } else if (scene) {
- return scene;
- }
- return null;
- };
- const ExcalidrawWrapper = () => {
- const [errorMessage, setErrorMessage] = useState("");
- const currentLangCode = languageDetector.detect() || defaultLang.code;
- const [langCode, setLangCode] = useState(currentLangCode);
- // initial state
- // ---------------------------------------------------------------------------
- const initialStatePromiseRef = useRef<{
- promise: ResolvablePromise<ImportedDataState | null>;
- }>({ promise: null! });
- if (!initialStatePromiseRef.current.promise) {
- initialStatePromiseRef.current.promise = resolvablePromise<ImportedDataState | null>();
- }
- useEffect(() => {
- // Delayed so that the app has a time to load the latest SW
- setTimeout(() => {
- trackEvent("load", "version", getVersion());
- }, VERSION_TIMEOUT);
- }, []);
- const [
- excalidrawAPI,
- excalidrawRefCallback,
- ] = useCallbackRefState<ExcalidrawImperativeAPI>();
- const collabAPI = useContext(CollabContext)?.api;
- useEffect(() => {
- if (!collabAPI || !excalidrawAPI) {
- return;
- }
- initializeScene({ collabAPI }).then((scene) => {
- initialStatePromiseRef.current.promise.resolve(scene);
- });
- const onHashChange = (event: HashChangeEvent) => {
- event.preventDefault();
- const hash = new URLSearchParams(window.location.hash.slice(1));
- const libraryUrl = hash.get(URL_HASH_KEYS.addLibrary);
- if (libraryUrl) {
- // If hash changed and it contains library url, import it and replace
- // the url to its previous state (important in case of collaboration
- // and similar).
- // Using history API won't trigger another hashchange.
- window.history.replaceState({}, "", event.oldURL);
- excalidrawAPI.importLibrary(libraryUrl, hash.get("token"));
- } else {
- initializeScene({ collabAPI }).then((scene) => {
- if (scene) {
- excalidrawAPI.updateScene(scene);
- }
- });
- }
- };
- const titleTimeout = setTimeout(
- () => (document.title = APP_NAME),
- TITLE_TIMEOUT,
- );
- window.addEventListener(EVENT.HASHCHANGE, onHashChange, false);
- window.addEventListener(EVENT.UNLOAD, onBlur, false);
- window.addEventListener(EVENT.BLUR, onBlur, false);
- return () => {
- window.removeEventListener(EVENT.HASHCHANGE, onHashChange, false);
- window.removeEventListener(EVENT.UNLOAD, onBlur, false);
- window.removeEventListener(EVENT.BLUR, onBlur, false);
- clearTimeout(titleTimeout);
- };
- }, [collabAPI, excalidrawAPI]);
- useEffect(() => {
- languageDetector.cacheUserLanguage(langCode);
- }, [langCode]);
- const onChange = (
- elements: readonly ExcalidrawElement[],
- appState: AppState,
- ) => {
- if (collabAPI?.isCollaborating()) {
- collabAPI.broadcastElements(elements);
- } else {
- // collab scenes are persisted to the server, so we don't have to persist
- // them locally, which has the added benefit of not overwriting whatever
- // the user was working on before joining
- saveDebounced(elements, appState);
- }
- };
- const onExportToBackend = async (
- exportedElements: readonly NonDeletedExcalidrawElement[],
- appState: AppState,
- canvas: HTMLCanvasElement | null,
- ) => {
- if (exportedElements.length === 0) {
- return window.alert(t("alerts.cannotExportEmptyCanvas"));
- }
- if (canvas) {
- try {
- await exportToBackend(exportedElements, {
- ...appState,
- viewBackgroundColor: appState.exportBackground
- ? appState.viewBackgroundColor
- : getDefaultAppState().viewBackgroundColor,
- });
- } catch (error) {
- if (error.name !== "AbortError") {
- const { width, height } = canvas;
- console.error(error, { width, height });
- setErrorMessage(error.message);
- }
- }
- }
- };
- const renderFooter = useCallback(
- (isMobile: boolean) => {
- const renderLanguageList = () => (
- <LanguageList
- onChange={(langCode) => {
- setLangCode(langCode);
- }}
- languages={languages}
- floating={!isMobile}
- currentLangCode={langCode}
- />
- );
- if (isMobile) {
- return (
- <fieldset>
- <legend>{t("labels.language")}</legend>
- {renderLanguageList()}
- </fieldset>
- );
- }
- return renderLanguageList();
- },
- [langCode],
- );
- const renderCustomStats = () => {
- return (
- <CustomStats
- setToastMessage={(message) => excalidrawAPI!.setToastMessage(message)}
- />
- );
- };
- return (
- <>
- <Excalidraw
- ref={excalidrawRefCallback}
- onChange={onChange}
- initialData={initialStatePromiseRef.current.promise}
- onCollabButtonClick={collabAPI?.onCollabButtonClick}
- isCollaborating={collabAPI?.isCollaborating()}
- onPointerUpdate={collabAPI?.onPointerUpdate}
- onExportToBackend={onExportToBackend}
- renderFooter={renderFooter}
- langCode={langCode}
- renderCustomStats={renderCustomStats}
- detectScroll={false}
- />
- {excalidrawAPI && <CollabWrapper excalidrawAPI={excalidrawAPI} />}
- {errorMessage && (
- <ErrorDialog
- message={errorMessage}
- onClose={() => setErrorMessage("")}
- />
- )}
- </>
- );
- };
- const ExcalidrawApp = () => {
- return (
- <TopErrorBoundary>
- <CollabContextConsumer>
- <ExcalidrawWrapper />
- </CollabContextConsumer>
- </TopErrorBoundary>
- );
- };
- export default ExcalidrawApp;
|