Add Box component to improve renderers performance

This commit is contained in:
Nilay Majorwar 2022-02-22 14:54:17 +05:30
parent 3e6fb8c780
commit 032f9a6792
6 changed files with 83 additions and 69 deletions

View File

@ -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>

View File

@ -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;

View File

@ -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>

View File

@ -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>

View File

@ -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>

View File

@ -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}