TelegramPWA/src/components/common/AnimatedIconWithPreview.tsx
2023-06-12 11:47:13 +02:00

62 lines
2.0 KiB
TypeScript

import React, { memo } from '../../lib/teact/teact';
import type { OwnProps as AnimatedIconProps } from './AnimatedIcon';
import buildClassName from '../../util/buildClassName';
import buildStyle from '../../util/buildStyle';
import useLastCallback from '../../hooks/useLastCallback';
import useMediaTransition from '../../hooks/useMediaTransition';
import useFlag from '../../hooks/useFlag';
import AnimatedIcon from './AnimatedIcon';
import styles from './AnimatedIconWithPreview.module.scss';
type OwnProps =
Partial<AnimatedIconProps>
& { previewUrl?: string; thumbDataUri?: string; noPreviewTransition?: boolean };
const loadedPreviewUrls = new Set();
function AnimatedIconWithPreview(props: OwnProps) {
const {
previewUrl, thumbDataUri, className, ...otherProps
} = props;
const [isPreviewLoaded, markPreviewLoaded] = useFlag(Boolean(thumbDataUri) || loadedPreviewUrls.has(previewUrl));
const transitionClassNames = useMediaTransition(isPreviewLoaded);
const [isAnimationReady, markAnimationReady] = useFlag(false);
const handlePreviewLoad = useLastCallback(() => {
markPreviewLoaded();
loadedPreviewUrls.add(previewUrl);
});
const { size } = props;
return (
<div
className={buildClassName(className, styles.root, transitionClassNames)}
style={buildStyle(size !== undefined && `width: ${size}px; height: ${size}px;`)}
>
{thumbDataUri && !isAnimationReady && (
// eslint-disable-next-line jsx-a11y/alt-text
<img src={thumbDataUri} className={styles.preview} />
)}
{previewUrl && !isAnimationReady && (
// eslint-disable-next-line jsx-a11y/alt-text
<img
src={previewUrl}
className={styles.preview}
onLoad={handlePreviewLoad}
/>
)}
{/* eslint-disable-next-line react/jsx-props-no-spreading */}
<AnimatedIcon {...otherProps} onLoad={markAnimationReady} noTransition />
</div>
);
}
export default memo(AnimatedIconWithPreview);