mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-23 18:04:12 +08:00
86 lines
2.1 KiB
TypeScript
86 lines
2.1 KiB
TypeScript
import React, { useMemo, useState } from 'react';
|
|
import { createStyles, css } from 'antd-style';
|
|
import classNames from 'classnames';
|
|
import { CSSMotionList } from 'rc-motion';
|
|
|
|
import { COLOR_IMAGES, getClosetColor } from './colorUtil';
|
|
|
|
export interface BackgroundImageProps {
|
|
colorPrimary?: string;
|
|
isLight?: boolean;
|
|
}
|
|
|
|
const useStyle = createStyles(({ token }) => ({
|
|
image: css`
|
|
transition: all ${token.motionDurationSlow};
|
|
position: absolute;
|
|
left: 0;
|
|
top: 0;
|
|
height: 100%;
|
|
width: 100%;
|
|
object-fit: cover;
|
|
object-position: right top;
|
|
`,
|
|
}));
|
|
|
|
const onShow = () => ({
|
|
opacity: 1,
|
|
});
|
|
const onHide = () => ({
|
|
opacity: 0,
|
|
});
|
|
|
|
const BackgroundImage: React.FC<BackgroundImageProps> = ({ colorPrimary, isLight }) => {
|
|
const activeColor = useMemo(() => getClosetColor(colorPrimary), [colorPrimary]);
|
|
const { styles } = useStyle();
|
|
|
|
const [keyList, setKeyList] = useState<string[]>([]);
|
|
|
|
React.useLayoutEffect(() => {
|
|
setKeyList([activeColor as string]);
|
|
}, [activeColor]);
|
|
|
|
return (
|
|
<CSSMotionList
|
|
keys={keyList}
|
|
motionName="transition"
|
|
onEnterStart={onHide}
|
|
onAppearStart={onHide}
|
|
onEnterActive={onShow}
|
|
onAppearActive={onShow}
|
|
onLeaveStart={onShow}
|
|
onLeaveActive={onHide}
|
|
motionDeadline={500}
|
|
>
|
|
{({ key: color, className, style }) => {
|
|
const cls = classNames(styles.image, className);
|
|
const entity = COLOR_IMAGES.find((ent) => ent.color === color);
|
|
|
|
if (!entity || !entity.url) {
|
|
return null as unknown as React.ReactElement;
|
|
}
|
|
|
|
const { opacity } = style || {};
|
|
|
|
return (
|
|
<picture>
|
|
<source srcSet={entity.webp} type="image/webp" />
|
|
<source srcSet={entity.url} type="image/jpeg" />
|
|
<img
|
|
className={cls}
|
|
style={{
|
|
...style,
|
|
opacity: isLight ? opacity : 0,
|
|
}}
|
|
src={entity.url}
|
|
alt=""
|
|
/>
|
|
</picture>
|
|
);
|
|
}}
|
|
</CSSMotionList>
|
|
);
|
|
};
|
|
|
|
export default BackgroundImage;
|