From febe31a3d818d2811b99a86d2a44b09a1bdc6aa4 Mon Sep 17 00:00:00 2001 From: Nilay Majorwar Date: Fri, 17 Dec 2021 15:05:28 +0530 Subject: [PATCH] Refactor state flow to boost performance For intervals < ~24ms, the main thread as unable to cope up in handling worker responses due to Mainframe rendering on each execution. To resolve this, this commit delegates all execution-time states to child components, controlled imperatively from Mainframe. This yields huge performance boost, with main thread keeping up with worker responses even at interval of 5ms. --- ui/Mainframe.tsx | 56 ++++++++++++++++++++++----------------- ui/code-editor/index.tsx | 35 +++++++++++++----------- ui/execution-controls.tsx | 4 +-- ui/output-viewer.tsx | 21 ++++++++++++--- ui/renderer-wrapper.tsx | 27 +++++++++++++++++++ 5 files changed, 97 insertions(+), 46 deletions(-) create mode 100644 ui/renderer-wrapper.tsx diff --git a/ui/Mainframe.tsx b/ui/Mainframe.tsx index d6ddd4f..5e6c779 100644 --- a/ui/Mainframe.tsx +++ b/ui/Mainframe.tsx @@ -3,34 +3,39 @@ import { CodeEditor, CodeEditorRef } from "../ui/code-editor"; import { InputEditor, InputEditorRef } from "../ui/input-editor"; import { MainLayout } from "../ui/MainLayout"; import { useExecController } from "../ui/use-exec-controller"; -import { - DocumentRange, - LanguageProvider, - StepExecutionResult, -} from "../engines/types"; +import { LanguageProvider, StepExecutionResult } from "../engines/types"; import BrainfuckProvider from "../engines/brainfuck"; -import { OutputViewer } from "../ui/output-viewer"; +import { OutputViewer, OutputViewerRef } from "../ui/output-viewer"; import { ExecutionControls } from "./execution-controls"; +import { RendererRef, RendererWrapper } from "./renderer-wrapper"; +/** + * React component that contains and controls the entire IDE. + * + * For performance reasons, Mainframe makes spare use of state hooks. This + * component is rather expensive to render, and will block the main thread on + * small execution intervals if rendered on every execution. All state management + * is delegated to imperatively controlled child components. + */ export const Mainframe = () => { - const codeEditorRef = React.useRef(null); - const inputEditorRef = React.useRef(null); + // Language provider and engine const providerRef = React.useRef>(BrainfuckProvider); const execController = useExecController(); - // UI states used in execution time + // Refs for controlling UI components + const codeEditorRef = React.useRef(null); + const inputEditorRef = React.useRef(null); + const outputEditorRef = React.useRef(null); + const rendererRef = React.useRef>(null); + + // Interval of execution const [execInterval, setExecInterval] = React.useState(20); - const [rendererState, setRendererState] = React.useState(null); - const [output, setOutput] = React.useState(null); - const [codeHighlights, setCodeHighlights] = React.useState< - DocumentRange | undefined - >(); /** Utility that updates UI with the provided execution result */ const updateWithResult = (result: StepExecutionResult) => { - setRendererState(result.rendererState); - setCodeHighlights(result.nextStepLocation || undefined); - setOutput((o) => (o || "") + (result.output || "")); + rendererRef.current!.updateState(result.rendererState); + codeEditorRef.current!.updateHighlights(result.nextStepLocation); + outputEditorRef.current!.append(result.output); }; /** Reset and begin a new execution */ @@ -43,8 +48,7 @@ export const Mainframe = () => { } // Reset any existing execution state - setOutput(""); - setRendererState(null); + outputEditorRef.current!.reset(); await execController.resetState(); await execController.prepare( codeEditorRef.current!.getValue(), @@ -104,9 +108,9 @@ export const Mainframe = () => { // Reset all execution states await execController.resetState(); - setOutput(null); - setRendererState(null); - setCodeHighlights(undefined); + outputEditorRef.current!.reset(); + rendererRef.current!.updateState(null); + codeEditorRef.current!.updateHighlights(null); }; /** Translate execution controller state to debug controls state */ @@ -123,7 +127,6 @@ export const Mainframe = () => { @@ -132,10 +135,13 @@ export const Mainframe = () => { /> )} renderRenderer={() => ( - + )} renderInput={() => } - renderOutput={() => } + renderOutput={() => } renderExecControls={() => ( string; + /** Update code highlights */ + updateHighlights: (highlights: DocumentRange | null) => void; } type Props = { @@ -18,8 +18,6 @@ type Props = { languageId: string; /** Default code to display in editor */ defaultValue: string; - /** Code range to highlight in the editor */ - highlights?: DocumentRange; /** Tokens provider for the language */ tokensProvider?: MonacoTokensProvider; /** Callback to update debugging breakpoints */ @@ -32,8 +30,8 @@ type Props = { */ const CodeEditorComponent = (props: Props, ref: React.Ref) => { const editorRef = React.useRef(null); + const highlightRange = React.useRef([]); const monacoInstance = useMonaco(); - const { highlights } = props; // Breakpoints useEditorBreakpoints({ @@ -48,21 +46,28 @@ const CodeEditorComponent = (props: Props, ref: React.Ref) => { tokensProvider: props.tokensProvider, }); - // Change editor highlights when prop changes - React.useEffect(() => { - if (!editorRef.current || !highlights) return; - const range = createHighlightRange(monacoInstance!, highlights); - const decors = editorRef.current!.deltaDecorations([], [range]); - return () => { - editorRef.current!.deltaDecorations(decors, []); - }; - }, [highlights]); + /** Update code highlights */ + const updateHighlights = React.useCallback( + (hl: DocumentRange | null) => { + // Remove previous highlights + const prevRange = highlightRange.current; + editorRef.current!.deltaDecorations(prevRange, []); + + // Add new highlights + if (!hl) return; + const newRange = createHighlightRange(monacoInstance!, hl); + const rangeStr = editorRef.current!.deltaDecorations([], [newRange]); + highlightRange.current = rangeStr; + }, + [monacoInstance] + ); // Provide handle to parent for accessing editor contents React.useImperativeHandle( ref, () => ({ getValue: () => editorRef.current!.getValue(), + updateHighlights, }), [] ); diff --git a/ui/execution-controls.tsx b/ui/execution-controls.tsx index 979b599..ac2fa77 100644 --- a/ui/execution-controls.tsx +++ b/ui/execution-controls.tsx @@ -33,9 +33,9 @@ const IntervalInput = (props: { return (
{ return value; // Non-empty output value }; -type Props = { - value: string | null; -}; +export interface OutputViewerRef { + /** Reset output to show placeholder text */ + reset: () => void; + /** Append string to the displayed output */ + append: (str?: string) => void; +} + +const OutputViewerComponent = (_: {}, ref: React.Ref) => { + const [value, setValue] = React.useState(null); + + React.useImperativeHandle(ref, () => ({ + reset: () => setValue(null), + append: (s) => setValue((o) => (o || "") + (s || "")), + })); -export const OutputViewer = ({ value }: Props) => { return (