193 lines
5 KiB
TypeScript
193 lines
5 KiB
TypeScript
// Copyright 2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { ReactNode } from 'react';
|
|
import React, { useState } from 'react';
|
|
|
|
import { Avatar, AvatarSize } from '../../Avatar';
|
|
import { AvatarLightbox } from '../../AvatarLightbox';
|
|
import type { ConversationType } from '../../../state/ducks/conversations';
|
|
import { GroupDescription } from '../GroupDescription';
|
|
import { About } from '../About';
|
|
import type { GroupV2Membership } from './ConversationDetailsMembershipList';
|
|
import type { LocalizerType, ThemeType } from '../../../types/Util';
|
|
import { bemGenerator } from './util';
|
|
import { BadgeDialog } from '../../BadgeDialog';
|
|
import type { BadgeType } from '../../../badges/types';
|
|
import { UserText } from '../../UserText';
|
|
|
|
export type Props = {
|
|
areWeASubscriber: boolean;
|
|
badges?: ReadonlyArray<BadgeType>;
|
|
canEdit: boolean;
|
|
conversation: ConversationType;
|
|
i18n: LocalizerType;
|
|
isGroup: boolean;
|
|
isMe: boolean;
|
|
memberships: ReadonlyArray<GroupV2Membership>;
|
|
startEditing: (isGroupTitle: boolean) => void;
|
|
theme: ThemeType;
|
|
};
|
|
|
|
enum ConversationDetailsHeaderActiveModal {
|
|
ShowingAvatar,
|
|
ShowingBadges,
|
|
}
|
|
|
|
const bem = bemGenerator('ConversationDetails-header');
|
|
|
|
export function ConversationDetailsHeader({
|
|
areWeASubscriber,
|
|
badges,
|
|
canEdit,
|
|
conversation,
|
|
i18n,
|
|
isGroup,
|
|
isMe,
|
|
memberships,
|
|
startEditing,
|
|
theme,
|
|
}: Props): JSX.Element {
|
|
const [activeModal, setActiveModal] = useState<
|
|
undefined | ConversationDetailsHeaderActiveModal
|
|
>();
|
|
|
|
let preferredBadge: undefined | BadgeType;
|
|
let subtitle: ReactNode;
|
|
if (isGroup) {
|
|
if (conversation.groupDescription) {
|
|
subtitle = (
|
|
<GroupDescription
|
|
i18n={i18n}
|
|
text={conversation.groupDescription}
|
|
title={conversation.title}
|
|
/>
|
|
);
|
|
} else if (canEdit) {
|
|
subtitle = i18n('icu:ConversationDetailsHeader--add-group-description');
|
|
} else {
|
|
subtitle = i18n('icu:ConversationDetailsHeader--members', {
|
|
number: memberships.length,
|
|
});
|
|
}
|
|
} else if (!isMe) {
|
|
subtitle = (
|
|
<>
|
|
<div className={bem('subtitle__about')}>
|
|
<About text={conversation.about} />
|
|
</div>
|
|
<div className={bem('subtitle__phone-number')}>
|
|
{conversation.phoneNumber}
|
|
</div>
|
|
</>
|
|
);
|
|
preferredBadge = badges?.[0];
|
|
}
|
|
|
|
const avatar = (
|
|
<Avatar
|
|
badge={preferredBadge}
|
|
conversationType={conversation.type}
|
|
i18n={i18n}
|
|
size={AvatarSize.EIGHTY}
|
|
{...conversation}
|
|
noteToSelf={isMe}
|
|
onClick={() => {
|
|
setActiveModal(ConversationDetailsHeaderActiveModal.ShowingAvatar);
|
|
}}
|
|
onClickBadge={() => {
|
|
setActiveModal(ConversationDetailsHeaderActiveModal.ShowingBadges);
|
|
}}
|
|
sharedGroupNames={[]}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
|
|
const contents = (
|
|
<div>
|
|
<div className={bem('title')}>
|
|
{isMe ? i18n('icu:noteToSelf') : <UserText text={conversation.title} />}
|
|
{isMe && <span className="ContactModal__official-badge__large" />}
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
let modal: ReactNode;
|
|
switch (activeModal) {
|
|
case ConversationDetailsHeaderActiveModal.ShowingAvatar:
|
|
modal = (
|
|
<AvatarLightbox
|
|
avatarColor={conversation.color}
|
|
avatarPath={conversation.avatarPath}
|
|
conversationTitle={conversation.title}
|
|
i18n={i18n}
|
|
isGroup={isGroup}
|
|
onClose={() => {
|
|
setActiveModal(undefined);
|
|
}}
|
|
/>
|
|
);
|
|
break;
|
|
case ConversationDetailsHeaderActiveModal.ShowingBadges:
|
|
modal = (
|
|
<BadgeDialog
|
|
areWeASubscriber={areWeASubscriber}
|
|
badges={badges || []}
|
|
firstName={conversation.firstName}
|
|
i18n={i18n}
|
|
onClose={() => {
|
|
setActiveModal(undefined);
|
|
}}
|
|
title={conversation.title}
|
|
/>
|
|
);
|
|
break;
|
|
default:
|
|
modal = null;
|
|
break;
|
|
}
|
|
|
|
if (canEdit) {
|
|
return (
|
|
<div className={bem('root')} data-testid="ConversationDetailsHeader">
|
|
{modal}
|
|
{avatar}
|
|
<button
|
|
type="button"
|
|
onClick={ev => {
|
|
ev.preventDefault();
|
|
ev.stopPropagation();
|
|
startEditing(true);
|
|
}}
|
|
className={bem('root', 'editable')}
|
|
>
|
|
{contents}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
onClick={ev => {
|
|
if (ev.target instanceof HTMLAnchorElement) {
|
|
return;
|
|
}
|
|
|
|
ev.preventDefault();
|
|
ev.stopPropagation();
|
|
startEditing(false);
|
|
}}
|
|
className={bem('root', 'editable')}
|
|
>
|
|
<div className={bem('subtitle')}>{subtitle}</div>
|
|
</button>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className={bem('root')} data-testid="ConversationDetailsHeader">
|
|
{modal}
|
|
{avatar}
|
|
{contents}
|
|
<div className={bem('subtitle')}>{subtitle}</div>
|
|
</div>
|
|
);
|
|
}
|