mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-02 06:50:44 +08:00
267 lines
8.7 KiB
TypeScript
267 lines
8.7 KiB
TypeScript
import React, { memo, useContext, useMemo, useRef, useState } from 'react';
|
|
import type { CSSProperties } from 'react';
|
|
import { SearchOutlined } from '@ant-design/icons';
|
|
import { Affix, Card, Col, Divider, Flex, Input, Row, Tag, Typography } from 'antd';
|
|
import { createStyles, useTheme } from 'antd-style';
|
|
import { useIntl, useLocation, useSidebarData } from 'dumi';
|
|
import debounce from 'lodash/debounce';
|
|
import scrollIntoView from 'scroll-into-view-if-needed';
|
|
|
|
import Link from '../../common/Link';
|
|
import SiteContext from '../../slots/SiteContext';
|
|
import type { Component } from './ProComponentsList';
|
|
import proComponentsList from './ProComponentsList';
|
|
|
|
const useStyle = createStyles(({ token, css }) => ({
|
|
componentsOverviewGroupTitle: css`
|
|
margin-bottom: ${token.marginLG}px !important;
|
|
`,
|
|
componentsOverviewTitle: css`
|
|
overflow: hidden;
|
|
color: ${token.colorTextHeading};
|
|
text-overflow: ellipsis;
|
|
`,
|
|
componentsOverviewImg: css`
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
height: 152px;
|
|
`,
|
|
componentsOverviewCard: css`
|
|
cursor: pointer;
|
|
transition: all 0.5s;
|
|
&:hover {
|
|
box-shadow:
|
|
0 6px 16px -8px #00000014,
|
|
0 9px 28px #0000000d,
|
|
0 12px 48px 16px #00000008;
|
|
}
|
|
`,
|
|
componentsOverviewAffix: css`
|
|
display: flex;
|
|
transition: all ${token.motionDurationSlow};
|
|
justify-content: space-between;
|
|
`,
|
|
componentsOverviewSearch: css`
|
|
padding: 0;
|
|
box-shadow: none !important;
|
|
.anticon-search {
|
|
color: ${token.colorTextDisabled};
|
|
}
|
|
`,
|
|
componentsOverviewContent: css`
|
|
&:empty:after {
|
|
display: block;
|
|
padding: ${token.padding}px 0 ${token.paddingMD * 2}px;
|
|
color: ${token.colorTextDisabled};
|
|
text-align: center;
|
|
border-bottom: 1px solid ${token.colorSplit};
|
|
content: 'Not Found';
|
|
}
|
|
`,
|
|
}));
|
|
|
|
const onClickCard = (pathname: string) => {
|
|
if (window.gtag) {
|
|
window.gtag('event', '点击', {
|
|
event_category: '组件总览卡片',
|
|
event_label: pathname,
|
|
});
|
|
}
|
|
};
|
|
|
|
const reportSearch = debounce<(value: string) => void>((value) => {
|
|
if (window.gtag) {
|
|
window.gtag('event', '搜索', {
|
|
event_category: '组件总览卡片',
|
|
event_label: value,
|
|
});
|
|
}
|
|
}, 2000);
|
|
|
|
const { Title } = Typography;
|
|
|
|
const Overview: React.FC = () => {
|
|
const { styles } = useStyle();
|
|
const { theme } = useContext(SiteContext);
|
|
|
|
const data = useSidebarData();
|
|
const [searchBarAffixed, setSearchBarAffixed] = useState<boolean>(false);
|
|
|
|
const token = useTheme();
|
|
const { borderRadius, colorBgContainer, fontSizeXL, anchorTop } = token;
|
|
|
|
const affixedStyle: CSSProperties = {
|
|
boxShadow: 'rgba(50, 50, 93, 0.25) 0 6px 12px -2px, rgba(0, 0, 0, 0.3) 0 3px 7px -3px',
|
|
padding: 8,
|
|
margin: -8,
|
|
borderRadius,
|
|
backgroundColor: colorBgContainer,
|
|
};
|
|
|
|
const { search: urlSearch } = useLocation();
|
|
const { locale, formatMessage } = useIntl();
|
|
|
|
const [search, setSearch] = useState<string>(() => {
|
|
const params = new URLSearchParams(urlSearch);
|
|
if (params.has('s')) {
|
|
return params.get('s') || '';
|
|
}
|
|
return '';
|
|
});
|
|
|
|
const sectionRef = useRef<HTMLElement>(null);
|
|
|
|
const onKeyDown: React.KeyboardEventHandler<HTMLInputElement> = (event) => {
|
|
if (event.keyCode === 13 && search.trim().length) {
|
|
sectionRef.current?.querySelector<HTMLElement>(`.${styles.componentsOverviewCard}`)?.click();
|
|
}
|
|
};
|
|
|
|
const groups = useMemo<{ title: string; children: Component[] }[]>(
|
|
() =>
|
|
data
|
|
.filter((item) => item?.title)
|
|
.map<{ title: string; children: Component[] }>((item) => ({
|
|
title: item?.title || '',
|
|
children: item.children.map((child) => ({
|
|
title: child.frontmatter?.title || '',
|
|
subtitle: child.frontmatter?.subtitle,
|
|
cover: child.frontmatter?.cover,
|
|
coverDark: child.frontmatter?.coverDark,
|
|
link: child.link,
|
|
})),
|
|
}))
|
|
.concat([
|
|
{
|
|
title: locale === 'zh-CN' ? '重型组件' : 'Others',
|
|
children:
|
|
locale === 'zh-CN'
|
|
? proComponentsList
|
|
: proComponentsList.map((component) => ({ ...component, subtitle: '' })),
|
|
},
|
|
]),
|
|
[data, locale],
|
|
);
|
|
return (
|
|
<section className="markdown" ref={sectionRef}>
|
|
<Divider />
|
|
<Affix offsetTop={anchorTop} onChange={(affixed) => setSearchBarAffixed(!!affixed)}>
|
|
<div
|
|
className={styles.componentsOverviewAffix}
|
|
style={searchBarAffixed ? affixedStyle : {}}
|
|
>
|
|
<Input
|
|
autoFocus
|
|
value={search}
|
|
placeholder={formatMessage({ id: 'app.components.overview.search' })}
|
|
className={styles.componentsOverviewSearch}
|
|
onChange={(e) => {
|
|
setSearch(e.target.value);
|
|
reportSearch(e.target.value);
|
|
if (sectionRef.current && searchBarAffixed) {
|
|
scrollIntoView(sectionRef.current, {
|
|
scrollMode: 'if-needed',
|
|
block: 'start',
|
|
behavior: (actions) =>
|
|
actions.forEach(({ el, top }) => {
|
|
el.scrollTop = top - 64;
|
|
}),
|
|
});
|
|
}
|
|
}}
|
|
onKeyDown={onKeyDown}
|
|
variant="borderless"
|
|
suffix={<SearchOutlined />}
|
|
style={{ fontSize: searchBarAffixed ? fontSizeXL - 2 : fontSizeXL }}
|
|
/>
|
|
</div>
|
|
</Affix>
|
|
<Divider />
|
|
<div className={styles.componentsOverviewContent}>
|
|
{groups
|
|
.filter((i) => i?.title)
|
|
.map((group) => {
|
|
const components = group?.children?.filter(
|
|
(component) =>
|
|
!search.trim() ||
|
|
component?.title?.toLowerCase()?.includes(search.trim().toLowerCase()) ||
|
|
(component?.subtitle || '').toLowerCase().includes(search.trim().toLowerCase()),
|
|
);
|
|
return components?.length ? (
|
|
<div key={group?.title}>
|
|
<Title level={2} className={styles.componentsOverviewGroupTitle}>
|
|
<Flex gap="small" align="center">
|
|
<span style={{ fontSize: 24 }}>{group?.title}</span>
|
|
<Tag style={{ display: 'block' }}>{components.length}</Tag>
|
|
</Flex>
|
|
</Title>
|
|
<Row gutter={[24, 24]}>
|
|
{components.map((component) => {
|
|
let url = component.link;
|
|
/** 是否是外链 */
|
|
const isExternalLink = url.startsWith('http');
|
|
|
|
if (!isExternalLink) {
|
|
url += urlSearch;
|
|
}
|
|
|
|
const cardContent = (
|
|
<Card
|
|
key={component.title}
|
|
onClick={() => onClickCard(url)}
|
|
styles={{
|
|
body: {
|
|
backgroundRepeat: 'no-repeat',
|
|
backgroundPosition: 'bottom right',
|
|
backgroundImage: `url(${component.tag || ''})`,
|
|
},
|
|
}}
|
|
size="small"
|
|
className={styles.componentsOverviewCard}
|
|
title={
|
|
<div className={styles.componentsOverviewTitle}>
|
|
{component.title} {component.subtitle}
|
|
</div>
|
|
}
|
|
>
|
|
<div className={styles.componentsOverviewImg}>
|
|
<img
|
|
src={
|
|
theme.includes('dark') && component.coverDark
|
|
? component.coverDark
|
|
: component.cover
|
|
}
|
|
alt={component.title}
|
|
/>
|
|
</div>
|
|
</Card>
|
|
);
|
|
|
|
const linkContent = isExternalLink ? (
|
|
<a href={url} key={component.title}>
|
|
{cardContent}
|
|
</a>
|
|
) : (
|
|
<Link to={url} prefetch key={component.title}>
|
|
{cardContent}
|
|
</Link>
|
|
);
|
|
|
|
return (
|
|
<Col xs={24} sm={12} lg={8} xl={6} key={component.title}>
|
|
{linkContent}
|
|
</Col>
|
|
);
|
|
})}
|
|
</Row>
|
|
</div>
|
|
) : null;
|
|
})}
|
|
</div>
|
|
</section>
|
|
);
|
|
};
|
|
|
|
export default memo(Overview);
|