2024-01-12 12:59:57 +01:00

203 lines
6.0 KiB
TypeScript

import type { FC } from '../../../lib/teact/teact';
import React, { memo } from '../../../lib/teact/teact';
import { getActions } from '../../../global';
import type { ApiMessage, ApiTypeStory } from '../../../api/types';
import type { ObserveFn } from '../../../hooks/useIntersectionObserver';
import type { ISettings } from '../../../types';
import { getMessageWebPage } from '../../../global/helpers';
import buildClassName from '../../../util/buildClassName';
import trimText from '../../../util/trimText';
import renderText from '../../common/helpers/renderText';
import { calculateMediaDimensions } from './helpers/mediaDimensions';
import { getWebpageButtonText } from './helpers/webpageType';
import useAppLayout from '../../../hooks/useAppLayout';
import useEnsureStory from '../../../hooks/useEnsureStory';
import useLang from '../../../hooks/useLang';
import useLastCallback from '../../../hooks/useLastCallback';
import EmojiIconBackground from '../../common/embedded/EmojiIconBackground';
import SafeLink from '../../common/SafeLink';
import Button from '../../ui/Button';
import BaseStory from './BaseStory';
import Photo from './Photo';
import Video from './Video';
import './WebPage.scss';
const MAX_TEXT_LENGTH = 170; // symbols
const WEBPAGE_STORY_TYPE = 'telegram_story';
type OwnProps = {
message: ApiMessage;
observeIntersection?: ObserveFn;
noAvatars?: boolean;
canAutoLoad?: boolean;
canAutoPlay?: boolean;
inPreview?: boolean;
asForwarded?: boolean;
isDownloading?: boolean;
isProtected?: boolean;
isConnected?: boolean;
backgroundEmojiId?: string;
theme: ISettings['theme'];
story?: ApiTypeStory;
onMediaClick?: () => void;
onCancelMediaTransfer?: () => void;
};
const WebPage: FC<OwnProps> = ({
message,
observeIntersection,
noAvatars,
canAutoLoad,
canAutoPlay,
inPreview,
asForwarded,
isDownloading = false,
isProtected,
isConnected,
story,
theme,
backgroundEmojiId,
onMediaClick,
onCancelMediaTransfer,
}) => {
const { openTelegramLink } = getActions();
const webPage = getMessageWebPage(message);
const { isMobile } = useAppLayout();
const lang = useLang();
const handleMediaClick = useLastCallback(() => {
onMediaClick!();
});
const handleQuickButtonClick = useLastCallback(() => {
if (!webPage) return;
openTelegramLink({
url: webPage.url,
});
});
const { story: storyData } = webPage || {};
useEnsureStory(storyData?.peerId, storyData?.id, story);
if (!webPage) {
return undefined;
}
const {
siteName,
url,
displayUrl,
title,
description,
photo,
video,
type,
} = webPage;
const isStory = type === WEBPAGE_STORY_TYPE;
const isExpiredStory = story && 'isDeleted' in story;
const quickButtonLangKey = !inPreview && !isExpiredStory ? getWebpageButtonText(type) : undefined;
const truncatedDescription = trimText(description, MAX_TEXT_LENGTH);
const isArticle = Boolean(truncatedDescription || title || siteName);
let isSquarePhoto = false;
if (isArticle && webPage?.photo && !webPage.video) {
const { width, height } = calculateMediaDimensions(message, undefined, undefined, isMobile);
isSquarePhoto = width === height;
}
const isMediaInteractive = (photo || video) && onMediaClick && !isSquarePhoto;
const className = buildClassName(
'WebPage',
inPreview && 'in-preview',
isSquarePhoto && 'with-square-photo',
!photo && !video && !inPreview && 'without-media',
video && 'with-video',
!isArticle && 'no-article',
quickButtonLangKey && 'with-quick-button',
);
function renderQuickButton(langKey: string) {
return (
<Button
className="WebPage--quick-button"
size="tiny"
color="translucent"
isRectangular
onClick={handleQuickButtonClick}
>
{lang(langKey)}
</Button>
);
}
return (
<div
className={className}
data-initial={(siteName || displayUrl)[0]}
dir="auto"
>
<div className={buildClassName('WebPage--content', isStory && 'is-story')}>
{isStory && (
<BaseStory story={story} isProtected={isProtected} isConnected={isConnected} isPreview />
)}
{photo && !video && (
<Photo
message={message}
observeIntersection={observeIntersection}
noAvatars={noAvatars}
canAutoLoad={canAutoLoad}
size={isSquarePhoto ? 'pictogram' : 'inline'}
asForwarded={asForwarded}
nonInteractive={!isMediaInteractive}
isDownloading={isDownloading}
isProtected={isProtected}
theme={theme}
onClick={isMediaInteractive ? handleMediaClick : undefined}
onCancelUpload={onCancelMediaTransfer}
/>
)}
{isArticle && (
<div className="WebPage-text">
{backgroundEmojiId && (
<EmojiIconBackground
emojiDocumentId={backgroundEmojiId}
className="WebPage--background-icons"
/>
)}
<SafeLink className="site-name" url={url} text={siteName || displayUrl} />
{!inPreview && title && (
<p className="site-title">{renderText(title)}</p>
)}
{truncatedDescription && (
<p className="site-description">{renderText(truncatedDescription, ['emoji', 'br'])}</p>
)}
</div>
)}
{!inPreview && video && (
<Video
message={message}
observeIntersectionForLoading={observeIntersection!}
noAvatars={noAvatars}
canAutoLoad={canAutoLoad}
canAutoPlay={canAutoPlay}
asForwarded={asForwarded}
isDownloading={isDownloading}
isProtected={isProtected}
onClick={isMediaInteractive ? handleMediaClick : undefined}
onCancelUpload={onCancelMediaTransfer}
/>
)}
</div>
{quickButtonLangKey && renderQuickButton(quickButtonLangKey)}
</div>
);
};
export default memo(WebPage);