import React, { useMemo, useState } from 'react'; type ObjectFit = React.CSSProperties['objectFit']; interface CrossfadeImageProps { src: string; width: string; height: string; objectFit?: ObjectFit; duration?: string; } const imgStyle: React.CSSProperties = { position: 'absolute', width: `100%`, height: `100%`, }; export default function CrossfadeImage({ src = '', width, height, objectFit = 'fill', duration = '1s', }: CrossfadeImageProps) { const spanStyle: React.CSSProperties = useMemo( () => ({ display: 'inline-block', position: 'relative', width, height, }), [width, height], ); const imgStyles = useMemo( () => [ { ...imgStyle, objectFit, opacity: 0, transition: `opacity ${duration}` }, { ...imgStyle, objectFit, opacity: 1, transition: `opacity ${duration}` }, { ...imgStyle, objectFit, opacity: 0 }, ], [objectFit, duration], ); const [key, setKey] = useState(0); const [srcs, setSrcs] = useState(['', '']); const nextSrc = src !== srcs[1] ? src : ''; const onLoadImg = () => { setKey((key + 1) % 3); setSrcs([srcs[1], nextSrc]); }; return ( {[...srcs, nextSrc].map( (src, index) => src !== '' && ( ), )} ); } CrossfadeImage.defaultProps = { objectFit: 'fill', duration: '3s', };