mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-12 15:47:59 +08:00
198 lines
4.5 KiB
TypeScript
198 lines
4.5 KiB
TypeScript
/* eslint-disable global-require */
|
|
import React, { useMemo } from 'react';
|
|
import { HistoryOutlined } from '@ant-design/icons';
|
|
import { Button, Drawer, Timeline, Typography, Grid } from 'antd';
|
|
import { createStyles } from 'antd-style';
|
|
|
|
import useFetch from '../../../hooks/useFetch';
|
|
import useLocale from '../../../hooks/useLocale';
|
|
import Link from '../Link';
|
|
|
|
const useStyle = createStyles(({ token, css }) => ({
|
|
history: css`
|
|
position: absolute;
|
|
top: 0;
|
|
inset-inline-end: 0;
|
|
`,
|
|
|
|
li: css`
|
|
// white-space: pre;
|
|
`,
|
|
|
|
ref: css`
|
|
margin-left: ${token.marginXS}px;
|
|
`,
|
|
}));
|
|
|
|
export interface ComponentChangelogProps {
|
|
pathname: string;
|
|
}
|
|
|
|
const locales = {
|
|
cn: {
|
|
full: '完整更新日志',
|
|
changelog: '更新日志',
|
|
loading: '加载中...',
|
|
empty: '暂无更新',
|
|
},
|
|
en: {
|
|
full: 'Full Changelog',
|
|
changelog: 'Changelog',
|
|
loading: 'loading...',
|
|
empty: 'Nothing update',
|
|
},
|
|
};
|
|
|
|
function ParseChangelog(props: { changelog: string; refs: string[]; styles: any }) {
|
|
const { changelog = '', refs = [], styles } = props;
|
|
|
|
const parsedChangelog = React.useMemo(() => {
|
|
const nodes: React.ReactNode[] = [];
|
|
|
|
let isQuota = false;
|
|
let lastStr = '';
|
|
|
|
for (let i = 0; i < changelog.length; i += 1) {
|
|
const char = changelog[i];
|
|
|
|
if (char !== '`') {
|
|
lastStr += char;
|
|
} else {
|
|
let node: React.ReactNode = lastStr;
|
|
if (isQuota) {
|
|
node = <code>{node}</code>;
|
|
}
|
|
|
|
nodes.push(node);
|
|
lastStr = '';
|
|
isQuota = !isQuota;
|
|
}
|
|
}
|
|
|
|
nodes.push(lastStr);
|
|
|
|
return nodes;
|
|
}, [changelog]);
|
|
|
|
return (
|
|
<>
|
|
{/* Changelog */}
|
|
<span>{parsedChangelog}</span>
|
|
|
|
{/* Refs */}
|
|
{refs?.map((ref) => (
|
|
<a className={styles.ref} key={ref} href={ref} target="_blank" rel="noreferrer">
|
|
#{ref.match(/^.*\/(\d+)$/)?.[1]}
|
|
</a>
|
|
))}
|
|
</>
|
|
);
|
|
}
|
|
|
|
type ChangelogInfo = {
|
|
version: string;
|
|
changelog: string;
|
|
refs: string[];
|
|
};
|
|
|
|
function useChangelog(componentPath: string, lang: 'cn' | 'en'): ChangelogInfo[] {
|
|
const data: any = useFetch(
|
|
lang === 'cn'
|
|
? {
|
|
key: 'component-changelog-cn',
|
|
request: () => import('../../../preset/components-changelog-cn.json'),
|
|
}
|
|
: {
|
|
key: 'component-changelog-en',
|
|
request: () => import('../../../preset/components-changelog-en.json'),
|
|
},
|
|
);
|
|
|
|
return useMemo(() => {
|
|
const component = componentPath.replace(/-/g, '');
|
|
|
|
const componentName = Object.keys(data).find(
|
|
(name) => name.toLowerCase() === component.toLowerCase(),
|
|
);
|
|
|
|
return data[componentName!];
|
|
}, [data, componentPath]);
|
|
}
|
|
|
|
export default function ComponentChangelog(props: ComponentChangelogProps) {
|
|
const { pathname = '' } = props;
|
|
const [locale, lang] = useLocale(locales);
|
|
const [show, setShow] = React.useState(false);
|
|
|
|
const { styles } = useStyle();
|
|
|
|
const componentPath = pathname.match(/\/components\/([^/]+)/)?.[1] || '';
|
|
|
|
const list = useChangelog(componentPath, lang);
|
|
|
|
const timelineItems = React.useMemo(() => {
|
|
const changelogMap: Record<string, ChangelogInfo[]> = {};
|
|
|
|
list?.forEach((info) => {
|
|
changelogMap[info.version] = changelogMap[info.version] || [];
|
|
changelogMap[info.version].push(info);
|
|
});
|
|
|
|
return Object.keys(changelogMap).map((version) => {
|
|
const changelogList = changelogMap[version];
|
|
|
|
return {
|
|
children: (
|
|
<Typography>
|
|
<h4>{version}</h4>
|
|
<ul>
|
|
{changelogList.map((info, index) => (
|
|
<li key={index} className={styles.li}>
|
|
<ParseChangelog {...info} styles={styles} />
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</Typography>
|
|
),
|
|
};
|
|
});
|
|
}, [list]);
|
|
|
|
const screens = Grid.useBreakpoint();
|
|
const width = screens.md ? '48vw' : '90vw';
|
|
|
|
if (!list || !list.length) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<Button
|
|
className={styles.history}
|
|
icon={<HistoryOutlined />}
|
|
onClick={() => {
|
|
setShow(true);
|
|
}}
|
|
>
|
|
{locale.changelog}
|
|
</Button>
|
|
<Drawer
|
|
title={locale.changelog}
|
|
extra={
|
|
<Link style={{ fontSize: 14 }} to={`/changelog${lang === 'cn' ? '-cn' : ''}`}>
|
|
{locale.full}
|
|
</Link>
|
|
}
|
|
open={show}
|
|
width={width}
|
|
onClose={() => {
|
|
setShow(false);
|
|
}}
|
|
destroyOnClose
|
|
>
|
|
<Timeline items={timelineItems} />
|
|
</Drawer>
|
|
</>
|
|
);
|
|
}
|