signal-desktop/ts/components/App.tsx

186 lines
5.3 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React, { useEffect } from 'react';
import { Globals } from '@react-spring/web';
2021-06-14 19:01:00 +00:00
import classNames from 'classnames';
import type { ExecuteMenuRoleType } from './TitleBarContainer';
import type { MenuOptionsType, MenuActionType } from '../types/menu';
import type { AnyToast } from '../types/Toast';
import type { ViewStoryActionCreatorType } from '../state/ducks/stories';
import type { LocalizerType } from '../types/Util';
import { ThemeType } from '../types/Util';
2021-06-14 19:01:00 +00:00
import { AppViewType } from '../state/ducks/app';
2021-12-16 15:02:22 +00:00
import { SmartInstallScreen } from '../state/smart/InstallScreen';
2021-06-14 19:01:00 +00:00
import { StandaloneRegistration } from './StandaloneRegistration';
import { TitleBarContainer } from './TitleBarContainer';
import { ToastManager } from './ToastManager';
2021-09-17 22:24:21 +00:00
import { usePageVisibility } from '../hooks/usePageVisibility';
import { useReducedMotion } from '../hooks/useReducedMotion';
2021-06-14 19:01:00 +00:00
type PropsType = {
2021-06-14 19:01:00 +00:00
appView: AppViewType;
2022-03-04 21:14:52 +00:00
openInbox: () => void;
registerSingleDevice: (number: string, code: string) => Promise<void>;
2021-06-17 21:15:09 +00:00
renderCallManager: () => JSX.Element;
renderGlobalModalContainer: () => JSX.Element;
2023-01-02 21:34:41 +00:00
i18n: LocalizerType;
hasSelectedStoryData: boolean;
renderStoryViewer: (closeView: () => unknown) => JSX.Element;
2022-12-10 02:02:22 +00:00
renderLightbox: () => JSX.Element | null;
2021-11-30 17:51:53 +00:00
requestVerification: (
type: 'sms' | 'voice',
number: string,
token: string
) => Promise<void>;
2021-06-14 19:01:00 +00:00
theme: ThemeType;
isMaximized: boolean;
isFullScreen: boolean;
menuOptions: MenuOptionsType;
onUndoArchive: (conversationId: string) => unknown;
2022-12-14 18:12:04 +00:00
openFileInFolder: (target: string) => unknown;
2022-07-05 16:44:53 +00:00
hasCustomTitleBar: boolean;
2023-01-18 23:31:10 +00:00
OS: string;
2023-01-13 00:24:59 +00:00
osClassName: string;
hideMenuBar: boolean;
executeMenuRole: ExecuteMenuRoleType;
executeMenuAction: (action: MenuActionType) => void;
hideToast: () => unknown;
titleBarDoubleClick: () => void;
toast?: AnyToast;
scrollToMessage: (conversationId: string, messageId: string) => unknown;
viewStory: ViewStoryActionCreatorType;
2023-01-02 21:34:41 +00:00
renderInbox: () => JSX.Element;
};
2021-06-14 19:01:00 +00:00
2022-11-18 00:45:19 +00:00
export function App({
2021-06-14 19:01:00 +00:00
appView,
2022-06-16 19:12:50 +00:00
executeMenuAction,
executeMenuRole,
2023-01-02 21:34:41 +00:00
hasCustomTitleBar,
hasSelectedStoryData,
hideMenuBar,
hideToast,
i18n,
isFullScreen,
2022-06-16 19:12:50 +00:00
isMaximized,
menuOptions,
onUndoArchive,
2022-12-14 18:12:04 +00:00
openFileInFolder,
2023-01-02 21:34:41 +00:00
openInbox,
2023-01-18 23:31:10 +00:00
OS,
2023-01-13 00:24:59 +00:00
osClassName,
2022-06-16 19:12:50 +00:00
registerSingleDevice,
2021-06-17 21:15:09 +00:00
renderCallManager,
renderGlobalModalContainer,
2023-01-02 21:34:41 +00:00
renderInbox,
2022-12-10 02:02:22 +00:00
renderLightbox,
2022-07-06 19:06:20 +00:00
renderStoryViewer,
2021-11-30 17:51:53 +00:00
requestVerification,
2021-06-14 19:01:00 +00:00
theme,
titleBarDoubleClick,
toast,
viewStory,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
2021-06-14 19:01:00 +00:00
let contents;
if (appView === AppViewType.Installer) {
2021-12-16 15:02:22 +00:00
contents = <SmartInstallScreen />;
2021-06-14 19:01:00 +00:00
} else if (appView === AppViewType.Standalone) {
2021-11-30 17:51:53 +00:00
const onComplete = () => {
2023-01-13 00:24:59 +00:00
window.IPC.removeSetupMenuItems();
2021-11-30 17:51:53 +00:00
openInbox();
};
contents = (
<StandaloneRegistration
onComplete={onComplete}
requestVerification={requestVerification}
registerSingleDevice={registerSingleDevice}
/>
);
2021-06-14 19:01:00 +00:00
} else if (appView === AppViewType.Inbox) {
2023-01-02 21:34:41 +00:00
contents = renderInbox();
2021-06-14 19:01:00 +00:00
}
// This are here so that themes are properly applied to anything that is
2021-06-17 21:15:09 +00:00
// created in a portal and exists outside of the <App /> 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]);
2023-01-13 00:24:59 +00:00
useEffect(() => {
document.body.classList.add(osClassName);
}, [osClassName]);
useEffect(() => {
document.body.classList.toggle('os-has-custom-titlebar', hasCustomTitleBar);
}, [hasCustomTitleBar]);
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]);
2021-06-14 19:01:00 +00:00
return (
<TitleBarContainer
theme={theme}
isMaximized={isMaximized}
isFullScreen={isFullScreen}
2022-07-05 16:44:53 +00:00
hasCustomTitleBar={hasCustomTitleBar}
executeMenuRole={executeMenuRole}
titleBarDoubleClick={titleBarDoubleClick}
hasMenu
hideMenuBar={hideMenuBar}
i18n={i18n}
menuOptions={menuOptions}
executeMenuAction={executeMenuAction}
2021-06-14 19:01:00 +00:00
>
<div
className={classNames({
App: true,
'light-theme': theme === ThemeType.light,
'dark-theme': theme === ThemeType.dark,
})}
>
{contents}
2022-12-14 18:12:04 +00:00
<ToastManager
2023-01-18 23:31:10 +00:00
OS={OS}
2022-12-14 18:12:04 +00:00
hideToast={hideToast}
i18n={i18n}
onUndoArchive={onUndoArchive}
2022-12-14 18:12:04 +00:00
openFileInFolder={openFileInFolder}
toast={toast}
/>
{renderGlobalModalContainer()}
{renderCallManager()}
2022-12-10 02:02:22 +00:00
{renderLightbox()}
{hasSelectedStoryData &&
renderStoryViewer(() => viewStory({ closeViewer: true }))}
</div>
</TitleBarContainer>
2021-06-14 19:01:00 +00:00
);
2022-11-18 00:45:19 +00:00
}