Add breakpoints to editor UI
This commit is contained in:
parent
01ba292b9f
commit
c9afb3a68b
@ -1,12 +1,19 @@
|
|||||||
.code-highlight {
|
.code-highlight {
|
||||||
background-color: #ffff0077;
|
background-color: #ffff00aa;
|
||||||
}
|
}
|
||||||
|
|
||||||
.breakpoint-glyph {
|
.breakpoint-glyph {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: 4%;
|
padding: 6%; /* Make the dot smaller in size */
|
||||||
|
margin-top: 2px; /* Fix dot appearing slightly above baseline */
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
margin-left: 10px;
|
|
||||||
background-color: #ff5555;
|
|
||||||
background-clip: content-box;
|
background-clip: content-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.breakpoint-glyph.solid {
|
||||||
|
background-color: #ff5555 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.breakpoint-glyph.hint {
|
||||||
|
background-color: #ff555555;
|
||||||
|
}
|
||||||
|
@ -66,6 +66,7 @@ export const Mainframe = () => {
|
|||||||
highlights={codeHighlights}
|
highlights={codeHighlights}
|
||||||
defaultValue={providerRef.current.sampleProgram}
|
defaultValue={providerRef.current.sampleProgram}
|
||||||
tokensProvider={providerRef.current.editorTokensProvider}
|
tokensProvider={providerRef.current.editorTokensProvider}
|
||||||
|
onUpdateBreakpoints={(newPoints) => console.log(newPoints)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
renderRenderer={() => (
|
renderRenderer={() => (
|
||||||
|
@ -1,24 +1,9 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import Editor, { useMonaco } from "@monaco-editor/react";
|
import Editor, { useMonaco } from "@monaco-editor/react";
|
||||||
import monaco from "monaco-editor";
|
import { useEditorLanguageConfig } from "./use-editor-lang-config";
|
||||||
import { DocumentRange, MonacoTokensProvider } from "../../engines/types";
|
import { DocumentRange, MonacoTokensProvider } from "../../engines/types";
|
||||||
import { useEditorConfig } from "./use-editor-config";
|
import { createHighlightRange, EditorInstance } from "./monaco-utils";
|
||||||
|
import { useEditorBreakpoints } from "./use-editor-breakpoints";
|
||||||
// Type aliases for the Monaco editor
|
|
||||||
type EditorInstance = monaco.editor.IStandaloneCodeEditor;
|
|
||||||
|
|
||||||
/** Create Monaco decoration range object from highlights */
|
|
||||||
const createRange = (
|
|
||||||
monacoInstance: typeof monaco,
|
|
||||||
highlights: DocumentRange
|
|
||||||
) => {
|
|
||||||
const lineNum = highlights.line;
|
|
||||||
const startChar = highlights.charRange?.start || 0;
|
|
||||||
const endChar = highlights.charRange?.end || 1000;
|
|
||||||
const range = new monacoInstance.Range(lineNum, startChar, lineNum, endChar);
|
|
||||||
const isWholeLine = !highlights.charRange;
|
|
||||||
return { range, options: { isWholeLine, inlineClassName: "code-highlight" } };
|
|
||||||
};
|
|
||||||
|
|
||||||
// Interface for interacting with the editor
|
// Interface for interacting with the editor
|
||||||
export interface CodeEditorRef {
|
export interface CodeEditorRef {
|
||||||
@ -37,6 +22,8 @@ type Props = {
|
|||||||
highlights?: DocumentRange;
|
highlights?: DocumentRange;
|
||||||
/** Tokens provider for the language */
|
/** Tokens provider for the language */
|
||||||
tokensProvider?: MonacoTokensProvider;
|
tokensProvider?: MonacoTokensProvider;
|
||||||
|
/** Callback to update debugging breakpoints */
|
||||||
|
onUpdateBreakpoints: (newBreakpoints: number[]) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -47,7 +34,16 @@ const CodeEditorComponent = (props: Props, ref: React.Ref<CodeEditorRef>) => {
|
|||||||
const editorRef = React.useRef<EditorInstance | null>(null);
|
const editorRef = React.useRef<EditorInstance | null>(null);
|
||||||
const monacoInstance = useMonaco();
|
const monacoInstance = useMonaco();
|
||||||
const { highlights } = props;
|
const { highlights } = props;
|
||||||
useEditorConfig({
|
|
||||||
|
// Breakpoints
|
||||||
|
useEditorBreakpoints({
|
||||||
|
editor: editorRef.current,
|
||||||
|
monaco: monacoInstance,
|
||||||
|
onUpdateBreakpoints: props.onUpdateBreakpoints,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Language config
|
||||||
|
useEditorLanguageConfig({
|
||||||
languageId: props.languageId,
|
languageId: props.languageId,
|
||||||
tokensProvider: props.tokensProvider,
|
tokensProvider: props.tokensProvider,
|
||||||
});
|
});
|
||||||
@ -55,7 +51,7 @@ const CodeEditorComponent = (props: Props, ref: React.Ref<CodeEditorRef>) => {
|
|||||||
// Change editor highlights when prop changes
|
// Change editor highlights when prop changes
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
if (!editorRef.current || !highlights) return;
|
if (!editorRef.current || !highlights) return;
|
||||||
const range = createRange(monacoInstance!, highlights);
|
const range = createHighlightRange(monacoInstance!, highlights);
|
||||||
const decors = editorRef.current!.deltaDecorations([], [range]);
|
const decors = editorRef.current!.deltaDecorations([], [range]);
|
||||||
return () => {
|
return () => {
|
||||||
editorRef.current!.deltaDecorations(decors, []);
|
editorRef.current!.deltaDecorations(decors, []);
|
||||||
@ -77,7 +73,7 @@ const CodeEditorComponent = (props: Props, ref: React.Ref<CodeEditorRef>) => {
|
|||||||
defaultLanguage="brainfuck"
|
defaultLanguage="brainfuck"
|
||||||
defaultValue={props.defaultValue}
|
defaultValue={props.defaultValue}
|
||||||
onMount={(editor) => (editorRef.current = editor)}
|
onMount={(editor) => (editorRef.current = editor)}
|
||||||
options={{ minimap: { enabled: false } }}
|
options={{ minimap: { enabled: false }, glyphMargin: true }}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
41
ui/code-editor/monaco-utils.ts
Normal file
41
ui/code-editor/monaco-utils.ts
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
import monaco from "monaco-editor";
|
||||||
|
import { DocumentRange } from "../../engines/types";
|
||||||
|
|
||||||
|
/** Type alias for an instance of Monaco editor */
|
||||||
|
export type EditorInstance = monaco.editor.IStandaloneCodeEditor;
|
||||||
|
|
||||||
|
/** Type alias for the Monaco global */
|
||||||
|
export type MonacoInstance = typeof monaco;
|
||||||
|
|
||||||
|
/** Type alias for Monaco mouse events */
|
||||||
|
export type MonacoMouseEvent = monaco.editor.IEditorMouseEvent;
|
||||||
|
|
||||||
|
/** Type alias for Monaco mouse-leave event */
|
||||||
|
export type MonacoMouseLeaveEvent = monaco.editor.IPartialEditorMouseEvent;
|
||||||
|
|
||||||
|
/** Type alias for Monaco decoration object */
|
||||||
|
export type MonacoDecoration = monaco.editor.IModelDeltaDecoration;
|
||||||
|
|
||||||
|
/** Create Monaco decoration range object for text highlighting */
|
||||||
|
export const createHighlightRange = (
|
||||||
|
monacoInstance: MonacoInstance,
|
||||||
|
highlights: DocumentRange
|
||||||
|
): MonacoDecoration => {
|
||||||
|
const lineNum = highlights.line;
|
||||||
|
const startChar = highlights.charRange?.start || 0;
|
||||||
|
const endChar = highlights.charRange?.end || 1000;
|
||||||
|
const range = new monacoInstance.Range(lineNum, startChar, lineNum, endChar);
|
||||||
|
const isWholeLine = !highlights.charRange;
|
||||||
|
return { range, options: { isWholeLine, inlineClassName: "code-highlight" } };
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Create Monaco decoration range object from highlights */
|
||||||
|
export const createBreakpointRange = (
|
||||||
|
monacoInstance: MonacoInstance,
|
||||||
|
lineNum: number,
|
||||||
|
hint?: boolean
|
||||||
|
): MonacoDecoration => {
|
||||||
|
const range = new monacoInstance.Range(lineNum, 0, lineNum, 1000);
|
||||||
|
const className = "breakpoint-glyph " + (hint ? "hint" : "solid");
|
||||||
|
return { range, options: { glyphMarginClassName: className } };
|
||||||
|
};
|
115
ui/code-editor/use-editor-breakpoints.ts
Normal file
115
ui/code-editor/use-editor-breakpoints.ts
Normal file
@ -0,0 +1,115 @@
|
|||||||
|
import React from "react";
|
||||||
|
import {
|
||||||
|
createBreakpointRange,
|
||||||
|
EditorInstance,
|
||||||
|
MonacoInstance,
|
||||||
|
MonacoMouseEvent,
|
||||||
|
MonacoMouseLeaveEvent,
|
||||||
|
} from "./monaco-utils";
|
||||||
|
|
||||||
|
type BreakpointsMap = { [k: number]: string[] };
|
||||||
|
type HoverBreakpoint = {
|
||||||
|
lineNum: number;
|
||||||
|
decorRanges: string[];
|
||||||
|
};
|
||||||
|
|
||||||
|
type Args = {
|
||||||
|
editor: EditorInstance | null;
|
||||||
|
monaco: MonacoInstance | null;
|
||||||
|
onUpdateBreakpoints: (newBreakpoints: number[]) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useEditorBreakpoints = ({
|
||||||
|
editor,
|
||||||
|
monaco,
|
||||||
|
onUpdateBreakpoints,
|
||||||
|
}: Args) => {
|
||||||
|
const breakpoints = React.useRef<BreakpointsMap>({});
|
||||||
|
const hoverBreakpoint = React.useRef<HoverBreakpoint | null>(null);
|
||||||
|
|
||||||
|
// Mouse clicks -> add or remove breakpoint
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!editor || !monaco) return;
|
||||||
|
const disposer = editor.onMouseDown((e: MonacoMouseEvent) => {
|
||||||
|
// Check if click is in glyph display channel
|
||||||
|
const glyphMarginType = monaco.editor.MouseTargetType.GUTTER_GLYPH_MARGIN;
|
||||||
|
const isGlyphMargin = e.target.type === glyphMarginType;
|
||||||
|
if (!isGlyphMargin) return;
|
||||||
|
|
||||||
|
const lineNum = e.target.position!.lineNumber;
|
||||||
|
const existingRange = breakpoints.current[lineNum];
|
||||||
|
if (existingRange) {
|
||||||
|
// Already has breakpoint - remove it
|
||||||
|
editor.deltaDecorations(existingRange, []);
|
||||||
|
delete breakpoints.current[lineNum];
|
||||||
|
} else {
|
||||||
|
// Add breakpoint to this line
|
||||||
|
const range = createBreakpointRange(monaco, lineNum);
|
||||||
|
const newRangeStr = editor.deltaDecorations([], [range]);
|
||||||
|
breakpoints.current[lineNum] = newRangeStr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update breakpoints to parent
|
||||||
|
const bpLineNumStrs = Object.keys(breakpoints.current);
|
||||||
|
const bpLineNums = bpLineNumStrs.map((numStr) => parseInt(numStr, 10));
|
||||||
|
onUpdateBreakpoints(bpLineNums);
|
||||||
|
});
|
||||||
|
return () => disposer.dispose();
|
||||||
|
}, [editor, monaco]);
|
||||||
|
|
||||||
|
// Mouse enter -> show semi-transparent breakpoint icon
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!editor || !monaco) return;
|
||||||
|
const disposer = editor.onMouseMove((e: MonacoMouseEvent) => {
|
||||||
|
// Check if click is in glyph display channel
|
||||||
|
const glyphMarginType = monaco.editor.MouseTargetType.GUTTER_GLYPH_MARGIN;
|
||||||
|
const isGlyphMargin = e.target.type === glyphMarginType;
|
||||||
|
|
||||||
|
// If mouse goes out of glyph channel...
|
||||||
|
if (!isGlyphMargin) {
|
||||||
|
if (hoverBreakpoint.current) {
|
||||||
|
editor.deltaDecorations(hoverBreakpoint.current.decorRanges, []);
|
||||||
|
hoverBreakpoint.current = null;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if hover is in already hinted line
|
||||||
|
const hoverLineNum = e.target.position!.lineNumber;
|
||||||
|
if (hoverLineNum === hoverBreakpoint.current?.lineNum) return;
|
||||||
|
|
||||||
|
// Add hover decoration to newly hovered line
|
||||||
|
const range = createBreakpointRange(monaco, hoverLineNum, true);
|
||||||
|
const newHoverRangeStr = editor.deltaDecorations([], [range]);
|
||||||
|
|
||||||
|
// Remove existing breakpoint hover
|
||||||
|
if (hoverBreakpoint.current)
|
||||||
|
editor.deltaDecorations(hoverBreakpoint.current.decorRanges, []);
|
||||||
|
|
||||||
|
// Set hover breakpoint state to new one
|
||||||
|
hoverBreakpoint.current = {
|
||||||
|
lineNum: hoverLineNum,
|
||||||
|
decorRanges: newHoverRangeStr,
|
||||||
|
};
|
||||||
|
|
||||||
|
// If breakpoint already on line, ignore
|
||||||
|
const lineNum = e.target.position!.lineNumber;
|
||||||
|
const existingRange = breakpoints.current[lineNum];
|
||||||
|
if (existingRange) return;
|
||||||
|
});
|
||||||
|
return () => disposer.dispose();
|
||||||
|
}, [editor, monaco]);
|
||||||
|
|
||||||
|
// Mouse leaves editor -> remove hover breakpoint hint
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!editor) return;
|
||||||
|
const disposer = editor.onMouseLeave((e: MonacoMouseLeaveEvent) => {
|
||||||
|
if (!hoverBreakpoint.current) return;
|
||||||
|
editor.deltaDecorations(hoverBreakpoint.current.decorRanges, []);
|
||||||
|
hoverBreakpoint.current = null;
|
||||||
|
});
|
||||||
|
return () => disposer.dispose();
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
return breakpoints;
|
||||||
|
};
|
@ -8,7 +8,7 @@ type ConfigParams = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/** Add custom language and relevant providers to Monaco */
|
/** Add custom language and relevant providers to Monaco */
|
||||||
export const useEditorConfig = (params: ConfigParams) => {
|
export const useEditorLanguageConfig = (params: ConfigParams) => {
|
||||||
const monaco = useMonaco();
|
const monaco = useMonaco();
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
Loading…
x
Reference in New Issue
Block a user