mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-12 15:47:59 +08:00
e8fa5938ab
* site: use CSS logical properties * Update components/anchor/demo/targetOffset.tsx Co-authored-by: afc163 <afc163@gmail.com> Signed-off-by: lijianan <574980606@qq.com> * fix: fix * fix: fix * chore: fix * fix: add more * fix: add more --------- Signed-off-by: lijianan <574980606@qq.com> Co-authored-by: afc163 <afc163@gmail.com>
194 lines
5.0 KiB
TypeScript
194 lines
5.0 KiB
TypeScript
import type { ComponentProps } from 'react';
|
|
import React, { useContext, useEffect, useMemo } from 'react';
|
|
import { Button, Tabs, Typography } from 'antd';
|
|
import { createStyles } from 'antd-style';
|
|
import toReactElement from 'jsonml-to-react-element';
|
|
import JsonML from 'jsonml.js/lib/utils';
|
|
import Prism from 'prismjs';
|
|
|
|
import DemoContext from '../slots/DemoContext';
|
|
import LiveCode from './LiveCode';
|
|
|
|
const useStyle = createStyles(({ token, css }) => {
|
|
const { colorIcon, antCls } = token;
|
|
|
|
return {
|
|
code: css`
|
|
position: relative;
|
|
margin-top: -${token.margin}px;
|
|
`,
|
|
|
|
copyButton: css`
|
|
color: ${colorIcon};
|
|
position: absolute;
|
|
z-index: 2;
|
|
top: 16px;
|
|
inset-inline-end: ${token.padding}px;
|
|
width: 32px;
|
|
text-align: center;
|
|
padding: 0;
|
|
`,
|
|
|
|
copyIcon: css`
|
|
${antCls}-typography-copy {
|
|
position: relative;
|
|
margin-inline-start: 0;
|
|
|
|
// expand clickable area
|
|
&::before {
|
|
content: '';
|
|
display: block;
|
|
position: absolute;
|
|
top: -5px;
|
|
inset-inline-start: -9px;
|
|
bottom: -5px;
|
|
inset-inline-end: -9px;
|
|
}
|
|
}
|
|
${antCls}-typography-copy:not(${antCls}-typography-copy-success) {
|
|
color: ${colorIcon};
|
|
|
|
&:hover {
|
|
color: ${colorIcon};
|
|
}
|
|
}
|
|
`,
|
|
};
|
|
});
|
|
|
|
const LANGS = {
|
|
tsx: 'TypeScript',
|
|
jsx: 'JavaScript',
|
|
style: 'CSS',
|
|
};
|
|
|
|
interface CodePreviewProps
|
|
extends Omit<ComponentProps<typeof LiveCode>, 'initialValue' | 'lang' | 'onChange'> {
|
|
sourceCode?: string;
|
|
jsxCode?: string;
|
|
styleCode?: string;
|
|
entryName: string;
|
|
onSourceChange?: (source: Record<string, string>) => void;
|
|
}
|
|
|
|
function toReactComponent(jsonML: any[]) {
|
|
return toReactElement(jsonML, [
|
|
[
|
|
(node: any) => JsonML.isElement(node) && JsonML.getTagName(node) === 'pre',
|
|
(node: any, index: number) => {
|
|
const attr = JsonML.getAttributes(node);
|
|
return (
|
|
<pre key={index} className={`language-${attr.lang}`}>
|
|
{/* biome-ignore lint/security/noDangerouslySetInnerHtml: it's for markdown */}
|
|
<code dangerouslySetInnerHTML={{ __html: attr.highlighted }} />
|
|
</pre>
|
|
);
|
|
},
|
|
],
|
|
]);
|
|
}
|
|
|
|
const CodePreview: React.FC<CodePreviewProps> = ({
|
|
sourceCode = '',
|
|
jsxCode = '',
|
|
styleCode = '',
|
|
entryName,
|
|
onSourceChange,
|
|
error,
|
|
}) => {
|
|
// 避免 Tabs 数量不稳定的闪动问题
|
|
const initialCodes: Partial<Record<'tsx' | 'jsx' | 'style', string>> = {};
|
|
if (sourceCode) {
|
|
initialCodes.tsx = '';
|
|
}
|
|
if (jsxCode) {
|
|
initialCodes.jsx = '';
|
|
}
|
|
if (styleCode) {
|
|
initialCodes.style = '';
|
|
}
|
|
const [highlightedCodes, setHighlightedCodes] = React.useState(initialCodes);
|
|
const { codeType, setCodeType } = useContext(DemoContext);
|
|
const sourceCodes = {
|
|
// omit trailing line break
|
|
tsx: sourceCode?.trim(),
|
|
jsx: jsxCode?.trim(),
|
|
style: styleCode?.trim(),
|
|
} as Record<'tsx' | 'jsx' | 'style', string>;
|
|
useEffect(() => {
|
|
const codes = {
|
|
tsx: Prism.highlight(sourceCode, Prism.languages.javascript, 'jsx'),
|
|
jsx: Prism.highlight(jsxCode, Prism.languages.javascript, 'jsx'),
|
|
style: Prism.highlight(styleCode, Prism.languages.css, 'css'),
|
|
};
|
|
// 去掉空的代码类型
|
|
(Object.keys(codes) as (keyof typeof codes)[]).forEach((key) => {
|
|
if (!codes[key]) {
|
|
delete codes[key];
|
|
}
|
|
});
|
|
setHighlightedCodes(codes);
|
|
}, [jsxCode, sourceCode, styleCode]);
|
|
|
|
const langList = Object.keys(highlightedCodes) as ('tsx' | 'jsx' | 'style')[];
|
|
|
|
const { styles } = useStyle();
|
|
|
|
const items = useMemo(
|
|
() =>
|
|
langList.map((lang: keyof typeof LANGS) => ({
|
|
label: LANGS[lang],
|
|
key: lang,
|
|
children: (
|
|
<div className={styles.code}>
|
|
{lang === 'tsx' ? (
|
|
<LiveCode
|
|
error={error}
|
|
lang={lang}
|
|
initialValue={sourceCodes[lang]}
|
|
onChange={(code: string) => {
|
|
onSourceChange?.({ [entryName]: code });
|
|
}}
|
|
/>
|
|
) : (
|
|
toReactComponent(['pre', { lang, highlighted: highlightedCodes[lang] }])
|
|
)}
|
|
<Button type="text" className={styles.copyButton}>
|
|
<Typography.Text className={styles.copyIcon} copyable={{ text: sourceCodes[lang] }} />
|
|
</Button>
|
|
</div>
|
|
),
|
|
})),
|
|
[JSON.stringify(highlightedCodes), styles.code, styles.copyButton, styles.copyIcon],
|
|
);
|
|
|
|
if (!langList.length) {
|
|
return null;
|
|
}
|
|
|
|
if (langList.length === 1) {
|
|
return (
|
|
<LiveCode
|
|
error={error}
|
|
lang={langList[0]}
|
|
initialValue={sourceCodes[langList[0]]}
|
|
onChange={(code: string) => {
|
|
onSourceChange?.({ [entryName]: code });
|
|
}}
|
|
/>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<Tabs
|
|
centered
|
|
className="highlight"
|
|
activeKey={codeType}
|
|
onChange={setCodeType}
|
|
items={items}
|
|
/>
|
|
);
|
|
};
|
|
|
|
export default CodePreview;
|