2021-03-09 19:16:56 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
2021-01-29 21:19:24 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ReactNode } from 'react';
|
|
|
|
import React, { useState } from 'react';
|
2021-01-29 21:19:24 +00:00
|
|
|
|
2022-12-09 20:37:45 +00:00
|
|
|
import { Avatar, AvatarSize } from '../../Avatar';
|
2021-08-06 00:17:05 +00:00
|
|
|
import { AvatarLightbox } from '../../AvatarLightbox';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ConversationType } from '../../../state/ducks/conversations';
|
2021-06-02 00:24:28 +00:00
|
|
|
import { GroupDescription } from '../GroupDescription';
|
2021-10-27 23:33:06 +00:00
|
|
|
import { About } from '../About';
|
2021-11-02 23:01:13 +00:00
|
|
|
import type { LocalizerType, ThemeType } from '../../../types/Util';
|
2024-02-06 02:13:13 +00:00
|
|
|
import { assertDev } from '../../../util/assert';
|
2021-11-02 23:01:13 +00:00
|
|
|
import { BadgeDialog } from '../../BadgeDialog';
|
|
|
|
import type { BadgeType } from '../../../badges/types';
|
2023-04-20 17:03:43 +00:00
|
|
|
import { UserText } from '../../UserText';
|
2024-03-25 19:22:35 +00:00
|
|
|
import { isInSystemContacts } from '../../../util/isInSystemContacts';
|
|
|
|
import { InContactsIcon } from '../../InContactsIcon';
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
|
|
export type Props = {
|
2021-11-30 16:29:57 +00:00
|
|
|
areWeASubscriber: boolean;
|
2021-11-02 23:01:13 +00:00
|
|
|
badges?: ReadonlyArray<BadgeType>;
|
2021-03-09 19:16:56 +00:00
|
|
|
canEdit: boolean;
|
2021-01-29 21:19:24 +00:00
|
|
|
conversation: ConversationType;
|
2021-03-09 19:16:56 +00:00
|
|
|
i18n: LocalizerType;
|
2021-10-20 23:46:41 +00:00
|
|
|
isGroup: boolean;
|
|
|
|
isMe: boolean;
|
2024-03-26 15:38:18 +00:00
|
|
|
membersCount: number | null;
|
2021-06-03 00:07:37 +00:00
|
|
|
startEditing: (isGroupTitle: boolean) => void;
|
2024-02-06 02:13:13 +00:00
|
|
|
toggleAboutContactModal: (contactId: string) => void;
|
2021-11-02 23:01:13 +00:00
|
|
|
theme: ThemeType;
|
2021-01-29 21:19:24 +00:00
|
|
|
};
|
|
|
|
|
2021-11-02 23:01:13 +00:00
|
|
|
enum ConversationDetailsHeaderActiveModal {
|
|
|
|
ShowingAvatar,
|
|
|
|
ShowingBadges,
|
|
|
|
}
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function ConversationDetailsHeader({
|
2021-11-30 16:29:57 +00:00
|
|
|
areWeASubscriber,
|
2021-11-02 23:01:13 +00:00
|
|
|
badges,
|
2021-03-09 19:16:56 +00:00
|
|
|
canEdit,
|
2021-01-29 21:19:24 +00:00
|
|
|
conversation,
|
2021-03-09 19:16:56 +00:00
|
|
|
i18n,
|
2021-10-20 23:46:41 +00:00
|
|
|
isGroup,
|
|
|
|
isMe,
|
2024-03-26 15:38:18 +00:00
|
|
|
membersCount,
|
2021-03-09 19:16:56 +00:00
|
|
|
startEditing,
|
2024-02-06 02:13:13 +00:00
|
|
|
toggleAboutContactModal,
|
2021-11-02 23:01:13 +00:00
|
|
|
theme,
|
2022-11-18 00:45:19 +00:00
|
|
|
}: Props): JSX.Element {
|
2021-11-02 23:01:13 +00:00
|
|
|
const [activeModal, setActiveModal] = useState<
|
|
|
|
undefined | ConversationDetailsHeaderActiveModal
|
|
|
|
>();
|
2021-08-06 00:17:05 +00:00
|
|
|
|
2021-11-02 23:01:13 +00:00
|
|
|
let preferredBadge: undefined | BadgeType;
|
2021-06-02 00:24:28 +00:00
|
|
|
let subtitle: ReactNode;
|
2023-04-24 20:18:00 +00:00
|
|
|
let hasNestedButton = false;
|
2021-10-20 23:46:41 +00:00
|
|
|
if (isGroup) {
|
|
|
|
if (conversation.groupDescription) {
|
|
|
|
subtitle = (
|
|
|
|
<GroupDescription
|
|
|
|
i18n={i18n}
|
|
|
|
text={conversation.groupDescription}
|
|
|
|
title={conversation.title}
|
|
|
|
/>
|
|
|
|
);
|
2023-04-24 20:18:00 +00:00
|
|
|
hasNestedButton = true;
|
2021-10-20 23:46:41 +00:00
|
|
|
} else if (canEdit) {
|
2023-03-30 00:03:25 +00:00
|
|
|
subtitle = i18n('icu:ConversationDetailsHeader--add-group-description');
|
2021-10-20 23:46:41 +00:00
|
|
|
} else {
|
2023-03-30 00:03:25 +00:00
|
|
|
subtitle = i18n('icu:ConversationDetailsHeader--members', {
|
2024-03-26 15:38:18 +00:00
|
|
|
number: membersCount ?? 0,
|
2023-03-27 23:37:39 +00:00
|
|
|
});
|
2021-10-20 23:46:41 +00:00
|
|
|
}
|
|
|
|
} else if (!isMe) {
|
2021-10-27 23:33:06 +00:00
|
|
|
subtitle = (
|
2024-02-13 21:57:45 +00:00
|
|
|
<div className="ConversationDetailsHeader__subtitle__about">
|
|
|
|
<About text={conversation.about} />
|
|
|
|
</div>
|
2021-10-27 23:33:06 +00:00
|
|
|
);
|
2021-11-02 23:01:13 +00:00
|
|
|
preferredBadge = badges?.[0];
|
2021-06-02 00:24:28 +00:00
|
|
|
}
|
|
|
|
|
2021-08-06 00:17:05 +00:00
|
|
|
const avatar = (
|
|
|
|
<Avatar
|
2021-11-02 23:01:13 +00:00
|
|
|
badge={preferredBadge}
|
2021-10-27 15:50:58 +00:00
|
|
|
conversationType={conversation.type}
|
2021-08-06 00:17:05 +00:00
|
|
|
i18n={i18n}
|
2022-12-09 20:37:45 +00:00
|
|
|
size={AvatarSize.EIGHTY}
|
2021-08-06 00:17:05 +00:00
|
|
|
{...conversation}
|
2021-10-20 23:46:41 +00:00
|
|
|
noteToSelf={isMe}
|
2021-11-02 23:01:13 +00:00
|
|
|
onClick={() => {
|
2021-11-18 20:01:53 +00:00
|
|
|
setActiveModal(ConversationDetailsHeaderActiveModal.ShowingAvatar);
|
|
|
|
}}
|
|
|
|
onClickBadge={() => {
|
|
|
|
setActiveModal(ConversationDetailsHeaderActiveModal.ShowingBadges);
|
2021-11-02 23:01:13 +00:00
|
|
|
}}
|
2021-08-06 00:17:05 +00:00
|
|
|
sharedGroupNames={[]}
|
2021-11-02 23:01:13 +00:00
|
|
|
theme={theme}
|
2021-08-06 00:17:05 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
2021-11-02 23:01:13 +00:00
|
|
|
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
|
2021-11-30 16:29:57 +00:00
|
|
|
areWeASubscriber={areWeASubscriber}
|
2021-11-02 23:01:13 +00:00
|
|
|
badges={badges || []}
|
|
|
|
firstName={conversation.firstName}
|
|
|
|
i18n={i18n}
|
|
|
|
onClose={() => {
|
|
|
|
setActiveModal(undefined);
|
|
|
|
}}
|
|
|
|
title={conversation.title}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
modal = null;
|
|
|
|
break;
|
|
|
|
}
|
2021-08-06 00:17:05 +00:00
|
|
|
|
2021-03-09 19:16:56 +00:00
|
|
|
if (canEdit) {
|
2024-02-06 02:13:13 +00:00
|
|
|
assertDev(isGroup, 'Only groups support editable title');
|
|
|
|
|
2021-03-09 19:16:56 +00:00
|
|
|
return (
|
2024-02-06 02:13:13 +00:00
|
|
|
<div
|
|
|
|
className="ConversationDetailsHeader"
|
|
|
|
data-testid="ConversationDetailsHeader"
|
|
|
|
>
|
2021-11-02 23:01:13 +00:00
|
|
|
{modal}
|
2021-08-06 00:17:05 +00:00
|
|
|
{avatar}
|
2021-06-03 00:07:37 +00:00
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
onClick={ev => {
|
|
|
|
ev.preventDefault();
|
|
|
|
ev.stopPropagation();
|
|
|
|
startEditing(true);
|
|
|
|
}}
|
2024-02-06 02:13:13 +00:00
|
|
|
className="ConversationDetailsHeader__edit-button"
|
2021-06-03 00:07:37 +00:00
|
|
|
>
|
2024-02-06 02:13:13 +00:00
|
|
|
<div className="ConversationDetailsHeader__title">
|
|
|
|
<UserText text={conversation.title} />
|
|
|
|
</div>
|
2021-06-03 00:07:37 +00:00
|
|
|
</button>
|
2023-04-24 20:18:00 +00:00
|
|
|
{hasNestedButton ? (
|
2024-02-06 02:13:13 +00:00
|
|
|
<div className="ConversationDetailsHeader__subtitle">{subtitle}</div>
|
2023-04-24 20:18:00 +00:00
|
|
|
) : (
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
onClick={ev => {
|
|
|
|
if (ev.target instanceof HTMLAnchorElement) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ev.preventDefault();
|
|
|
|
ev.stopPropagation();
|
|
|
|
startEditing(false);
|
|
|
|
}}
|
2024-02-06 02:13:13 +00:00
|
|
|
className="ConversationDetailsHeader__edit-button"
|
2023-04-24 20:18:00 +00:00
|
|
|
>
|
2024-02-06 02:13:13 +00:00
|
|
|
<div className="ConversationDetailsHeader__subtitle">
|
|
|
|
{subtitle}
|
|
|
|
</div>
|
2023-04-24 20:18:00 +00:00
|
|
|
</button>
|
|
|
|
)}
|
2021-06-03 00:07:37 +00:00
|
|
|
</div>
|
2021-03-09 19:16:56 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-02-06 02:13:13 +00:00
|
|
|
let title: JSX.Element;
|
|
|
|
|
|
|
|
if (isMe) {
|
|
|
|
title = (
|
|
|
|
<div className="ConversationDetailsHeader__title">
|
|
|
|
{i18n('icu:noteToSelf')}
|
|
|
|
<span className="ContactModal__official-badge__large" />
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
} else if (isGroup) {
|
|
|
|
title = (
|
|
|
|
<div className="ConversationDetailsHeader__title">
|
|
|
|
<UserText text={conversation.title} />
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
title = (
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
onClick={ev => {
|
|
|
|
ev.preventDefault();
|
|
|
|
ev.stopPropagation();
|
|
|
|
toggleAboutContactModal(conversation.id);
|
|
|
|
}}
|
|
|
|
className="ConversationDetailsHeader__about-button"
|
|
|
|
>
|
|
|
|
<div className="ConversationDetailsHeader__title">
|
|
|
|
<UserText text={conversation.title} />
|
2024-03-25 19:22:35 +00:00
|
|
|
{isInSystemContacts(conversation) && (
|
|
|
|
<span>
|
|
|
|
{' '}
|
|
|
|
<InContactsIcon
|
|
|
|
className="ConversationDetailsHeader__title-contact-icon"
|
|
|
|
i18n={i18n}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
)}
|
2024-02-06 02:13:13 +00:00
|
|
|
<span className="ConversationDetailsHeader__about-icon" />
|
|
|
|
</div>
|
|
|
|
</button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-08-06 00:17:05 +00:00
|
|
|
return (
|
2024-02-06 02:13:13 +00:00
|
|
|
<div
|
|
|
|
className="ConversationDetailsHeader"
|
|
|
|
data-testid="ConversationDetailsHeader"
|
|
|
|
>
|
2021-11-02 23:01:13 +00:00
|
|
|
{modal}
|
2021-08-06 00:17:05 +00:00
|
|
|
{avatar}
|
2024-02-06 02:13:13 +00:00
|
|
|
{title}
|
|
|
|
<div className="ConversationDetailsHeader__subtitle">{subtitle}</div>
|
2021-08-06 00:17:05 +00:00
|
|
|
</div>
|
|
|
|
);
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|