1
1
import { useState , useEffect } from "preact/hooks" ;
2
- import CodeMirror from "@uiw/react-codemirror" ;
2
+ import CodeMirror , { EditorView } from "@uiw/react-codemirror" ;
3
3
import { javascript } from "@codemirror/lang-javascript" ;
4
4
5
5
import { CodeFrame } from "./frame" ;
6
+ import { CopyCodeButton } from "../CopyCodeButton" ;
7
+ import CircleButton from "../CircleButton" ;
8
+ import { Icon } from "../Icon" ;
6
9
/*
7
10
* A more featured code embed component that uses CodeMirror
8
11
*
9
12
* Props: {
10
13
* initialValue?: string;
11
14
* editable: boolean;
12
15
* previewable: boolean;
16
+ * previewHeight?: number;
17
+ * previewWidth?: number;
13
18
* }
14
19
*/
15
20
export const CodeEmbed = ( props ) => {
@@ -19,7 +24,10 @@ export const CodeEmbed = (props) => {
19
24
// instead of a normal one, but these break the code frame, so we replace them here.
20
25
// We also replace them in CodeFrame, but replacing here too ensures people don't
21
26
// accidentally copy-and-paste them out of the embedded editor.
22
- const [ codeString , setCodeString ] = useState ( initialCode . replace ( / \u00A0 / g, ' ' ) ) ;
27
+ const [ codeString , setCodeString ] = useState (
28
+ initialCode . replace ( / \u00A0 / g, " " ) ,
29
+ ) ;
30
+
23
31
const [ previewCodeString , setPreviewCodeString ] = useState ( codeString ) ;
24
32
25
33
useEffect ( ( ) => {
@@ -29,53 +37,72 @@ export const CodeEmbed = (props) => {
29
37
if ( ! rendered ) return < div className = "code-placeholder" /> ;
30
38
31
39
return (
32
- < div className = "mb-md flex w-full flex-col overflow-hidden md :flex-row" >
40
+ < div className = "mb-md flex w-full flex-col overflow-hidden lg :flex-row" >
33
41
{ props . previewable ? (
34
- < div >
35
- < CodeFrame jsCode = { previewCodeString } width = { 150 } height = { 200 } />
36
- { /* TODO: Actual button styles */ }
37
- < button
38
- className = "bg-bg-gray-40 rounded-full p-xs"
39
- onClick = { ( ) => {
40
- console . log ( "updating code" ) ;
41
- setPreviewCodeString ( codeString ) ;
42
- } }
43
- >
44
- Run
45
- </ button >
46
- < button
47
- className = "bg-bg-gray-40 rounded-full p-xs"
48
- onClick = { ( ) => {
49
- console . log ( "resetting code" ) ;
50
- setCodeString ( initialCode ) ;
51
- setPreviewCodeString ( initialCode ) ;
52
- } }
53
- >
54
- Reset
55
- </ button >
42
+ < div className = "flex lg:flex-col" >
43
+ < CodeFrame
44
+ jsCode = { previewCodeString }
45
+ width = { props . previewWidth }
46
+ height = { props . previewHeight }
47
+ />
48
+ { /* Looks more visually balanced with a slight leftward nudge */ }
49
+ < div className = "gap-xs lg:flex" >
50
+ < CircleButton
51
+ className = "!bg-bg-gray-40 !p-sm lg:ml-[-2px]"
52
+ onClick = { ( ) => {
53
+ setPreviewCodeString ( codeString ) ;
54
+ } }
55
+ >
56
+ < Icon kind = "play" />
57
+ </ CircleButton >
58
+ < CircleButton
59
+ className = "!bg-bg-gray-40 !p-sm"
60
+ onClick = { ( ) => {
61
+ setPreviewCodeString ( "" ) ;
62
+ } }
63
+ >
64
+ < Icon kind = "stop" />
65
+ </ CircleButton >
66
+ </ div >
56
67
</ div >
57
68
) : null }
58
- < div className = "w-full md:w-[calc(100%-150px)]" >
69
+ < div className = "relative w-full md:w-[calc(100%-150px)]" >
59
70
< CodeMirror
60
71
value = { codeString }
61
72
theme = "light"
62
73
width = "100%"
63
74
minimalSetup = { {
64
75
highlightSpecialChars : false ,
65
- history : true ,
76
+ history : false ,
66
77
drawSelection : true ,
67
78
syntaxHighlighting : true ,
68
79
defaultKeymap : true ,
69
80
historyKeymap : true ,
70
81
} }
71
- indentWithTab = { false }
72
- extensions = { [ javascript ( ) ] }
82
+ basicSetup = { {
83
+ lineNumbers : false ,
84
+ foldGutter : false ,
85
+ autocompletion : false ,
86
+ } }
87
+ extensions = { [ javascript ( ) , EditorView . lineWrapping ] }
73
88
onChange = { ( val ) => setCodeString ( val ) }
74
89
editable = { props . editable }
75
90
onCreateEditor = { ( editorView ) =>
76
91
( editorView . contentDOM . ariaLabel = "Code Editor" )
77
92
}
78
93
/>
94
+ < div className = "absolute right-0 top-0 flex gap-xs p-xs" >
95
+ < CopyCodeButton textToCopy = { codeString || initialCode } />
96
+ < CircleButton
97
+ onClick = { ( ) => {
98
+ setCodeString ( initialCode ) ;
99
+ setPreviewCodeString ( initialCode ) ;
100
+ } }
101
+ ariaLabel = "Reset code to initial value"
102
+ >
103
+ < Icon kind = "refresh" />
104
+ </ CircleButton >
105
+ </ div >
79
106
</ div >
80
107
</ div >
81
108
) ;
0 commit comments