Add interval setter UI

This commit is contained in:
Nilay Majorwar 2021-12-16 21:37:53 +05:30
parent 7d9fb457ff
commit c9346da331
2 changed files with 64 additions and 15 deletions

View File

@ -3,7 +3,11 @@ import { CodeEditor, CodeEditorRef } from "../ui/code-editor";
import { InputEditor, InputEditorRef } from "../ui/input-editor"; import { InputEditor, InputEditorRef } from "../ui/input-editor";
import { MainLayout } from "../ui/MainLayout"; import { MainLayout } from "../ui/MainLayout";
import { useExecController } from "../ui/use-exec-controller"; import { useExecController } from "../ui/use-exec-controller";
import { DocumentRange, LanguageProvider } from "../engines/types"; import {
DocumentRange,
LanguageProvider,
StepExecutionResult,
} from "../engines/types";
import BrainfuckProvider from "../engines/brainfuck"; import BrainfuckProvider from "../engines/brainfuck";
import { OutputViewer } from "../ui/output-viewer"; import { OutputViewer } from "../ui/output-viewer";
import { ExecutionControls } from "./execution-controls"; import { ExecutionControls } from "./execution-controls";
@ -15,12 +19,20 @@ export const Mainframe = () => {
const execController = useExecController(); const execController = useExecController();
// UI states used in execution time // UI states used in execution time
const [execInterval, setExecInterval] = React.useState(20);
const [rendererState, setRendererState] = React.useState<any>(null); const [rendererState, setRendererState] = React.useState<any>(null);
const [output, setOutput] = React.useState<string | null>(null); const [output, setOutput] = React.useState<string | null>(null);
const [codeHighlights, setCodeHighlights] = React.useState< const [codeHighlights, setCodeHighlights] = React.useState<
DocumentRange | undefined DocumentRange | undefined
>(); >();
/** Utility that updates UI with the provided execution result */
const updateWithResult = (result: StepExecutionResult<any>) => {
setRendererState(result.rendererState);
setCodeHighlights(result.nextStepLocation || undefined);
setOutput((o) => (o || "") + (result.output || ""));
};
/** Reset and begin a new execution */ /** Reset and begin a new execution */
const runProgram = async () => { const runProgram = async () => {
// Check if controller is free for execution // Check if controller is free for execution
@ -40,11 +52,7 @@ export const Mainframe = () => {
); );
// Begin execution // Begin execution
await execController.execute((result) => { await execController.execute(updateWithResult, execInterval);
setRendererState(result.rendererState);
setCodeHighlights(result.nextStepLocation || undefined);
setOutput((o) => (o || "") + (result.output || ""));
}, 40);
}; };
/** Pause the ongoing execution */ /** Pause the ongoing execution */
@ -67,9 +75,7 @@ export const Mainframe = () => {
// Run and update execution states // Run and update execution states
const result = await execController.executeStep(); const result = await execController.executeStep();
setRendererState(result.rendererState); updateWithResult(result);
setCodeHighlights(result.nextStepLocation || undefined);
setOutput((o) => (o || "") + (result.output || ""));
}; };
/** Resume the currently paused execution */ /** Resume the currently paused execution */
@ -81,11 +87,7 @@ export const Mainframe = () => {
} }
// Begin execution // Begin execution
await execController.execute((result) => { await execController.execute(updateWithResult, execInterval);
setRendererState(result.rendererState);
setCodeHighlights(result.nextStepLocation || undefined);
setOutput((o) => (o || "") + (result.output || ""));
}, 40);
}; };
/** Stop the currently active execution */ /** Stop the currently active execution */
@ -142,6 +144,7 @@ export const Mainframe = () => {
onResume={resumeExecution} onResume={resumeExecution}
onStep={executeStep} onStep={executeStep}
onStop={stopExecution} onStop={stopExecution}
onChangeInterval={setExecInterval}
/> />
)} )}
/> />

View File

@ -1,11 +1,52 @@
import { Button, ButtonGroup, Icon } from "@blueprintjs/core"; import {
Button,
ButtonGroup,
Icon,
NumericInput,
Tag,
} from "@blueprintjs/core";
const styles = { const styles = {
container: { container: {
display: "flex", display: "flex",
alignItems: "center", alignItems: "center",
paddingRight: 5, paddingRight: 5,
marginRight: -15,
}, },
inputWrapper: {
/**
* As of Dec'21, NumericInput doesn't have `small` prop yet,
* so we instead use `transform` to hack up a smaller input.
*/
transform: "scale(0.8)",
},
input: {
width: 125,
},
};
/** Input field for changing execution interval */
const IntervalInput = (props: {
disabled: boolean;
onChange: (v: number) => void;
}) => {
return (
<div style={styles.inputWrapper}>
<NumericInput
min={20}
defaultValue={20}
stepSize={10}
minorStepSize={null}
leftIcon="time"
clampValueOnBlur
style={styles.input}
disabled={props.disabled}
onValueChange={(v) => props.onChange(v)}
rightElement={<Tag minimal>ms</Tag>}
allowNumericCharactersOnly
/>
</div>
);
}; };
/** Button for starting code execution */ /** Button for starting code execution */
@ -59,6 +100,7 @@ type Props = {
onResume: () => void; onResume: () => void;
onStep: () => void; onStep: () => void;
onStop: () => void; onStop: () => void;
onChangeInterval: (value: number) => void;
}; };
export const ExecutionControls = (props: Props) => { export const ExecutionControls = (props: Props) => {
@ -75,6 +117,10 @@ export const ExecutionControls = (props: Props) => {
onStop={props.onStop} onStop={props.onStop}
/> />
)} )}
<IntervalInput
disabled={props.state === "running"}
onChange={props.onChangeInterval}
/>
</div> </div>
); );
}; };