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, STORAGE_KEYS, TITLE_TIMEOUT, URL_HASH_KEYS, VERSION_TIMEOUT, } from "../constants"; import { loadFromBlob } from "../data/blob"; import { 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, LibraryItems } 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"; import { restoreAppState, RestoredDataState } from "../data/restore"; import { Tooltip } from "../components/Tooltip"; import { shield } from "../components/icons"; import "./index.scss"; 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 => { 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 localDataState = importFromLocalStorage(); let scene: RestoredDataState & { scrollToContent?: boolean; } = await loadScene(null, null, localDataState); 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, localDataState); } else if (jsonBackendMatch) { scene = await loadScene( jsonBackendMatch[1], jsonBackendMatch[2], localDataState, ); } 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 PlusLinkJSX = (

Introducing Excalidraw+
Try out now!

); const ExcalidrawWrapper = () => { const [errorMessage, setErrorMessage] = useState(""); const currentLangCode = languageDetector.detect() || defaultLang.code; const [langCode, setLangCode] = useState(currentLangCode); // initial state // --------------------------------------------------------------------------- const initialStatePromiseRef = useRef<{ promise: ResolvablePromise; }>({ promise: null! }); if (!initialStatePromiseRef.current.promise) { initialStatePromiseRef.current.promise = resolvablePromise(); } 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(); const collabAPI = useContext(CollabContext)?.api; useEffect(() => { if (!collabAPI || !excalidrawAPI) { return; } initializeScene({ collabAPI }).then((scene) => { if (scene) { try { scene.libraryItems = JSON.parse( localStorage.getItem( STORAGE_KEYS.LOCAL_STORAGE_LIBRARY, ) as string, ) || []; } catch (e) { console.error(e); } } 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, appState: restoreAppState(scene.appState, null), }); } }); } }; 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 renderTopRightUI = useCallback( (isMobile: boolean, appState: AppState) => { return (
{/* */} {/* FIXME remove after 2021-05-20 */} {PlusLinkJSX}
); }, [], ); const renderFooter = useCallback( (isMobile: boolean) => { const renderEncryptedIcon = () => ( {shield} ); const renderLanguageList = () => ( { setLangCode(langCode); }} languages={languages} floating={!isMobile} currentLangCode={langCode} /> ); if (isMobile) { const isTinyDevice = window.innerWidth < 362; return (
{t("labels.language")} {renderLanguageList()}
{/* FIXME remove after 2021-05-20 */}
{PlusLinkJSX}
); } return ( <> {renderEncryptedIcon()} {renderLanguageList()} ); }, [langCode], ); const renderCustomStats = () => { return ( excalidrawAPI!.setToastMessage(message)} /> ); }; const onLibraryChange = async (items: LibraryItems) => { if (!items.length) { localStorage.removeItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY); return; } const serializedItems = JSON.stringify(items); localStorage.setItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY, serializedItems); }; return ( <> {excalidrawAPI && } {errorMessage && ( setErrorMessage("")} /> )} ); }; const ExcalidrawApp = () => { return ( ); }; export default ExcalidrawApp;