Add Box component to improve renderers performance
This commit is contained in:
parent
3e6fb8c780
commit
032f9a6792
@ -1,10 +1,18 @@
|
|||||||
import { Card, Colors, Icon, IconName } from "@blueprintjs/core";
|
import * as React from "react";
|
||||||
|
import { Colors, Icon, IconName } from "@blueprintjs/core";
|
||||||
import { RendererProps } from "../types";
|
import { RendererProps } from "../types";
|
||||||
|
import { Box } from "../ui-utils";
|
||||||
import { Bfg93Direction, Bfg93RS } from "./constants";
|
import { Bfg93Direction, Bfg93RS } from "./constants";
|
||||||
|
|
||||||
/** Common border color for dark and light, using transparency */
|
/** Common border color for dark and light, using transparency */
|
||||||
export const BorderColor = Colors.GRAY3 + "55";
|
export const BorderColor = Colors.GRAY3 + "55";
|
||||||
|
|
||||||
|
// Parameters for cell sizing, balanced to span the full row width
|
||||||
|
// Constraint: `(width% + 2 * margin%) * ROWSIZE = 100%`
|
||||||
|
const ROWSIZE = 8;
|
||||||
|
const CELL_WIDTH = "12%";
|
||||||
|
const CELL_MARGIN = "5px 0.25%";
|
||||||
|
|
||||||
const styles = {
|
const styles = {
|
||||||
placeholderDiv: {
|
placeholderDiv: {
|
||||||
height: "100%",
|
height: "100%",
|
||||||
@ -33,9 +41,9 @@ const styles = {
|
|||||||
},
|
},
|
||||||
stackItem: {
|
stackItem: {
|
||||||
// Sizing
|
// Sizing
|
||||||
width: "10%",
|
width: CELL_WIDTH,
|
||||||
height: "40px",
|
margin: CELL_MARGIN,
|
||||||
margin: "5px 0.25%",
|
height: "50px",
|
||||||
// Center-align values
|
// Center-align values
|
||||||
display: "flex",
|
display: "flex",
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
@ -50,10 +58,10 @@ const DirectionIcons: { [k: string]: IconName } = {
|
|||||||
[Bfg93Direction.DOWN]: "arrow-down",
|
[Bfg93Direction.DOWN]: "arrow-down",
|
||||||
};
|
};
|
||||||
|
|
||||||
const StackItem = ({ value }: { value: number }) => {
|
/** Component for displaying a single stack item */
|
||||||
const cellStyle = { ...styles.stackItem };
|
const StackItem = React.memo(({ value }: { value: number }) => {
|
||||||
return <Card style={cellStyle}>{value}</Card>;
|
return <Box style={styles.stackItem}>{value}</Box>;
|
||||||
};
|
});
|
||||||
|
|
||||||
export const Renderer = ({ state }: RendererProps<Bfg93RS>) => {
|
export const Renderer = ({ state }: RendererProps<Bfg93RS>) => {
|
||||||
if (state == null)
|
if (state == null)
|
||||||
@ -64,7 +72,6 @@ export const Renderer = ({ state }: RendererProps<Bfg93RS>) => {
|
|||||||
<div style={styles.dirnContainer}>
|
<div style={styles.dirnContainer}>
|
||||||
<span style={{ fontWeight: "bold", marginRight: 5 }}>Direction: </span>
|
<span style={{ fontWeight: "bold", marginRight: 5 }}>Direction: </span>
|
||||||
<Icon icon={DirectionIcons[state.direction]} />
|
<Icon icon={DirectionIcons[state.direction]} />
|
||||||
{/* <span style={{ marginLeft: 10 }} /> */}
|
|
||||||
<span style={{ marginLeft: 30, fontWeight: "bold", marginRight: 5 }}>
|
<span style={{ marginLeft: 30, fontWeight: "bold", marginRight: 5 }}>
|
||||||
String mode:{" "}
|
String mode:{" "}
|
||||||
</span>
|
</span>
|
||||||
|
@ -81,9 +81,12 @@ export const editorTokensProvider: MonacoTokensProvider = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/** Serialize tape from object format into linear array */
|
/** Serialize tape from object format into linear array */
|
||||||
export const serializeTapeMap = (tape: BFRS["tape"]): number[] => {
|
export const serializeTapeMap = (
|
||||||
|
tape: BFRS["tape"],
|
||||||
|
minCells: number = 0
|
||||||
|
): number[] => {
|
||||||
const cellIdxs = Object.keys(tape).map((s) => parseInt(s, 10));
|
const cellIdxs = Object.keys(tape).map((s) => parseInt(s, 10));
|
||||||
const maxCellIdx = Math.max(15, ...cellIdxs);
|
const maxCellIdx = Math.max(minCells - 1, ...cellIdxs);
|
||||||
const linearTape: number[] = Array(maxCellIdx + 1).fill(0);
|
const linearTape: number[] = Array(maxCellIdx + 1).fill(0);
|
||||||
cellIdxs.forEach((i) => (linearTape[i] = tape[i] || 0));
|
cellIdxs.forEach((i) => (linearTape[i] = tape[i] || 0));
|
||||||
return linearTape;
|
return linearTape;
|
||||||
|
@ -1,14 +1,17 @@
|
|||||||
import { Card, Colors } from "@blueprintjs/core";
|
import * as React from "react";
|
||||||
import { CSSProperties } from "react";
|
|
||||||
import { useDarkMode } from "../../ui/providers/dark-mode-provider";
|
|
||||||
import { RendererProps } from "../types";
|
import { RendererProps } from "../types";
|
||||||
|
import { Box } from "../ui-utils";
|
||||||
import { BFRS, serializeTapeMap } from "./common";
|
import { BFRS, serializeTapeMap } from "./common";
|
||||||
|
|
||||||
// Colors used as background of active cells
|
/** Number of cells shown in a single row */
|
||||||
const darkActiveBG = Colors.DARK_GRAY2;
|
const ROWSIZE = 8;
|
||||||
const lightActiveBG = Colors.LIGHT_GRAY3;
|
|
||||||
|
|
||||||
const styles: { [k: string]: CSSProperties } = {
|
// Parameters for cell sizing, balanced to span the full row width
|
||||||
|
// Constraint: `(width% + 2 * margin%) * ROWSIZE = 100%`
|
||||||
|
const CELL_WIDTH = "12%";
|
||||||
|
const CELL_MARGIN = "5px 0.25%";
|
||||||
|
|
||||||
|
const styles: { [k: string]: React.CSSProperties } = {
|
||||||
container: {
|
container: {
|
||||||
padding: 10,
|
padding: 10,
|
||||||
height: "100%",
|
height: "100%",
|
||||||
@ -19,9 +22,9 @@ const styles: { [k: string]: CSSProperties } = {
|
|||||||
},
|
},
|
||||||
cell: {
|
cell: {
|
||||||
// Sizing
|
// Sizing
|
||||||
width: "12%",
|
width: CELL_WIDTH,
|
||||||
|
margin: CELL_MARGIN,
|
||||||
height: "50px",
|
height: "50px",
|
||||||
margin: "5px 0.25%",
|
|
||||||
// Center-align values
|
// Center-align values
|
||||||
display: "flex",
|
display: "flex",
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
@ -30,22 +33,24 @@ const styles: { [k: string]: CSSProperties } = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/** Component for displaying a single tape cell */
|
/** Component for displaying a single tape cell */
|
||||||
const Cell = ({ value, active }: { value: number; active: boolean }) => {
|
const Cell = React.memo(
|
||||||
const { isDark } = useDarkMode();
|
({ value, active }: { value: number; active: boolean }) => {
|
||||||
const cellStyle = { ...styles.cell };
|
return (
|
||||||
const activeBg = isDark ? darkActiveBG : lightActiveBG;
|
<Box
|
||||||
if (active) {
|
intent={active ? "active" : "plain"}
|
||||||
cellStyle.backgroundColor = activeBg;
|
style={{ ...styles.cell, fontWeight: active ? "bold" : undefined }}
|
||||||
cellStyle.fontWeight = "bold";
|
>
|
||||||
|
{value}
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
return <Card style={cellStyle}>{value}</Card>;
|
);
|
||||||
};
|
|
||||||
|
|
||||||
/** Renderer for Brainfuck */
|
/** Renderer for Brainfuck */
|
||||||
export const Renderer = ({ state }: RendererProps<BFRS>) => {
|
export const Renderer = ({ state }: RendererProps<BFRS>) => {
|
||||||
return (
|
return (
|
||||||
<div style={styles.container}>
|
<div style={styles.container}>
|
||||||
{serializeTapeMap(state?.tape || {}).map((num, i) => (
|
{serializeTapeMap(state?.tape || {}, 2 * ROWSIZE).map((num, i) => (
|
||||||
<Cell value={num} key={i} active={(state?.pointer || 0) === i} />
|
<Cell value={num} key={i} active={(state?.pointer || 0) === i} />
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
import { Box } from "../../ui-utils";
|
||||||
import { CharacterValue } from "../common";
|
import { CharacterValue } from "../common";
|
||||||
import { SimpleTag } from "./utils";
|
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
name: string;
|
name: string;
|
||||||
@ -13,9 +13,9 @@ export const CharacterRow = (props: Props) => {
|
|||||||
<div style={{ margin: "20px 10px" }}>
|
<div style={{ margin: "20px 10px" }}>
|
||||||
<div>
|
<div>
|
||||||
<b style={{ marginRight: 5 }}>{name}:</b>{" "}
|
<b style={{ marginRight: 5 }}>{name}:</b>{" "}
|
||||||
<pre style={{ display: "inline" }}>{value.value}</pre>
|
<pre style={{ display: "inline", marginRight: 15 }}>{value.value}</pre>
|
||||||
{value.stack.map((v, i) => (
|
{value.stack.map((v, i) => (
|
||||||
<SimpleTag key={i}>{v}</SimpleTag>
|
<Box key={i}>{v}</Box>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
<div></div>
|
<div></div>
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import { Tag, Text } from "@blueprintjs/core";
|
import React from "react";
|
||||||
import { SimpleTag } from "./utils";
|
import { Text } from "@blueprintjs/core";
|
||||||
|
import { Box } from "../../ui-utils";
|
||||||
|
|
||||||
const styles = {
|
const styles = {
|
||||||
charChip: {
|
charChip: {
|
||||||
@ -22,20 +23,16 @@ export const TopBar = (props: Props) => {
|
|||||||
|
|
||||||
const characterChips =
|
const characterChips =
|
||||||
charactersOnStage.length === 0 ? (
|
charactersOnStage.length === 0 ? (
|
||||||
<Tag large minimal>
|
<Box>The stage is empty</Box>
|
||||||
The stage is empty
|
|
||||||
</Tag>
|
|
||||||
) : (
|
) : (
|
||||||
charactersOnStage.map((character) => {
|
charactersOnStage.map((character) => (
|
||||||
return (
|
<Box
|
||||||
<SimpleTag
|
key={character}
|
||||||
key={character}
|
intent={character === currSpeaker ? "active" : "plain"}
|
||||||
intent={character === currSpeaker ? "active" : undefined}
|
>
|
||||||
>
|
{character}
|
||||||
{character}
|
</Box>
|
||||||
</SimpleTag>
|
))
|
||||||
);
|
|
||||||
})
|
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -46,9 +43,9 @@ export const TopBar = (props: Props) => {
|
|||||||
<Text tagName="span" style={styles.questionText}>
|
<Text tagName="span" style={styles.questionText}>
|
||||||
Answer to question:
|
Answer to question:
|
||||||
</Text>
|
</Text>
|
||||||
<SimpleTag intent={questionState ? "success" : "danger"}>
|
<Box intent={questionState ? "success" : "danger"}>
|
||||||
{questionState ? "yes" : "no"}
|
{questionState ? "yes" : "no"}
|
||||||
</SimpleTag>
|
</Box>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,41 +1,43 @@
|
|||||||
|
import { CSSProperties } from "react";
|
||||||
import { Colors } from "@blueprintjs/core";
|
import { Colors } from "@blueprintjs/core";
|
||||||
import { useDarkMode } from "../../../ui/providers/dark-mode-provider";
|
import { useDarkMode } from "../ui/providers/dark-mode-provider";
|
||||||
|
|
||||||
const backgroundColorsLight = {
|
const backgroundColorsLight = {
|
||||||
success: Colors.GREEN3,
|
success: Colors.GREEN5,
|
||||||
danger: Colors.RED3,
|
danger: Colors.RED5,
|
||||||
plain: Colors.GRAY3,
|
plain: Colors.LIGHT_GRAY1,
|
||||||
active: Colors.DARK_GRAY1,
|
active: Colors.GRAY4,
|
||||||
};
|
};
|
||||||
|
|
||||||
const backgroundColorsDark = {
|
const backgroundColorsDark = {
|
||||||
success: Colors.GREEN3,
|
success: Colors.GREEN1,
|
||||||
danger: Colors.RED3,
|
danger: Colors.RED1,
|
||||||
plain: Colors.GRAY3,
|
plain: Colors.DARK_GRAY5,
|
||||||
active: Colors.LIGHT_GRAY5,
|
active: Colors.GRAY1,
|
||||||
};
|
};
|
||||||
|
|
||||||
const foregroundColorsLight = {
|
const foregroundColorsLight = {
|
||||||
success: Colors.GREEN2,
|
success: Colors.GREEN1,
|
||||||
danger: Colors.RED2,
|
danger: Colors.RED1,
|
||||||
plain: Colors.DARK_GRAY1,
|
plain: Colors.DARK_GRAY1,
|
||||||
active: Colors.LIGHT_GRAY5,
|
active: Colors.DARK_GRAY1,
|
||||||
};
|
};
|
||||||
|
|
||||||
const foregroundColorsDark = {
|
const foregroundColorsDark = {
|
||||||
success: Colors.GREEN5,
|
success: Colors.GREEN5,
|
||||||
danger: Colors.RED5,
|
danger: Colors.RED5,
|
||||||
plain: Colors.LIGHT_GRAY5,
|
plain: Colors.LIGHT_GRAY5,
|
||||||
active: Colors.DARK_GRAY1,
|
active: Colors.LIGHT_GRAY5,
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility component that renders a tag similar to BlueprintJS tags, but underneath
|
* Utility component for rendering a simple general-purpose stylable box. Useful
|
||||||
* is just a single span tag with no frills and high performance.
|
* for performance-critical components in visualisation renderers.
|
||||||
*/
|
*/
|
||||||
export const SimpleTag = (props: {
|
export const Box = (props: {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
intent?: "success" | "danger" | "active";
|
intent?: "plain" | "success" | "danger" | "active";
|
||||||
|
style?: CSSProperties;
|
||||||
}) => {
|
}) => {
|
||||||
const { isDark } = useDarkMode();
|
const { isDark } = useDarkMode();
|
||||||
const intent = props.intent == null ? "plain" : props.intent;
|
const intent = props.intent == null ? "plain" : props.intent;
|
||||||
@ -49,9 +51,9 @@ export const SimpleTag = (props: {
|
|||||||
margin: 5,
|
margin: 5,
|
||||||
padding: "5px 10px",
|
padding: "5px 10px",
|
||||||
borderRadius: 3,
|
borderRadius: 3,
|
||||||
backgroundColor:
|
backgroundColor: backgroundMap[intent],
|
||||||
backgroundMap[intent] + (intent === "active" ? "aa" : "55"),
|
|
||||||
color: foregroundMap[intent],
|
color: foregroundMap[intent],
|
||||||
|
...props.style,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{props.children}
|
{props.children}
|
Loading…
x
Reference in New Issue
Block a user