// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useEffect } from 'react'; import { Globals } from '@react-spring/web'; import classNames from 'classnames'; import type { AnyToast } from '../types/Toast'; import type { ViewStoryActionCreatorType } from '../state/ducks/stories'; import type { LocalizerType } from '../types/Util'; import type { VerificationTransport } from '../types/VerificationTransport'; import { ThemeType } from '../types/Util'; import { AppViewType } from '../state/ducks/app'; import { SmartInstallScreen } from '../state/smart/InstallScreen'; import { StandaloneRegistration } from './StandaloneRegistration'; import { ToastManager } from './ToastManager'; import { usePageVisibility } from '../hooks/usePageVisibility'; import { useReducedMotion } from '../hooks/useReducedMotion'; type PropsType = { appView: AppViewType; openInbox: () => void; registerSingleDevice: ( number: string, code: string, sessionId: string ) => Promise; renderCallManager: () => JSX.Element; renderGlobalModalContainer: () => JSX.Element; i18n: LocalizerType; hasSelectedStoryData: boolean; renderStoryViewer: (closeView: () => unknown) => JSX.Element; renderLightbox: () => JSX.Element | null; requestVerification: ( number: string, captcha: string, transport: VerificationTransport ) => Promise<{ sessionId: string }>; theme: ThemeType; isMaximized: boolean; isFullScreen: boolean; onUndoArchive: (conversationId: string) => unknown; openFileInFolder: (target: string) => unknown; OS: string; osClassName: string; hideToast: () => unknown; toast?: AnyToast; scrollToMessage: (conversationId: string, messageId: string) => unknown; viewStory: ViewStoryActionCreatorType; renderInbox: () => JSX.Element; }; export function App({ appView, hasSelectedStoryData, hideToast, i18n, isFullScreen, isMaximized, onUndoArchive, openFileInFolder, openInbox, OS, osClassName, registerSingleDevice, renderCallManager, renderGlobalModalContainer, renderInbox, renderLightbox, renderStoryViewer, requestVerification, theme, toast, viewStory, }: PropsType): JSX.Element { let contents; if (appView === AppViewType.Installer) { contents = ; } else if (appView === AppViewType.Standalone) { const onComplete = () => { window.IPC.removeSetupMenuItems(); openInbox(); }; contents = ( ); } else if (appView === AppViewType.Inbox) { contents = renderInbox(); } // This are here so that themes are properly applied to anything that is // created in a portal and exists outside of the container. useEffect(() => { document.body.classList.remove('light-theme'); document.body.classList.remove('dark-theme'); if (theme === ThemeType.dark) { document.body.classList.add('dark-theme'); } if (theme === ThemeType.light) { document.body.classList.add('light-theme'); } }, [theme]); useEffect(() => { document.body.classList.add(osClassName); }, [osClassName]); useEffect(() => { document.body.classList.toggle('full-screen', isFullScreen); document.body.classList.toggle('maximized', isMaximized); }, [isFullScreen, isMaximized]); const isPageVisible = usePageVisibility(); useEffect(() => { document.body.classList.toggle('page-is-visible', isPageVisible); }, [isPageVisible]); // A11y settings for react-spring const prefersReducedMotion = useReducedMotion(); useEffect(() => { Globals.assign({ skipAnimation: prefersReducedMotion, }); }, [prefersReducedMotion]); return (
{contents} {renderGlobalModalContainer()} {renderCallManager()} {renderLightbox()} {hasSelectedStoryData && renderStoryViewer(() => viewStory({ closeViewer: true }))}
); }