mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-24 02:20:01 +08:00
183 lines
4.5 KiB
TypeScript
183 lines
4.5 KiB
TypeScript
import React, { useContext } from 'react';
|
|
import { Badge, Carousel, Flex, Skeleton, Typography } from 'antd';
|
|
import { createStyles } from 'antd-style';
|
|
import classNames from 'classnames';
|
|
|
|
import useLocale from '../../../hooks/useLocale';
|
|
import SiteContext from '../../../theme/slots/SiteContext';
|
|
import type { Extra, Icon } from './util';
|
|
import { getCarouselStyle, useSiteData } from './util';
|
|
|
|
const useStyle = createStyles(({ token, css, cx }) => {
|
|
const { carousel } = getCarouselStyle();
|
|
|
|
const itemBase = css`
|
|
display: flex;
|
|
flex: 1 1 0;
|
|
flex-direction: column;
|
|
align-items: stretch;
|
|
text-decoration: none;
|
|
background: ${token.colorBgContainer};
|
|
border: ${token.lineWidth}px solid ${token.colorBorderSecondary};
|
|
border-radius: ${token.borderRadiusLG}px;
|
|
transition: all ${token.motionDurationSlow};
|
|
padding-block: ${token.paddingMD}px;
|
|
padding-inline: ${token.paddingLG}px;
|
|
box-sizing: border-box;
|
|
`;
|
|
|
|
return {
|
|
itemBase,
|
|
ribbon: css`
|
|
& > .${cx(itemBase)} {
|
|
height: 100%;
|
|
}
|
|
`,
|
|
cardItem: css`
|
|
&:hover {
|
|
box-shadow: ${token.boxShadowCard};
|
|
border-color: transparent;
|
|
}
|
|
`,
|
|
sliderItem: css`
|
|
margin: 0 ${token.margin}px;
|
|
text-align: start;
|
|
`,
|
|
container: css`
|
|
display: flex;
|
|
width: 100%;
|
|
max-width: 100%;
|
|
margin-inline: auto;
|
|
box-sizing: border-box;
|
|
column-gap: ${token.paddingMD * 2}px;
|
|
align-items: stretch;
|
|
text-align: start;
|
|
min-height: 178px;
|
|
> * {
|
|
width: calc((100% - ${token.marginXXL * 2}px) / 3);
|
|
}
|
|
`,
|
|
carousel,
|
|
bannerBg: css`
|
|
height: ${token.fontSize}px;
|
|
`,
|
|
};
|
|
});
|
|
|
|
interface RecommendItemProps {
|
|
extra: Extra;
|
|
index: number;
|
|
icons: Icon[];
|
|
className?: string;
|
|
}
|
|
|
|
const RecommendItem: React.FC<RecommendItemProps> = ({ extra, index, icons, className }) => {
|
|
const { styles } = useStyle();
|
|
|
|
if (!extra) {
|
|
return <Skeleton key={index} />;
|
|
}
|
|
const icon = icons.find((i) => i.name === extra.source);
|
|
|
|
const card = (
|
|
<a
|
|
key={extra?.title}
|
|
href={extra.href}
|
|
target="_blank"
|
|
className={classNames(styles.itemBase, className)}
|
|
rel="noreferrer"
|
|
>
|
|
<Typography.Title level={5}>{extra?.title}</Typography.Title>
|
|
<Typography.Paragraph type="secondary" style={{ flex: 'auto' }}>
|
|
{extra.description}
|
|
</Typography.Paragraph>
|
|
<Flex justify="space-between" align="center">
|
|
<Typography.Text>{extra.date}</Typography.Text>
|
|
{icon && <img src={icon.href} draggable={false} className={styles.bannerBg} alt="banner" />}
|
|
</Flex>
|
|
</a>
|
|
);
|
|
|
|
if (index === 0) {
|
|
return (
|
|
<Badge.Ribbon text="HOT" color="red" rootClassName={styles.ribbon}>
|
|
{card}
|
|
</Badge.Ribbon>
|
|
);
|
|
}
|
|
|
|
return card;
|
|
};
|
|
|
|
export const BannerRecommendsFallback: React.FC = () => {
|
|
const { isMobile } = useContext(SiteContext);
|
|
const { styles } = useStyle();
|
|
|
|
const list = new Array(3).fill(1);
|
|
|
|
return isMobile ? (
|
|
<Carousel className={styles.carousel}>
|
|
{list.map((_, index) => (
|
|
<div key={index} className={styles.itemBase}>
|
|
<Skeleton active style={{ padding: '0 24px' }} />
|
|
</div>
|
|
))}
|
|
</Carousel>
|
|
) : (
|
|
<div className={styles.container}>
|
|
{list.map((_, index) => (
|
|
<div key={index} className={styles.itemBase}>
|
|
<Skeleton active />
|
|
</div>
|
|
))}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const BannerRecommends: React.FC = () => {
|
|
const { styles } = useStyle();
|
|
const [, lang] = useLocale();
|
|
const { isMobile } = React.useContext(SiteContext);
|
|
const data = useSiteData();
|
|
const extras = data?.extras?.[lang];
|
|
const icons = data?.icons || [];
|
|
const first3 = !extras || extras.length === 0 ? new Array(3).fill(null) : extras.slice(0, 3);
|
|
|
|
if (!data) {
|
|
return <BannerRecommendsFallback />;
|
|
}
|
|
|
|
if (isMobile) {
|
|
return (
|
|
<Carousel className={styles.carousel}>
|
|
{first3.map((extra, index) => (
|
|
<div key={index}>
|
|
<RecommendItem
|
|
extra={extra}
|
|
index={index}
|
|
icons={icons}
|
|
className={styles.sliderItem}
|
|
/>
|
|
</div>
|
|
))}
|
|
</Carousel>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className={styles.container}>
|
|
{first3.map((extra, index) => (
|
|
<RecommendItem
|
|
extra={extra}
|
|
index={index}
|
|
icons={icons}
|
|
className={styles.cardItem}
|
|
key={index}
|
|
/>
|
|
))}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default BannerRecommends;
|