-
Notifications
You must be signed in to change notification settings - Fork 3.1k
/
Copy pathcode.js
141 lines (128 loc) · 3.98 KB
/
code.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
import React from 'react'
import {Box, Text, Button, themeGet} from '@primer/react'
import {Octicon} from '@primer/react/deprecated'
import {Highlight, Prism} from 'prism-react-renderer'
import styled from 'styled-components'
import {CheckIcon, CopyIcon} from '@primer/octicons-react'
import copyToClipboard from 'copy-to-clipboard'
import {announce} from '../util/aria-live'
import {usePrismTheme} from '../hooks/use-prism-theme'
;(typeof global !== 'undefined' ? global : window).Prism = Prism
require('prismjs/components/prism-bash')
const ClipboardCopy = ({value, ...props}) => {
const [copied, setCopied] = React.useState(false)
React.useEffect(() => {
const timeout = setTimeout(() => {
if (copied) {
setCopied(false)
}
}, 1000)
return () => clearTimeout(timeout)
}, [copied])
return (
<Button
{...props}
aria-label="Copy to clipboard"
onClick={() => {
copyToClipboard(value)
setCopied(true)
announce(`Copied to clipboard`)
}}
>
<Octicon icon={copied ? CheckIcon : CopyIcon} sx={{color: copied ? 'success.fg' : 'fg.muted'}} />
</Button>
)
}
export const InlineCode = styled.code`
padding: 0.2em 0.4em;
font-family: ${themeGet('fonts.mono')};
font-size: 85%;
background-color: ${themeGet('colors.neutral.muted')};
border-radius: ${themeGet('radii.2')};
`
const colorMap = {
'token comment': '#747458',
'token function': '#cf3846',
'token parameter variable': '#277d7b',
'token assign-left variable': '#277d7b',
}
const MonoText = props => <Text sx={{fontFamily: 'mono', fontSize: 1}} {...props} />
const CodeBlock = ({children, code, className, style}) => (
<Box
sx={{
// Make <pre> adjust to the width of the container
// https://stackoverflow.com/a/14406386
display: 'table',
tableLayout: 'fixed',
width: '100%',
mb: 3,
}}
>
<Box
style={style}
sx={{
...(code ? {display: 'flex', justifyContent: 'space-between', flexDirection: 'row-reverse'} : {}),
borderRadius: 2,
borderStyle: 'solid',
borderWidth: 1,
borderColor: 'border.muted',
}}
>
{code ? (
<ClipboardCopy
value={code}
sx={{
borderRadius: 0,
borderStyle: 'solid',
borderWidth: 1,
borderColor: 'border.muted',
marginTop: '-1px',
marginRight: '-1px',
borderTopRightRadius: 2,
borderBottomLeftRadius: 2,
}}
/>
) : null}
<Box sx={{m: 0, p: 3, overflowX: 'auto'}}>
<Box as="pre" className={className} tabIndex={0} sx={{m: 0}}>
{children}
</Box>
</Box>
</Box>
</Box>
)
function Code({className = '', prompt, children}) {
const {theme: codeTheme} = usePrismTheme()
if (prompt) {
return (
<CodeBlock style={codeTheme.plain}>
<MonoText>{children}</MonoText>
</CodeBlock>
)
}
const code = children.trim()
const isBlock = className.startsWith('language-') || code.includes('\n')
if (!isBlock) {
return <InlineCode className={className}>{code}</InlineCode>
}
return (
<Highlight code={code} language={className.replace(/language-/, '') || 'bash'} theme={codeTheme}>
{({className: highlightClassName, style, tokens, getLineProps, getTokenProps}) => (
<CodeBlock className={highlightClassName} style={style} code={code}>
{tokens.map((line, i) => (
<Box key={i} {...getLineProps({line, key: i})}>
{line.map((token, key) => {
const tokenProps = getTokenProps({token, key})
const tokenStyle = colorMap[tokenProps.className]
? {...tokenProps.style, color: colorMap[tokenProps.className]}
: tokenProps.style
return <MonoText key={key} {...tokenProps} style={tokenStyle} />
})}
</Box>
))}
</CodeBlock>
)}
</Highlight>
)
}
export default Code