Add interval setter UI
This commit is contained in:
parent
7d9fb457ff
commit
c9346da331
@ -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}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user