// Copyright 2023 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React from 'react'; import { noop } from 'lodash'; import { DialogType } from '../../types/Dialogs'; import { InstallScreenStep } from '../../types/InstallScreen'; import type { LocalizerType } from '../../types/Util'; import { PRODUCTION_DOWNLOAD_URL, BETA_DOWNLOAD_URL, UNSUPPORTED_OS_URL, } from '../../types/support'; import type { UpdatesStateType } from '../../state/ducks/updates'; import { isBeta } from '../../util/version'; import { missingCaseError } from '../../util/missingCaseError'; import { roundFractionForProgressBar } from '../../util/numbers'; import { ConfirmationDialog } from '../ConfirmationDialog'; import { Modal } from '../Modal'; import { I18n } from '../I18n'; import { formatFileSize } from '../../util/formatFileSize'; export type PropsType = UpdatesStateType & Readonly<{ i18n: LocalizerType; step: InstallScreenStep; forceUpdate: () => void; startUpdate: () => void; currentVersion: string; OS: string; onClose?: () => void; }>; export function InstallScreenUpdateDialog({ i18n, step, dialogType, isCheckingForUpdates, downloadSize, downloadedSize, forceUpdate, startUpdate, currentVersion, OS, onClose = noop, }: PropsType): JSX.Element | null { const learnMoreLink = (parts: Array) => ( {parts} ); const dialogName = `InstallScreenUpdateDialog.${dialogType}`; if (dialogType === DialogType.None) { if (step === InstallScreenStep.BackupImport) { if (isCheckingForUpdates) { return ; } return ( {i18n('icu:InstallScreenUpdateDialog--update-required__body')} ); } return null; } if (dialogType === DialogType.UnsupportedOS) { return ( ); } if ( dialogType === DialogType.AutoUpdate || // Manual update with an action button dialogType === DialogType.DownloadReady || dialogType === DialogType.FullDownloadReady || dialogType === DialogType.DownloadedUpdate ) { let title = i18n('icu:autoUpdateNewVersionTitle'); let actionText: string | JSX.Element = i18n( 'icu:autoUpdateRestartButtonLabel' ); let bodyText = i18n('icu:InstallScreenUpdateDialog--auto-update__body'); if ( dialogType === DialogType.DownloadReady || dialogType === DialogType.FullDownloadReady ) { actionText = ( ({formatFileSize(downloadSize ?? 0)}) ), }} /> ); } if (dialogType === DialogType.DownloadedUpdate) { title = i18n('icu:DialogUpdate__downloaded'); bodyText = i18n('icu:InstallScreenUpdateDialog--downloaded__body'); } return ( {bodyText} ); } if (dialogType === DialogType.Downloading) { const fractionComplete = roundFractionForProgressBar( (downloadedSize || 0) / (downloadSize || 1) ); return ; } if ( dialogType === DialogType.Cannot_Update || dialogType === DialogType.Cannot_Update_Require_Manual ) { const url = isBeta(currentVersion) ? BETA_DOWNLOAD_URL : PRODUCTION_DOWNLOAD_URL; const title = i18n('icu:cannotUpdate'); const body = ( {url} ), }} /> ); if (dialogType === DialogType.Cannot_Update) { return ( {body} ); } return ( {body} ); } if (dialogType === DialogType.MacOS_Read_Only) { // No focus trap, because there are no focusable elements. return ( Signal.app, folder: /Applications, }} i18n={i18n} id="icu:readOnlyVolume" /> ); } throw missingCaseError(dialogType); } export function DownloadingModal({ i18n, width, }: { i18n: LocalizerType; width: number; }): JSX.Element { // Focus trap can't be used because there are no elements that can be // focused within the modal. return (
); }