// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { AudioDevice } from '@signalapp/ringrtc'; import type { ReactNode } from 'react'; import React, { useCallback, useEffect, useMemo, useRef, useState, } from 'react'; import { noop, partition } from 'lodash'; import classNames from 'classnames'; import uuid from 'uuid'; import * as LocaleMatcher from '@formatjs/intl-localematcher'; import type { MediaDeviceSettings } from '../types/Calling'; import type { NotificationSettingType, SentMediaQualitySettingType, ZoomFactorType, } from '../types/Storage.d'; import type { ThemeSettingType } from '../types/StorageUIKeys'; import type { AnyToast } from '../types/Toast'; import { ToastType } from '../types/Toast'; import type { ConversationType } from '../state/ducks/conversations'; import type { ConversationColorType, CustomColorType, DefaultConversationColorType, } from '../types/Colors'; import type { LocalizerType, SentMediaQualityType, ThemeType, } from '../types/Util'; import { Button, ButtonVariant } from './Button'; import { ChatColorPicker } from './ChatColorPicker'; import { Checkbox } from './Checkbox'; import { WidthBreakpoint } from './_util'; import { CircleCheckbox, Variant as CircleCheckboxVariant, } from './CircleCheckbox'; import { ConfirmationDialog } from './ConfirmationDialog'; import { DisappearingTimeDialog } from './DisappearingTimeDialog'; import { PhoneNumberDiscoverability } from '../util/phoneNumberDiscoverability'; import { PhoneNumberSharingMode } from '../util/phoneNumberSharingMode'; import { Select } from './Select'; import { Spinner } from './Spinner'; import { ToastManager } from './ToastManager'; import { getCustomColorStyle } from '../util/getCustomColorStyle'; import { shouldNeverBeCalled } from '../util/shouldNeverBeCalled'; import { DEFAULT_DURATIONS_IN_SECONDS, DEFAULT_DURATIONS_SET, format as formatExpirationTimer, } from '../util/expirationTimer'; import { DurationInSeconds } from '../util/durations'; import { useEscapeHandling } from '../hooks/useEscapeHandling'; import { useUniqueId } from '../hooks/useUniqueId'; import { focusableSelector } from '../util/focusableSelectors'; import { Modal } from './Modal'; import { SearchInput } from './SearchInput'; import { removeDiacritics } from '../util/removeDiacritics'; import { assertDev } from '../util/assert'; import { I18n } from './I18n'; type CheckboxChangeHandlerType = (value: boolean) => unknown; type SelectChangeHandlerType = (value: T) => unknown; export type PropsDataType = { // Settings blockedCount: number; customColors: Record; defaultConversationColor: DefaultConversationColorType; deviceName?: string; hasAudioNotifications?: boolean; hasAutoConvertEmoji: boolean; hasAutoDownloadUpdate: boolean; hasAutoLaunch: boolean; hasCallNotifications: boolean; hasCallRingtoneNotification: boolean; hasCountMutedConversations: boolean; hasHideMenuBar?: boolean; hasIncomingCallNotifications: boolean; hasLinkPreviews: boolean; hasMediaCameraPermissions: boolean; hasMediaPermissions: boolean; hasMessageAudio: boolean; hasMinimizeToAndStartInSystemTray: boolean; hasMinimizeToSystemTray: boolean; hasNotificationAttention: boolean; hasNotifications: boolean; hasReadReceipts: boolean; hasRelayCalls?: boolean; hasSpellCheck: boolean; hasStoriesDisabled: boolean; hasTextFormatting: boolean; hasTypingIndicators: boolean; lastSyncTime?: number; notificationContent: NotificationSettingType; phoneNumber: string | undefined; selectedCamera?: string; selectedMicrophone?: AudioDevice; selectedSpeaker?: AudioDevice; sentMediaQualitySetting: SentMediaQualitySettingType; themeSetting: ThemeSettingType; universalExpireTimer: DurationInSeconds; whoCanFindMe: PhoneNumberDiscoverability; whoCanSeeMe: PhoneNumberSharingMode; zoomFactor: ZoomFactorType; // Localization availableLocales: ReadonlyArray; localeOverride: string | null; preferredSystemLocales: ReadonlyArray; resolvedLocale: string; // Other props initialSpellCheckSetting: boolean; // Limited support features isAutoDownloadUpdatesSupported: boolean; isAutoLaunchSupported: boolean; isHideMenuBarSupported: boolean; isNotificationAttentionSupported: boolean; isSyncSupported: boolean; isSystemTraySupported: boolean; isMinimizeToAndStartInSystemTraySupported: boolean; availableCameras: Array< Pick >; } & Omit; type PropsFunctionType = { // Other props addCustomColor: (color: CustomColorType) => unknown; closeSettings: () => unknown; doDeleteAllData: () => unknown; doneRendering: () => unknown; editCustomColor: (colorId: string, color: CustomColorType) => unknown; getConversationsWithCustomColor: ( colorId: string ) => Promise>; makeSyncRequest: () => unknown; removeCustomColor: (colorId: string) => unknown; removeCustomColorOnConversations: (colorId: string) => unknown; resetAllChatColors: () => unknown; resetDefaultChatColor: () => unknown; setGlobalDefaultConversationColor: ( color: ConversationColorType, customColorData?: { id: string; value: CustomColorType; } ) => unknown; // Change handlers onAudioNotificationsChange: CheckboxChangeHandlerType; onAutoConvertEmojiChange: CheckboxChangeHandlerType; onAutoDownloadUpdateChange: CheckboxChangeHandlerType; onAutoLaunchChange: CheckboxChangeHandlerType; onCallNotificationsChange: CheckboxChangeHandlerType; onCallRingtoneNotificationChange: CheckboxChangeHandlerType; onCountMutedConversationsChange: CheckboxChangeHandlerType; onHasStoriesDisabledChanged: SelectChangeHandlerType; onHideMenuBarChange: CheckboxChangeHandlerType; onIncomingCallNotificationsChange: CheckboxChangeHandlerType; onLastSyncTimeChange: (time: number) => unknown; onLocaleChange: (locale: string | null) => void; onMediaCameraPermissionsChange: CheckboxChangeHandlerType; onMediaPermissionsChange: CheckboxChangeHandlerType; onMessageAudioChange: CheckboxChangeHandlerType; onMinimizeToAndStartInSystemTrayChange: CheckboxChangeHandlerType; onMinimizeToSystemTrayChange: CheckboxChangeHandlerType; onNotificationAttentionChange: CheckboxChangeHandlerType; onNotificationContentChange: SelectChangeHandlerType; onNotificationsChange: CheckboxChangeHandlerType; onRelayCallsChange: CheckboxChangeHandlerType; onSelectedCameraChange: SelectChangeHandlerType; onSelectedMicrophoneChange: SelectChangeHandlerType; onSelectedSpeakerChange: SelectChangeHandlerType; onSentMediaQualityChange: SelectChangeHandlerType; onSpellCheckChange: CheckboxChangeHandlerType; onTextFormattingChange: CheckboxChangeHandlerType; onThemeChange: SelectChangeHandlerType; onUniversalExpireTimerChange: SelectChangeHandlerType; onWhoCanSeeMeChange: SelectChangeHandlerType; onWhoCanFindMeChange: SelectChangeHandlerType; onZoomFactorChange: SelectChangeHandlerType; // Localization i18n: LocalizerType; }; export type PropsType = PropsDataType & PropsFunctionType; export type PropsPreloadType = Omit; enum Page { // Accessible through left nav General = 'General', Appearance = 'Appearance', Chats = 'Chats', Calls = 'Calls', Notifications = 'Notifications', Privacy = 'Privacy', // Sub pages ChatColor = 'ChatColor', PNP = 'PNP', } enum LanguageDialog { Selection, Confirmation, } const DEFAULT_ZOOM_FACTORS = [ { text: '75%', value: 0.75, }, { text: '100%', value: 1, }, { text: '125%', value: 1.25, }, { text: '150%', value: 1.5, }, { text: '200%', value: 2, }, ]; export function Preferences({ addCustomColor, availableCameras, availableLocales, availableMicrophones, availableSpeakers, blockedCount, closeSettings, customColors, defaultConversationColor, deviceName = '', doDeleteAllData, doneRendering, editCustomColor, getConversationsWithCustomColor, hasAudioNotifications, hasAutoConvertEmoji, hasAutoDownloadUpdate, hasAutoLaunch, hasCallNotifications, hasCallRingtoneNotification, hasCountMutedConversations, hasHideMenuBar, hasIncomingCallNotifications, hasLinkPreviews, hasMediaCameraPermissions, hasMediaPermissions, hasMessageAudio, hasMinimizeToAndStartInSystemTray, hasMinimizeToSystemTray, hasNotificationAttention, hasNotifications, hasReadReceipts, hasRelayCalls, hasSpellCheck, hasStoriesDisabled, hasTextFormatting, hasTypingIndicators, i18n, initialSpellCheckSetting, isAutoDownloadUpdatesSupported, isAutoLaunchSupported, isHideMenuBarSupported, isNotificationAttentionSupported, isSyncSupported, isSystemTraySupported, isMinimizeToAndStartInSystemTraySupported, lastSyncTime, makeSyncRequest, notificationContent, onAudioNotificationsChange, onAutoConvertEmojiChange, onAutoDownloadUpdateChange, onAutoLaunchChange, onCallNotificationsChange, onCallRingtoneNotificationChange, onCountMutedConversationsChange, onHasStoriesDisabledChanged, onHideMenuBarChange, onIncomingCallNotificationsChange, onLastSyncTimeChange, onLocaleChange, onMediaCameraPermissionsChange, onMediaPermissionsChange, onMessageAudioChange, onMinimizeToAndStartInSystemTrayChange, onMinimizeToSystemTrayChange, onNotificationAttentionChange, onNotificationContentChange, onNotificationsChange, onRelayCallsChange, onSelectedCameraChange, onSelectedMicrophoneChange, onSelectedSpeakerChange, onSentMediaQualityChange, onSpellCheckChange, onTextFormattingChange, onThemeChange, onUniversalExpireTimerChange, onWhoCanSeeMeChange, onWhoCanFindMeChange, onZoomFactorChange, phoneNumber = '', preferredSystemLocales, removeCustomColor, removeCustomColorOnConversations, resetAllChatColors, resetDefaultChatColor, resolvedLocale, selectedCamera, selectedMicrophone, selectedSpeaker, sentMediaQualitySetting, setGlobalDefaultConversationColor, localeOverride, themeSetting, universalExpireTimer = DurationInSeconds.ZERO, whoCanFindMe, whoCanSeeMe, zoomFactor, }: PropsType): JSX.Element { const storiesId = useUniqueId(); const themeSelectId = useUniqueId(); const zoomSelectId = useUniqueId(); const languageId = useUniqueId(); const [confirmDelete, setConfirmDelete] = useState(false); const [confirmStoriesOff, setConfirmStoriesOff] = useState(false); const [page, setPage] = useState(Page.General); const [showSyncFailed, setShowSyncFailed] = useState(false); const [nowSyncing, setNowSyncing] = useState(false); const [showDisappearingTimerDialog, setShowDisappearingTimerDialog] = useState(false); const [languageDialog, setLanguageDialog] = useState( null ); const [selectedLanguageLocale, setSelectedLanguageLocale] = useState< string | null >(localeOverride); const [languageSearchInput, setLanguageSearchInput] = useState(''); const [toast, setToast] = useState(); const [confirmPnpNotDiscoverable, setConfirmPnpNoDiscoverable] = useState(false); function closeLanguageDialog() { setLanguageDialog(null); setSelectedLanguageLocale(localeOverride); } useEffect(() => { doneRendering(); }, [doneRendering]); useEscapeHandling(() => { if (languageDialog != null) { closeLanguageDialog(); } else { closeSettings(); } }); const onZoomSelectChange = useCallback( (value: string) => { const number = parseFloat(value); onZoomFactorChange(number as unknown as ZoomFactorType); }, [onZoomFactorChange] ); const onAudioInputSelectChange = useCallback( (value: string) => { if (value === 'undefined') { onSelectedMicrophoneChange(undefined); } else { onSelectedMicrophoneChange(availableMicrophones[parseInt(value, 10)]); } }, [onSelectedMicrophoneChange, availableMicrophones] ); const settingsPaneRef = useRef(null); useEffect(() => { const settingsPane = settingsPaneRef.current; if (!settingsPane) { return; } const elements = settingsPane.querySelectorAll< | HTMLAnchorElement | HTMLButtonElement | HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement >(focusableSelector); if (!elements.length) { return; } elements[0]?.focus(); }, [page]); const onAudioOutputSelectChange = useCallback( (value: string) => { if (value === 'undefined') { onSelectedSpeakerChange(undefined); } else { onSelectedSpeakerChange(availableSpeakers[parseInt(value, 10)]); } }, [onSelectedSpeakerChange, availableSpeakers] ); const localeDisplayNames = window.SignalContext.getLocaleDisplayNames(); const getLocaleDisplayName = useCallback( (inLocale: string, ofLocale: string): string => { const displayName = localeDisplayNames[inLocale]?.[ofLocale]; assertDev( displayName != null, `Locale display name in ${inLocale} of ${ofLocale} does not exist` ); return ( displayName ?? new Intl.DisplayNames(inLocale, { type: 'language', languageDisplay: 'standard', style: 'long', fallback: 'code', }).of(ofLocale) ); }, [localeDisplayNames] ); const localeSearchOptions = useMemo(() => { const collator = new Intl.Collator('en', { usage: 'sort' }); const availableLocalesOptions = availableLocales .map(locale => { const currentLocaleLabel = getLocaleDisplayName(resolvedLocale, locale); const matchingLocaleLabel = getLocaleDisplayName(locale, locale); return { locale, currentLocaleLabel, matchingLocaleLabel }; }) .sort((a, b) => { return collator.compare(a.locale, b.locale); }); const [localeOverrideMatches, localeOverrideNonMatches] = partition( availableLocalesOptions, option => { return option.locale === localeOverride; } ); const preferredSystemLocaleMatch = LocaleMatcher.match( preferredSystemLocales as Array, // bad types availableLocales as Array, // bad types 'en', { algorithm: 'best fit' } ); return [ ...localeOverrideMatches, { locale: null, currentLocaleLabel: i18n('icu:Preferences__Language__SystemLanguage'), matchingLocaleLabel: getLocaleDisplayName( preferredSystemLocaleMatch, preferredSystemLocaleMatch ), }, ...localeOverrideNonMatches, ]; }, [ i18n, availableLocales, resolvedLocale, localeOverride, preferredSystemLocales, getLocaleDisplayName, ]); const localeSearchResults = useMemo(() => { return localeSearchOptions.filter(option => { const input = removeDiacritics(languageSearchInput.trim().toLowerCase()); if (input === '') { return true; } function isMatch(value: string) { return removeDiacritics(value.toLowerCase()).includes(input); } return ( isMatch(option.currentLocaleLabel) || (option.matchingLocaleLabel && isMatch(option.matchingLocaleLabel)) ); }); }, [localeSearchOptions, languageSearchInput]); let settings: JSX.Element | undefined; if (page === Page.General) { settings = ( <>
{i18n('icu:Preferences__button--general')}
{isAutoLaunchSupported && ( )} {isHideMenuBarSupported && ( )} {isSystemTraySupported && ( <> {isMinimizeToAndStartInSystemTraySupported && ( )} )} {isAutoDownloadUpdatesSupported && ( )} ); } else if (page === Page.Appearance) { let zoomFactors = DEFAULT_ZOOM_FACTORS; if (!zoomFactors.some(({ value }) => value === zoomFactor)) { zoomFactors = [ ...zoomFactors, { text: `${Math.round(zoomFactor * 100)}%`, value: zoomFactor, }, ].sort((a, b) => a.value - b.value); } settings = ( <>
{i18n('icu:Preferences__button--appearance')}
{localeOverride != null ? getLocaleDisplayName(resolvedLocale, localeOverride) : i18n('icu:Preferences__Language__SystemLanguage')} } onClick={() => { setLanguageDialog(LanguageDialog.Selection); }} /> {languageDialog === LanguageDialog.Selection && ( { setLanguageSearchInput(event.currentTarget.value); }} /> } modalFooter={ <> } > {localeSearchResults.length === 0 && (
{i18n('icu:Preferences__Language__NoResults', { searchTerm: languageSearchInput.trim(), })}
)} {localeSearchResults.map(option => { const id = `${languageId}:${option.locale ?? 'system'}`; const isSelected = option.locale === selectedLanguageLocale; return ( ); })}
)} {languageDialog === LanguageDialog.Confirmation && ( { onLocaleChange(selectedLanguageLocale); }, }, ]} > {i18n('icu:Preferences__LanguageModal__Restart__Description')} )} {i18n('icu:Preferences--theme')} } right={ } />
); } else if (page === Page.Chats) { let spellCheckDirtyText: string | undefined; if (initialSpellCheckSetting !== hasSpellCheck) { spellCheckDirtyText = hasSpellCheck ? i18n('icu:spellCheckWillBeEnabled') : i18n('icu:spellCheckWillBeDisabled'); } const lastSyncDate = new Date(lastSyncTime || 0); settings = ( <>
{i18n('icu:Preferences__button--chats')}
} label={i18n('icu:Preferences__auto-convert-emoji--title')} moduleClassName="Preferences__checkbox" name="autoConvertEmoji" onChange={onAutoConvertEmojiChange} /> } /> {isSyncSupported && (
{i18n('icu:sync')}
{i18n('icu:syncExplanation')}{' '} {i18n('icu:Preferences--lastSynced', { date: lastSyncDate.toLocaleDateString(), time: lastSyncDate.toLocaleTimeString(), })}
{showSyncFailed && (
{i18n('icu:syncFailed')}
)} } right={
} />
)} ); } else if (page === Page.Calls) { settings = ( <>
{i18n('icu:Preferences__button--calls')}
({ text: localizeDefault(i18n, device.name), value: device.index, })) : [ { text: i18n( 'icu:callingDeviceSelection__select--no-device' ), value: 'undefined', }, ] } value={selectedMicrophone?.index} /> } right={
} /> { if ( value === String(universalExpireTimer) || value === '-1' ) { setShowDisappearingTimerDialog(true); return; } onUniversalExpireTimerChange(parseInt(value, 10)); }} options={DEFAULT_DURATIONS_IN_SECONDS.map(seconds => { const text = formatExpirationTimer(i18n, seconds, { capitalizeOff: true, }); return { value: seconds, text, }; }).concat([ { value: isCustomDisappearingMessageValue ? universalExpireTimer : DurationInSeconds.fromSeconds(-1), text: isCustomDisappearingMessageValue ? formatExpirationTimer(i18n, universalExpireTimer) : i18n('icu:selectedCustomDisappearingTimeOption'), }, ])} value={universalExpireTimer} /> } />
{i18n('icu:Stories__settings-toggle--title')}
{i18n('icu:Stories__settings-toggle--description')}
} right={ hasStoriesDisabled ? ( ) : ( ) } />
{i18n('icu:clearDataHeader')}
{i18n('icu:clearDataExplanation')}
} right={
} />
{confirmDelete ? ( { setConfirmDelete(false); }} title={i18n('icu:deleteAllDataHeader')} > {i18n('icu:deleteAllDataBody')} ) : null} {confirmStoriesOff ? ( onHasStoriesDisabledChanged(true), style: 'negative', text: i18n('icu:Preferences__turn-stories-off--action'), }, ]} i18n={i18n} onClose={() => { setConfirmStoriesOff(false); }} > {i18n('icu:Preferences__turn-stories-off--body')} ) : null} ); } else if (page === Page.ChatColor) { settings = ( <>
); } else if (page === Page.PNP) { let sharingDescription: string; if (whoCanSeeMe === PhoneNumberSharingMode.Everybody) { sharingDescription = i18n( 'icu:Preferences__pnp__sharing--description--everyone' ); } else if (whoCanFindMe === PhoneNumberDiscoverability.Discoverable) { sharingDescription = i18n( 'icu:Preferences__pnp__sharing--description--nobody' ); } else { sharingDescription = i18n( 'icu:Preferences__pnp__sharing--description--nobody--not-discoverable' ); } settings = ( <>
{sharingDescription}
{ if (value === PhoneNumberDiscoverability.NotDiscoverable) { setConfirmPnpNoDiscoverable(true); } else { onWhoCanFindMeChange(value); } }} options={[ { text: i18n('icu:Preferences__pnp__discoverability__everyone'), value: PhoneNumberDiscoverability.Discoverable, }, { text: i18n('icu:Preferences__pnp__discoverability__nobody'), value: PhoneNumberDiscoverability.NotDiscoverable, readOnly: whoCanSeeMe === PhoneNumberSharingMode.Everybody, onClick: whoCanSeeMe === PhoneNumberSharingMode.Everybody ? () => setToast({ toastType: ToastType.WhoCanFindMeReadOnly }) : noop, }, ]} value={whoCanFindMe} />
{whoCanFindMe === PhoneNumberDiscoverability.Discoverable ? i18n( 'icu:Preferences__pnp__discoverability--description--everyone' ) : i18n( 'icu:Preferences__pnp__discoverability--description--nobody' )}
{confirmPnpNotDiscoverable && ( { setConfirmPnpNoDiscoverable(false); }} actions={[ { action: () => onWhoCanFindMeChange( PhoneNumberDiscoverability.NotDiscoverable ), style: 'affirmative', text: i18n('icu:ok'), }, ]} > {i18n( 'icu:Preferences__pnp__discoverability__nobody__confirmModal__description', { // This is a rare instance where we want to interpolate the exact // text of the string into quotes in the translation as an // explanation. settingTitle: i18n( 'icu:Preferences__pnp__discoverability--title' ), nobodyLabel: i18n( 'icu:Preferences__pnp__discoverability__nobody' ), } )} )} ); } return ( <>
{settings}
setToast(undefined)} i18n={i18n} onShowDebugLog={shouldNeverBeCalled} onUndoArchive={shouldNeverBeCalled} openFileInFolder={shouldNeverBeCalled} toast={toast} containerWidthBreakpoint={WidthBreakpoint.Narrow} /> ); } function SettingsRow({ children, title, className, }: { children: ReactNode; title?: string; className?: string; }): JSX.Element { return (
{title && {title}} {children}
); } function Control({ icon, left, onClick, right, }: { /** A className or `true` to leave room for icon */ icon?: string | true; left: ReactNode; onClick?: () => unknown; right: ReactNode; }): JSX.Element { const content = ( <> {icon && (
)}
{left}
{right}
); if (onClick) { return ( ); } return
{content}
; } type SettingsRadioOptionType = Readonly<{ text: string; value: Enum; readOnly?: boolean; onClick?: () => void; }>; function SettingsRadio({ value, options, onChange, }: { value: Enum; options: ReadonlyArray>; onChange: (value: Enum) => void; }): JSX.Element { const htmlIds = useMemo(() => { return Array.from({ length: options.length }, () => uuid()); }, [options.length]); return (
{options.map(({ text, value: optionValue, readOnly, onClick }, i) => { const htmlId = htmlIds[i]; return ( ); })}
); } function localizeDefault(i18n: LocalizerType, deviceLabel: string): string { return deviceLabel.toLowerCase().startsWith('default') ? deviceLabel.replace( /default/i, i18n('icu:callingDeviceSelection__select--default') ) : deviceLabel; }