mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-21 00:14:44 +08:00
ae53a452f2
* docs: add help text for token customization * docs: add help text for token customization * docs: add help text for token customization * docs: add help text for token customization
239 lines
5.9 KiB
TypeScript
239 lines
5.9 KiB
TypeScript
import { RightOutlined, LinkOutlined, QuestionCircleOutlined } from '@ant-design/icons';
|
|
import { createStyles, css, useTheme } from 'antd-style';
|
|
import { getDesignToken } from 'antd-token-previewer';
|
|
import React, { useMemo, useState } from 'react';
|
|
import tokenMeta from 'antd/es/version/token-meta.json';
|
|
import tokenData from 'antd/es/version/token.json';
|
|
import { ConfigProvider, Table, Popover, Typography } from 'antd';
|
|
import useLocale from '../../../hooks/useLocale';
|
|
import { useColumns } from '../TokenTable';
|
|
|
|
const defaultToken = getDesignToken();
|
|
|
|
const locales = {
|
|
cn: {
|
|
token: 'Token 名称',
|
|
description: '描述',
|
|
type: '类型',
|
|
value: '默认值',
|
|
componentToken: '组件 Token',
|
|
globalToken: '全局 Token',
|
|
help: '如何定制?',
|
|
customizeTokenLink: '/docs/react/customize-theme-cn#修改主题变量',
|
|
customizeComponentTokenLink: '/docs/react/customize-theme-cn#修改组件变量',
|
|
},
|
|
en: {
|
|
token: 'Token Name',
|
|
description: 'Description',
|
|
type: 'Type',
|
|
value: 'Default Value',
|
|
componentToken: 'Component Token',
|
|
globalToken: 'Global Token',
|
|
help: 'How to use?',
|
|
customizeTokenLink: '/docs/react/customize-theme#customize-design-token',
|
|
customizeComponentTokenLink: 'docs/react/customize-theme#customize-component-token',
|
|
},
|
|
};
|
|
|
|
const useStyle = createStyles(() => ({
|
|
tableTitle: css`
|
|
cursor: pointer;
|
|
position: relative;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: flex-start;
|
|
line-height: 40px;
|
|
`,
|
|
arrowIcon: css`
|
|
font-size: 16px;
|
|
margin-right: 8px;
|
|
& svg {
|
|
transition: all 0.3s;
|
|
}
|
|
`,
|
|
help: css`
|
|
margin-left: 8px;
|
|
font-size: 12px;
|
|
font-weight: normal;
|
|
color: #999;
|
|
a {
|
|
color: #999;
|
|
}
|
|
`,
|
|
}));
|
|
|
|
interface SubTokenTableProps {
|
|
defaultOpen?: boolean;
|
|
title: string;
|
|
helpText: React.ReactNode;
|
|
helpLink: string;
|
|
tokens: string[];
|
|
component?: string;
|
|
}
|
|
|
|
const SubTokenTable: React.FC<SubTokenTableProps> = ({
|
|
defaultOpen,
|
|
tokens,
|
|
title,
|
|
helpText,
|
|
helpLink,
|
|
component,
|
|
}) => {
|
|
const [, lang] = useLocale(locales);
|
|
const token = useTheme();
|
|
const columns = useColumns();
|
|
|
|
const [open, setOpen] = useState<boolean>(defaultOpen || process.env.NODE_ENV !== 'production');
|
|
|
|
const { styles } = useStyle();
|
|
|
|
if (!tokens.length) {
|
|
return null;
|
|
}
|
|
|
|
const data = tokens
|
|
.sort(
|
|
component
|
|
? undefined
|
|
: (token1, token2) => {
|
|
const hasColor1 = token1.toLowerCase().includes('color');
|
|
const hasColor2 = token2.toLowerCase().includes('color');
|
|
|
|
if (hasColor1 && !hasColor2) {
|
|
return -1;
|
|
}
|
|
|
|
if (!hasColor1 && hasColor2) {
|
|
return 1;
|
|
}
|
|
|
|
return token1 < token2 ? -1 : 1;
|
|
},
|
|
)
|
|
.map((name) => {
|
|
const meta = component
|
|
? tokenMeta.components[component].find((item) => item.token === name)
|
|
: tokenMeta.global[name];
|
|
|
|
if (!meta) {
|
|
return null;
|
|
}
|
|
|
|
return {
|
|
name,
|
|
desc: lang === 'cn' ? meta.desc : meta.descEn,
|
|
type: meta.type,
|
|
value: component ? tokenData[component].component[name] : defaultToken[name],
|
|
};
|
|
})
|
|
.filter(Boolean);
|
|
|
|
const code = component
|
|
? `<ConfigProvider
|
|
theme={{
|
|
components: {
|
|
${component}: {
|
|
/* here is your component tokens */
|
|
},
|
|
},
|
|
}}
|
|
>
|
|
...
|
|
</ConfigProvider>`
|
|
: `<ConfigProvider
|
|
theme={{
|
|
token: {
|
|
/* here is your global tokens */
|
|
},
|
|
}}
|
|
>
|
|
...
|
|
</ConfigProvider>`;
|
|
|
|
return (
|
|
<>
|
|
<div className={styles.tableTitle} onClick={() => setOpen(!open)}>
|
|
<RightOutlined className={styles.arrowIcon} rotate={open ? 90 : 0} />
|
|
<h3>
|
|
{title}
|
|
<Popover
|
|
title={null}
|
|
popupStyle={{ width: 400 }}
|
|
content={
|
|
<Typography>
|
|
<pre style={{ fontSize: 12 }}>{code}</pre>
|
|
<a href={helpLink} target="_blank" rel="noreferrer">
|
|
<LinkOutlined style={{ marginRight: 4 }} />
|
|
{helpText}
|
|
</a>
|
|
</Typography>
|
|
}
|
|
>
|
|
<span className={styles.help}>
|
|
<QuestionCircleOutlined style={{ marginRight: 3 }} />
|
|
{helpText}
|
|
</span>
|
|
</Popover>
|
|
</h3>
|
|
</div>
|
|
{open && (
|
|
<ConfigProvider theme={{ token: { borderRadius: 0 } }}>
|
|
<Table
|
|
size="middle"
|
|
columns={columns}
|
|
bordered
|
|
dataSource={data}
|
|
style={{ marginBottom: token.margin }}
|
|
pagination={false}
|
|
rowKey={(record) => record.name}
|
|
/>
|
|
</ConfigProvider>
|
|
)}
|
|
</>
|
|
);
|
|
};
|
|
|
|
export interface ComponentTokenTableProps {
|
|
component: string;
|
|
}
|
|
|
|
const ComponentTokenTable: React.FC<ComponentTokenTableProps> = ({ component }) => {
|
|
const [locale] = useLocale(locales);
|
|
const [mergedGlobalTokens] = useMemo(() => {
|
|
const globalTokenSet = new Set<string>();
|
|
|
|
component.split(',').forEach((comp) => {
|
|
const { global: globalTokens = [] } = tokenData[comp] || {};
|
|
|
|
globalTokens.forEach((token: string) => {
|
|
globalTokenSet.add(token);
|
|
});
|
|
});
|
|
|
|
return [Array.from(globalTokenSet)] as const;
|
|
}, [component]);
|
|
|
|
return (
|
|
<>
|
|
{tokenMeta.components[component] && (
|
|
<SubTokenTable
|
|
title={locale.componentToken}
|
|
helpText={locale.help}
|
|
helpLink={locale.customizeTokenLink}
|
|
tokens={tokenMeta.components[component].map((item) => item.token)}
|
|
component={component}
|
|
defaultOpen
|
|
/>
|
|
)}
|
|
<SubTokenTable
|
|
title={locale.globalToken}
|
|
helpText={locale.help}
|
|
helpLink={locale.customizeComponentTokenLink}
|
|
tokens={mergedGlobalTokens}
|
|
/>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default React.memo(ComponentTokenTable);
|