2021-11-02 23:01:13 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import React, { useEffect, useState } from 'react';
|
2021-11-16 16:45:16 +00:00
|
|
|
import classNames from 'classnames';
|
2021-11-02 23:01:13 +00:00
|
|
|
|
|
|
|
import { strictAssert } from '../util/assert';
|
|
|
|
import type { LocalizerType } from '../types/Util';
|
|
|
|
import type { BadgeType } from '../badges/types';
|
2021-11-16 16:45:16 +00:00
|
|
|
import { BadgeCategory } from '../badges/BadgeCategory';
|
2021-11-02 23:01:13 +00:00
|
|
|
import { Modal } from './Modal';
|
2021-11-16 16:45:16 +00:00
|
|
|
import { Button, ButtonSize } from './Button';
|
2021-11-02 23:01:13 +00:00
|
|
|
import { BadgeDescription } from './BadgeDescription';
|
|
|
|
import { BadgeImage } from './BadgeImage';
|
|
|
|
import { BadgeCarouselIndex } from './BadgeCarouselIndex';
|
2021-11-16 16:45:16 +00:00
|
|
|
import { BadgeSustainerInstructionsDialog } from './BadgeSustainerInstructionsDialog';
|
2021-11-02 23:01:13 +00:00
|
|
|
|
2023-10-11 19:06:43 +00:00
|
|
|
export type PropsType = Readonly<{
|
2021-11-30 16:29:57 +00:00
|
|
|
areWeASubscriber: boolean;
|
2021-11-02 23:01:13 +00:00
|
|
|
badges: ReadonlyArray<BadgeType>;
|
|
|
|
firstName?: string;
|
|
|
|
i18n: LocalizerType;
|
|
|
|
onClose: () => unknown;
|
|
|
|
title: string;
|
|
|
|
}>;
|
|
|
|
|
|
|
|
export function BadgeDialog(props: PropsType): null | JSX.Element {
|
2021-11-16 16:45:16 +00:00
|
|
|
const { badges, i18n, onClose } = props;
|
|
|
|
|
|
|
|
const [isShowingInstructions, setIsShowingInstructions] = useState(false);
|
2021-11-02 23:01:13 +00:00
|
|
|
|
|
|
|
const hasBadges = badges.length > 0;
|
|
|
|
useEffect(() => {
|
2021-11-16 16:45:16 +00:00
|
|
|
if (!hasBadges && !isShowingInstructions) {
|
2021-11-02 23:01:13 +00:00
|
|
|
onClose();
|
|
|
|
}
|
2021-11-16 16:45:16 +00:00
|
|
|
}, [hasBadges, isShowingInstructions, onClose]);
|
|
|
|
|
|
|
|
if (isShowingInstructions) {
|
|
|
|
return (
|
|
|
|
<BadgeSustainerInstructionsDialog
|
|
|
|
i18n={i18n}
|
|
|
|
onClose={() => setIsShowingInstructions(false)}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
2021-11-02 23:01:13 +00:00
|
|
|
|
2021-11-16 16:45:16 +00:00
|
|
|
return hasBadges ? (
|
|
|
|
<BadgeDialogWithBadges
|
|
|
|
{...props}
|
|
|
|
onShowInstructions={() => setIsShowingInstructions(true)}
|
|
|
|
/>
|
|
|
|
) : null;
|
2021-11-02 23:01:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function BadgeDialogWithBadges({
|
2021-11-30 16:29:57 +00:00
|
|
|
areWeASubscriber,
|
2021-11-02 23:01:13 +00:00
|
|
|
badges,
|
|
|
|
firstName,
|
|
|
|
i18n,
|
|
|
|
onClose,
|
2021-11-16 16:45:16 +00:00
|
|
|
onShowInstructions,
|
2021-11-02 23:01:13 +00:00
|
|
|
title,
|
2021-11-16 16:45:16 +00:00
|
|
|
}: PropsType & { onShowInstructions: () => unknown }): JSX.Element {
|
2021-11-02 23:01:13 +00:00
|
|
|
const firstBadge = badges[0];
|
|
|
|
strictAssert(
|
|
|
|
firstBadge,
|
|
|
|
'<BadgeDialogWithBadges> got an empty array of badges'
|
|
|
|
);
|
|
|
|
|
|
|
|
const [currentBadgeId, setCurrentBadgeId] = useState(firstBadge.id);
|
|
|
|
|
|
|
|
let currentBadge: BadgeType;
|
|
|
|
let currentBadgeIndex: number = badges.findIndex(
|
|
|
|
b => b.id === currentBadgeId
|
|
|
|
);
|
|
|
|
if (currentBadgeIndex === -1) {
|
|
|
|
currentBadgeIndex = 0;
|
|
|
|
currentBadge = firstBadge;
|
|
|
|
} else {
|
|
|
|
currentBadge = badges[currentBadgeIndex];
|
|
|
|
}
|
|
|
|
|
|
|
|
const setCurrentBadgeIndex = (index: number): void => {
|
|
|
|
const newBadge = badges[index];
|
|
|
|
strictAssert(newBadge, '<BadgeDialog> tried to select a nonexistent badge');
|
|
|
|
setCurrentBadgeId(newBadge.id);
|
|
|
|
};
|
|
|
|
|
|
|
|
const navigate = (change: number): void => {
|
|
|
|
setCurrentBadgeIndex(currentBadgeIndex + change);
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Modal
|
2022-09-27 20:24:21 +00:00
|
|
|
modalName="BadgeDialog"
|
2021-11-02 23:01:13 +00:00
|
|
|
hasXButton
|
|
|
|
moduleClassName="BadgeDialog"
|
|
|
|
i18n={i18n}
|
|
|
|
onClose={onClose}
|
|
|
|
>
|
2021-11-16 16:45:16 +00:00
|
|
|
<div className="BadgeDialog__contents">
|
|
|
|
<button
|
2023-03-30 00:03:25 +00:00
|
|
|
aria-label={i18n('icu:previous')}
|
2021-11-16 16:45:16 +00:00
|
|
|
className="BadgeDialog__nav BadgeDialog__nav--previous"
|
|
|
|
disabled={currentBadgeIndex === 0}
|
|
|
|
onClick={() => navigate(-1)}
|
|
|
|
type="button"
|
|
|
|
/>
|
|
|
|
<div className="BadgeDialog__main">
|
|
|
|
<BadgeImage badge={currentBadge} size={160} />
|
|
|
|
<div className="BadgeDialog__name">{currentBadge.name}</div>
|
|
|
|
<div className="BadgeDialog__description">
|
|
|
|
<BadgeDescription
|
|
|
|
firstName={firstName}
|
|
|
|
template={currentBadge.descriptionTemplate}
|
|
|
|
title={title}
|
|
|
|
/>
|
|
|
|
</div>
|
2021-11-30 16:29:57 +00:00
|
|
|
{!areWeASubscriber && (
|
|
|
|
<Button
|
|
|
|
className={classNames(
|
|
|
|
'BadgeDialog__instructions-button',
|
|
|
|
currentBadge.category !== BadgeCategory.Donor &&
|
|
|
|
'BadgeDialog__instructions-button--hidden'
|
|
|
|
)}
|
|
|
|
onClick={onShowInstructions}
|
|
|
|
size={ButtonSize.Large}
|
|
|
|
>
|
2023-03-30 00:03:25 +00:00
|
|
|
{i18n('icu:BadgeDialog__become-a-sustainer-button')}
|
2021-11-30 16:29:57 +00:00
|
|
|
</Button>
|
|
|
|
)}
|
2021-11-16 16:45:16 +00:00
|
|
|
<BadgeCarouselIndex
|
|
|
|
currentIndex={currentBadgeIndex}
|
|
|
|
totalCount={badges.length}
|
2021-11-02 23:01:13 +00:00
|
|
|
/>
|
|
|
|
</div>
|
2021-11-16 16:45:16 +00:00
|
|
|
<button
|
2023-03-30 00:03:25 +00:00
|
|
|
aria-label={i18n('icu:next')}
|
2021-11-16 16:45:16 +00:00
|
|
|
className="BadgeDialog__nav BadgeDialog__nav--next"
|
|
|
|
disabled={currentBadgeIndex === badges.length - 1}
|
|
|
|
onClick={() => navigate(1)}
|
|
|
|
type="button"
|
2021-11-02 23:01:13 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</Modal>
|
|
|
|
);
|
|
|
|
}
|