signal-desktop/ts/components/SafetyNumberViewer.tsx

262 lines
7.3 KiB
TypeScript
Raw Normal View History

2020-10-30 20:34:04 +00:00
// Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React, { useState, useCallback } from 'react';
import classNames from 'classnames';
import { Button, ButtonVariant } from './Button';
import { QrCode } from './QrCode';
import type { ConversationType } from '../state/ducks/conversations';
2020-07-24 01:35:32 +00:00
import { Intl } from './Intl';
import { Emojify } from './conversation/Emojify';
import type { LocalizerType } from '../types/Util';
import type { SafetyNumberType } from '../types/safetyNumber';
import { SAFETY_NUMBER_MIGRATION_URL } from '../types/support';
import {
SafetyNumberIdentifierType,
SafetyNumberMode,
} from '../types/safetyNumber';
import { arrow } from '../util/keyboard';
export type PropsType = {
2022-03-02 18:24:28 +00:00
contact: ConversationType;
generateSafetyNumber: (contact: ConversationType) => void;
i18n: LocalizerType;
2022-03-02 18:24:28 +00:00
onClose: () => void;
safetyNumberMode: SafetyNumberMode;
safetyNumbers?: ReadonlyArray<SafetyNumberType>;
toggleVerified: (contact: ConversationType) => void;
showOnboarding?: () => void;
verificationDisabled: boolean;
};
2022-11-18 00:45:19 +00:00
export function SafetyNumberViewer({
contact,
generateSafetyNumber,
i18n,
onClose,
safetyNumberMode,
safetyNumbers,
toggleVerified,
showOnboarding,
verificationDisabled,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element | null {
const hasSafetyNumbers = safetyNumbers != null;
2020-09-12 00:46:52 +00:00
React.useEffect(() => {
if (!contact) {
return;
}
generateSafetyNumber(contact);
}, [contact, generateSafetyNumber]);
// Keyboard navigation
const [selectedIndex, setSelectedIndex] = useState(0);
2020-09-12 00:46:52 +00:00
const selectPrevNumber = useCallback(() => {
setSelectedIndex(x => Math.max(x - 1, 0));
}, []);
const selectNextNumber = useCallback(() => {
if (!safetyNumbers || safetyNumbers.length === 0) {
setSelectedIndex(0);
return;
}
setSelectedIndex(x => Math.min(x + 1, safetyNumbers.length - 1));
}, [safetyNumbers]);
React.useEffect(() => {
const handleKeyDown = (ev: KeyboardEvent) => {
if (ev.key === arrow('start')) {
selectPrevNumber();
}
if (ev.key === arrow('end')) {
selectNextNumber();
}
};
document.addEventListener('keydown', handleKeyDown);
return () => {
document.removeEventListener('keydown', handleKeyDown);
};
}, [selectPrevNumber, selectNextNumber]);
if (!contact || !hasSafetyNumbers) {
return null;
}
if (!safetyNumbers.length) {
return (
<div className="module-SafetyNumberViewer">
2023-03-30 00:03:25 +00:00
<div>{i18n('icu:cannotGenerateSafetyNumber')}</div>
2022-03-02 18:24:28 +00:00
<div className="module-SafetyNumberViewer__buttons">
<Button
className="module-SafetyNumberViewer__button"
onClick={() => onClose?.()}
variant={ButtonVariant.Primary}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:ok')}
2022-03-02 18:24:28 +00:00
</Button>
</div>
</div>
);
}
const boldName = (
<span className="module-SafetyNumberViewer__bold-name">
<Emojify text={contact.title} />
</span>
2020-07-24 01:35:32 +00:00
);
2020-09-12 00:46:52 +00:00
const { isVerified } = contact;
const verifyButtonText = isVerified
? i18n('icu:SafetyNumberViewer__clearVerification')
: i18n('icu:SafetyNumberViewer__markAsVerified');
2023-07-18 23:40:14 +00:00
const isMigrationVisible = safetyNumberMode !== SafetyNumberMode.JustE164;
const visibleSafetyNumber = safetyNumbers.at(selectedIndex);
if (!visibleSafetyNumber) {
return null;
}
const cardClassName = classNames('module-SafetyNumberViewer__card', {
'module-SafetyNumberViewer__card--aci':
visibleSafetyNumber.identifierType ===
SafetyNumberIdentifierType.ACIIdentifier,
'module-SafetyNumberViewer__card--e164':
visibleSafetyNumber.identifierType ===
SafetyNumberIdentifierType.E164Identifier,
});
const numberBlocks = visibleSafetyNumber.numberBlocks.join(' ');
const safetyNumberCard = (
<div className="module-SafetyNumberViewer__card-container">
<div className={cardClassName}>
<QrCode
className="module-SafetyNumberViewer__card__qr"
data={visibleSafetyNumber.qrData}
alt={i18n('icu:Install__scan-this-code')}
/>
<div className="module-SafetyNumberViewer__card__number">
{numberBlocks}
</div>
{selectedIndex > 0 && (
<button
type="button"
aria-label={i18n('icu:SafetyNumberViewer__card__prev')}
className="module-SafetyNumberViewer__card__prev"
onClick={selectPrevNumber}
/>
)}
{selectedIndex < safetyNumbers.length - 1 && (
<button
type="button"
aria-label={i18n('icu:SafetyNumberViewer__card__next')}
className="module-SafetyNumberViewer__card__next"
onClick={selectNextNumber}
/>
)}
</div>
</div>
);
const carousel = (
<div className="module-SafetyNumberViewer__carousel">
{safetyNumbers.map(({ identifierType }, index) => {
return (
<button
type="button"
aria-label={i18n('icu:SafetyNumberViewer__carousel__dot', {
index: index + 1,
total: safetyNumbers.length,
})}
aria-pressed={index === selectedIndex}
key={identifierType}
className="module-SafetyNumberViewer__carousel__dot"
onClick={() => setSelectedIndex(index)}
/>
);
})}
</div>
);
return (
<div className="module-SafetyNumberViewer">
{isMigrationVisible && (
<div className="module-SafetyNumberViewer__migration">
<div className="module-SafetyNumberViewer__migration__icon" />
<div className="module-SafetyNumberViewer__migration__text">
<p>
<Intl i18n={i18n} id="icu:SafetyNumberViewer__migration__text" />
</p>
<p>
<a
href={SAFETY_NUMBER_MIGRATION_URL}
rel="noreferrer"
target="_blank"
onClick={e => {
if (showOnboarding) {
e.preventDefault();
showOnboarding();
}
}}
>
<Intl
i18n={i18n}
id="icu:SafetyNumberViewer__migration__learn_more"
/>
</a>
</p>
</div>
</div>
)}
{safetyNumberCard}
{safetyNumbers.length > 1 && carousel}
<div className="module-SafetyNumberViewer__help">
{isMigrationVisible ? (
2023-03-30 00:03:25 +00:00
<Intl
i18n={i18n}
id="icu:SafetyNumberViewer__hint--migration"
2023-03-30 00:03:25 +00:00
components={{ name: boldName }}
/>
) : (
2023-03-27 23:37:39 +00:00
<Intl
i18n={i18n}
id="icu:SafetyNumberViewer__hint--normal"
2023-03-27 23:37:39 +00:00
components={{ name: boldName }}
/>
)}
<br />
<a href={SAFETY_NUMBER_MIGRATION_URL} rel="noreferrer" target="_blank">
<Intl
i18n={i18n}
id="icu:SafetyNumberViewer__migration__learn_more"
/>
</a>
</div>
<div className="module-SafetyNumberViewer__button">
<Button
disabled={verificationDisabled}
onClick={() => {
toggleVerified(contact);
}}
variant={ButtonVariant.Secondary}
>
{verifyButtonText}
</Button>
</div>
</div>
);
2022-11-18 00:45:19 +00:00
}