2023-09-28 01:55:22 +02:00

214 lines
6.5 KiB
TypeScript

import type { FC } from '../../lib/teact/teact';
import React, {
memo, useEffect, useRef, useState,
} from '../../lib/teact/teact';
import type { ApiVideo } from '../../api/types';
import type { ObserveFn } from '../../hooks/useIntersectionObserver';
import { ApiMediaFormat } from '../../api/types';
import buildClassName from '../../util/buildClassName';
import { IS_TOUCH_ENV } from '../../util/windowEnvironment';
import { preventMessageInputBlurWithBubbling } from '../middle/helpers/preventMessageInputBlur';
import useBuffering from '../../hooks/useBuffering';
import useCanvasBlur from '../../hooks/useCanvasBlur';
import useContextMenuHandlers from '../../hooks/useContextMenuHandlers';
import { useIsIntersecting } from '../../hooks/useIntersectionObserver';
import useLang from '../../hooks/useLang';
import useLastCallback from '../../hooks/useLastCallback';
import useMedia from '../../hooks/useMedia';
import useMenuPosition from '../../hooks/useMenuPosition';
import Button from '../ui/Button';
import Menu from '../ui/Menu';
import MenuItem from '../ui/MenuItem';
import OptimizedVideo from '../ui/OptimizedVideo';
import Spinner from '../ui/Spinner';
import './GifButton.scss';
type OwnProps = {
gif: ApiVideo;
observeIntersection: ObserveFn;
isDisabled?: boolean;
className?: string;
onClick?: (gif: ApiVideo, isSilent?: boolean, shouldSchedule?: boolean) => void;
onUnsaveClick?: (gif: ApiVideo) => void;
isSavedMessages?: boolean;
};
const GifButton: FC<OwnProps> = ({
gif,
isDisabled,
className,
observeIntersection,
onClick,
onUnsaveClick,
isSavedMessages,
}) => {
// eslint-disable-next-line no-null/no-null
const ref = useRef<HTMLDivElement>(null);
const lang = useLang();
const localMediaHash = `gif${gif.id}`;
const isIntersecting = useIsIntersecting(ref, observeIntersection);
const loadAndPlay = isIntersecting && !isDisabled;
const previewBlobUrl = useMedia(`${localMediaHash}?size=m`, !loadAndPlay, ApiMediaFormat.BlobUrl);
const [withThumb] = useState(gif.thumbnail?.dataUri && !previewBlobUrl);
const thumbRef = useCanvasBlur(gif.thumbnail?.dataUri, !withThumb);
const videoData = useMedia(localMediaHash, !loadAndPlay, ApiMediaFormat.BlobUrl);
const shouldRenderVideo = Boolean(loadAndPlay && videoData);
const { isBuffered, bufferingHandlers } = useBuffering(true);
const shouldRenderSpinner = loadAndPlay && !isBuffered;
const isVideoReady = loadAndPlay && isBuffered;
const {
isContextMenuOpen, contextMenuPosition,
handleBeforeContextMenu, handleContextMenu,
handleContextMenuClose, handleContextMenuHide,
} = useContextMenuHandlers(ref);
const getTriggerElement = useLastCallback(() => ref.current);
const getRootElement = useLastCallback(() => ref.current!.closest('.custom-scroll, .no-scrollbar'));
const getMenuElement = useLastCallback(() => ref.current!.querySelector('.gif-context-menu .bubble'));
const {
positionX, positionY, transformOriginX, transformOriginY, style: menuStyle,
} = useMenuPosition(
contextMenuPosition,
getTriggerElement,
getRootElement,
getMenuElement,
);
const handleClick = useLastCallback(() => {
if (isContextMenuOpen || !onClick) return;
onClick({
...gif,
blobUrl: videoData,
});
});
const handleUnsaveClick = useLastCallback((e: React.MouseEvent) => {
e.stopPropagation();
e.preventDefault();
onUnsaveClick!(gif);
});
const handleContextDelete = useLastCallback(() => {
onUnsaveClick?.(gif);
});
const handleSendQuiet = useLastCallback(() => {
onClick!({
...gif,
blobUrl: videoData,
}, true);
});
const handleSendScheduled = useLastCallback(() => {
onClick!({
...gif,
blobUrl: videoData,
}, undefined, true);
});
const handleMouseDown = useLastCallback((e: React.MouseEvent<HTMLElement>) => {
preventMessageInputBlurWithBubbling(e);
handleBeforeContextMenu(e);
});
useEffect(() => {
if (isDisabled) handleContextMenuClose();
}, [handleContextMenuClose, isDisabled]);
const fullClassName = buildClassName(
'GifButton',
gif.width && gif.height && gif.width < gif.height ? 'vertical' : 'horizontal',
onClick && 'interactive',
localMediaHash,
className,
);
return (
<div
ref={ref}
className={fullClassName}
onMouseDown={handleMouseDown}
onClick={handleClick}
onContextMenu={handleContextMenu}
>
{!IS_TOUCH_ENV && onUnsaveClick && (
<Button
className="gif-unsave-button"
color="dark"
pill
noFastClick
onClick={handleUnsaveClick}
>
<i className="icon icon-close gif-unsave-button-icon" />
</Button>
)}
{withThumb && (
<canvas
ref={thumbRef}
className="thumbnail"
// We need to always render to avoid blur re-calculation
style={isVideoReady ? 'display: none;' : undefined}
/>
)}
{previewBlobUrl && !isVideoReady && (
<img
src={previewBlobUrl}
alt=""
className="preview"
draggable={false}
/>
)}
{shouldRenderVideo && (
<OptimizedVideo
canPlay
src={videoData}
autoPlay
loop
muted
disablePictureInPicture
playsInline
preload="none"
// eslint-disable-next-line react/jsx-props-no-spreading
{...bufferingHandlers}
/>
)}
{shouldRenderSpinner && (
<Spinner color={previewBlobUrl || withThumb ? 'white' : 'black'} />
)}
{onClick && contextMenuPosition !== undefined && (
<Menu
isOpen={isContextMenuOpen}
transformOriginX={transformOriginX}
transformOriginY={transformOriginY}
positionX={positionX}
positionY={positionY}
style={menuStyle}
className="gif-context-menu"
autoClose
onClose={handleContextMenuClose}
onCloseAnimationEnd={handleContextMenuHide}
>
{!isSavedMessages && <MenuItem onClick={handleSendQuiet} icon="mute">{lang('SendWithoutSound')}</MenuItem>}
<MenuItem onClick={handleSendScheduled} icon="calendar">
{lang(isSavedMessages ? 'SetReminder' : 'ScheduleMessage')}
</MenuItem>
{onUnsaveClick && (
<MenuItem destructive icon="delete" onClick={handleContextDelete}>{lang('Delete')}</MenuItem>
)}
</Menu>
)}
</div>
);
};
export default memo(GifButton);