TelegramPWA/src/components/common/FullNameTitle.tsx
2025-01-21 18:29:38 +01:00

164 lines
4.8 KiB
TypeScript

import type { FC } from '../../lib/teact/teact';
import React, { memo, useMemo } from '../../lib/teact/teact';
import { getActions } from '../../global';
import type {
ApiPeer,
} from '../../api/types';
import type { ObserveFn } from '../../hooks/useIntersectionObserver';
import type { CustomPeer } from '../../types';
import { EMOJI_STATUS_LOOP_LIMIT } from '../../config';
import {
getChatTitle,
getUserFullName,
isAnonymousForwardsChat,
isChatWithRepliesBot,
isChatWithVerificationCodesBot,
isPeerUser,
} from '../../global/helpers';
import buildClassName from '../../util/buildClassName';
import { copyTextToClipboard } from '../../util/clipboard';
import stopEvent from '../../util/stopEvent';
import renderText from './helpers/renderText';
import useLastCallback from '../../hooks/useLastCallback';
import useOldLang from '../../hooks/useOldLang';
import Transition from '../ui/Transition';
import CustomEmoji from './CustomEmoji';
import FakeIcon from './FakeIcon';
import StarIcon from './icons/StarIcon';
import VerifiedIcon from './VerifiedIcon';
import styles from './FullNameTitle.module.scss';
type OwnProps = {
peer: ApiPeer | CustomPeer;
className?: string;
noVerified?: boolean;
noFake?: boolean;
withEmojiStatus?: boolean;
emojiStatusSize?: number;
isSavedMessages?: boolean;
isSavedDialog?: boolean;
noLoopLimit?: boolean;
canCopyTitle?: boolean;
iconElement?: React.ReactNode;
onEmojiStatusClick?: NoneToVoidFunction;
observeIntersection?: ObserveFn;
};
const FullNameTitle: FC<OwnProps> = ({
className,
peer,
noVerified,
noFake,
withEmojiStatus,
emojiStatusSize,
isSavedMessages,
isSavedDialog,
noLoopLimit,
canCopyTitle,
iconElement,
onEmojiStatusClick,
observeIntersection,
}) => {
const lang = useOldLang();
const { showNotification } = getActions();
const realPeer = 'id' in peer ? peer : undefined;
const customPeer = 'isCustomPeer' in peer ? peer : undefined;
const isUser = realPeer && isPeerUser(realPeer);
const title = realPeer && (isUser ? getUserFullName(realPeer) : getChatTitle(lang, realPeer));
const isPremium = isUser && realPeer.isPremium;
const canShowEmojiStatus = withEmojiStatus && !isSavedMessages && realPeer;
const handleTitleClick = useLastCallback((e) => {
if (!title || !canCopyTitle) {
return;
}
stopEvent(e);
copyTextToClipboard(title);
showNotification({ message: `${isUser ? 'User' : 'Chat'} name was copied` });
});
const specialTitle = useMemo(() => {
if (customPeer) {
return customPeer.title || lang(customPeer.titleKey!);
}
if (isSavedMessages) {
return lang(isSavedDialog ? 'MyNotes' : 'SavedMessages');
}
if (isAnonymousForwardsChat(realPeer!.id)) {
return lang('AnonymousForward');
}
if (isChatWithRepliesBot(realPeer!.id)) {
return lang('RepliesTitle');
}
if (isChatWithVerificationCodesBot(realPeer!.id)) {
return lang('VerifyCodesNotifications');
}
return undefined;
}, [customPeer, isSavedDialog, isSavedMessages, lang, realPeer]);
const botVerificationIconId = realPeer?.botVerificationIconId;
return (
<div className={buildClassName('title', styles.root, className)}>
{botVerificationIconId && (
<CustomEmoji
documentId={botVerificationIconId}
size={emojiStatusSize}
loopLimit={!noLoopLimit ? EMOJI_STATUS_LOOP_LIMIT : undefined}
observeIntersectionForLoading={observeIntersection}
/>
)}
<h3
dir="auto"
role="button"
className={buildClassName(
'fullName',
styles.fullName,
canCopyTitle && styles.canCopy,
)}
onClick={handleTitleClick}
>
{specialTitle || renderText(title || '')}
</h3>
{!iconElement && peer && (
<>
{!noVerified && peer?.isVerified && <VerifiedIcon />}
{!noFake && peer?.fakeType && <FakeIcon fakeType={peer.fakeType} />}
{canShowEmojiStatus && realPeer.emojiStatus && (
<Transition
className={styles.transition}
activeKey={Number(realPeer.emojiStatus.documentId)}
name="fade"
shouldCleanup
shouldRestoreHeight
>
<CustomEmoji
forceAlways
documentId={realPeer.emojiStatus.documentId}
size={emojiStatusSize}
loopLimit={!noLoopLimit ? EMOJI_STATUS_LOOP_LIMIT : undefined}
observeIntersectionForLoading={observeIntersection}
onClick={onEmojiStatusClick}
/>
</Transition>
)}
{canShowEmojiStatus && !realPeer.emojiStatus && isPremium && <StarIcon />}
</>
)}
{iconElement}
</div>
);
};
export default memo(FullNameTitle);