200 lines
6.3 KiB
JavaScript
200 lines
6.3 KiB
JavaScript
/**
|
|
* Copyright (c) 2017-present, Facebook, Inc.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
import React, { useEffect, useState, useRef } from 'react';
|
|
import classnames from 'classnames';
|
|
import Highlight, { defaultProps } from 'prism-react-renderer';
|
|
|
|
const { Prism } = require("prism-react-renderer");
|
|
|
|
Prism.languages = {
|
|
...Prism.languages,
|
|
pascaligo: {
|
|
'comment': [
|
|
/\(\*[\s\S]+?\*\)/,
|
|
// /\{[\s\S]+?\}/,
|
|
/\/\/.*/
|
|
],
|
|
'string': {
|
|
pattern: /(?:'(?:''|[^'\r\n])*'|#[&$%]?[a-f\d]+)+|\^[a-z]/i,
|
|
greedy: true
|
|
},
|
|
'keyword': [
|
|
{
|
|
// Turbo Pascal
|
|
pattern: /(^|[^&])\b(?:absolute|array|asm|begin|case|const|constructor|destructor|do|downto|else|end|file|for|function|goto|if|implementation|inherited|inline|interface|label|nil|object|of|operator|packed|procedure|program|record|reintroduce|repeat|self|set|string|then|to|type|unit|until|uses|var|while|with)\b/i,
|
|
lookbehind: true
|
|
},
|
|
{
|
|
// Free Pascal
|
|
pattern: /(^|[^&])\b(?:dispose|exit|false|new|true)\b/i,
|
|
lookbehind: true
|
|
},
|
|
{
|
|
// Object Pascal
|
|
pattern: /(^|[^&])\b(?:class|dispinterface|except|exports|finalization|finally|initialization|inline|library|on|out|packed|property|raise|resourcestring|threadvar|try)\b/i,
|
|
lookbehind: true
|
|
},
|
|
{
|
|
// Modifiers
|
|
pattern: /(^|[^&])\b(?:absolute|abstract|alias|assembler|bitpacked|break|cdecl|continue|cppdecl|cvar|default|deprecated|dynamic|enumerator|experimental|export|external|far|far16|forward|generic|helper|implements|index|interrupt|iochecks|local|message|name|near|nodefault|noreturn|nostackframe|oldfpccall|otherwise|overload|override|pascal|platform|private|protected|public|published|read|register|reintroduce|result|safecall|saveregisters|softfloat|specialize|static|stdcall|stored|strict|unaligned|unimplemented|varargs|virtual|write)\b/i,
|
|
lookbehind: true
|
|
}
|
|
],
|
|
'number': [
|
|
// Hexadecimal, octal and binary
|
|
/(?:[&%]\d+|\$[a-f\d]+)/i,
|
|
// Decimal
|
|
/\b\d+(?:\.\d+)?(?:e[+-]?\d+)?/i
|
|
],
|
|
'operator': [
|
|
/\.\.|\*\*|:=|<[<=>]?|>[>=]?|[+\-*\/]=?|[@^=]/i,
|
|
{
|
|
pattern: /(^|[^&])\b(?:and|as|div|exclude|in|include|is|mod|not|or|shl|shr|xor)\b/,
|
|
lookbehind: true
|
|
}
|
|
],
|
|
'punctuation': /\(\.|\.\)|[()\[\]:;,.]/
|
|
},
|
|
reasonligo:
|
|
{
|
|
...Prism.languages.reason,
|
|
'comment': [
|
|
/(^|[^\\])\/\*[\s\S]*?\*\//,
|
|
/\(\*[\s\S]*?\*\)/,
|
|
/\/\/.*/
|
|
]
|
|
|
|
},
|
|
cameligo: {
|
|
...Prism.languages.ocaml,
|
|
'comment': [
|
|
/(^|[^\\])\/\*[\s\S]*?\*\//,
|
|
/\(\*[\s\S]*?\*\)/,
|
|
/\/\/.*/
|
|
]
|
|
}
|
|
};
|
|
import defaultTheme from 'prism-react-renderer/themes/palenight';
|
|
import Clipboard from 'clipboard';
|
|
import rangeParser from 'parse-numeric-range';
|
|
import useDocusaurusContext from '@docusaurus/useDocusaurusContext';
|
|
import useThemeContext from '@theme/hooks/useThemeContext';
|
|
|
|
import styles from './styles.module.css';
|
|
|
|
const highlightLinesRangeRegex = /{([\d,-]+)}/;
|
|
|
|
export default ({ children, className: languageClassName, metastring }) => {
|
|
const {
|
|
siteConfig: {
|
|
themeConfig: { prism = {} },
|
|
},
|
|
} = useDocusaurusContext();
|
|
|
|
const [showCopied, setShowCopied] = useState(false);
|
|
const [mounted, setMounted] = useState(false);
|
|
// The Prism theme on SSR is always the default theme but the site theme
|
|
// can be in a different mode. React hydration doesn't update DOM styles
|
|
// that come from SSR. Hence force a re-render after mounting to apply the
|
|
// current relevant styles. There will be a flash seen of the original
|
|
// styles seen using this current approach but that's probably ok. Fixing
|
|
// the flash will require changing the theming approach and is not worth it
|
|
// at this point.
|
|
useEffect(() => {
|
|
setMounted(true);
|
|
}, []);
|
|
|
|
const target = useRef(null);
|
|
const button = useRef(null);
|
|
let highlightLines = [];
|
|
|
|
const { isDarkTheme } = useThemeContext();
|
|
const lightModeTheme = prism.theme || defaultTheme;
|
|
const darkModeTheme = prism.darkTheme || lightModeTheme;
|
|
const prismTheme = isDarkTheme ? darkModeTheme : lightModeTheme;
|
|
|
|
if (metastring && highlightLinesRangeRegex.test(metastring)) {
|
|
const highlightLinesRange = metastring.match(highlightLinesRangeRegex)[1];
|
|
highlightLines = rangeParser.parse(highlightLinesRange).filter(n => n > 0);
|
|
}
|
|
|
|
useEffect(() => {
|
|
let clipboard;
|
|
|
|
if (button.current) {
|
|
clipboard = new Clipboard(button.current, {
|
|
target: () => target.current,
|
|
});
|
|
}
|
|
|
|
return () => {
|
|
if (clipboard) {
|
|
clipboard.destroy();
|
|
}
|
|
};
|
|
}, [button.current, target.current]);
|
|
|
|
let language =
|
|
languageClassName && languageClassName.replace(/language-/, '');
|
|
|
|
if (!language && prism.defaultLanguage) {
|
|
language = prism.defaultLanguage;
|
|
}
|
|
|
|
const handleCopyCode = () => {
|
|
window.getSelection().empty();
|
|
setShowCopied(true);
|
|
|
|
setTimeout(() => setShowCopied(false), 2000);
|
|
};
|
|
|
|
return (
|
|
<Highlight
|
|
{...defaultProps}
|
|
key={mounted}
|
|
theme={prismTheme}
|
|
code={children.trim()}
|
|
language={language}>
|
|
{({ className, style, tokens, getLineProps, getTokenProps }) => (
|
|
<pre className={classnames(className, styles.codeBlock)}>
|
|
<button
|
|
ref={button}
|
|
type="button"
|
|
aria-label="Copy code to clipboard"
|
|
className={styles.copyButton}
|
|
onClick={handleCopyCode}>
|
|
{showCopied ? 'Copied' : 'Copy'}
|
|
</button>
|
|
|
|
<code ref={target} className={styles.codeBlockLines}>
|
|
{tokens.map((line, i) => {
|
|
if (line.length === 1 && line[0].content === '') {
|
|
line[0].content = '\n'; // eslint-disable-line no-param-reassign
|
|
}
|
|
|
|
const lineProps = getLineProps({ line, key: i });
|
|
|
|
if (highlightLines.includes(i + 1)) {
|
|
lineProps.className = `${lineProps.className} docusaurus-highlight-code-line`;
|
|
}
|
|
|
|
return (
|
|
<div key={i} {...lineProps}>
|
|
{line.map((token, key) => (
|
|
<span key={key} {...getTokenProps({ token, key })} />
|
|
))}
|
|
</div>
|
|
);
|
|
})}
|
|
</code>
|
|
</pre>
|
|
)}
|
|
</Highlight>
|
|
);
|
|
};
|