mirror of
https://github.com/CHOMPStation2/CHOMPStation2.git
synced 2025-12-11 10:43:20 +00:00
278 lines
8.2 KiB
JavaScript
278 lines
8.2 KiB
JavaScript
import { map, sortBy } from 'common/collections';
|
|
import { flow } from 'common/fp';
|
|
import { toFixed } from 'common/math';
|
|
import { pureComponentHooks } from 'common/react';
|
|
import { Fragment } from 'inferno';
|
|
import { useBackend, useLocalState } from '../backend';
|
|
import { Box, Button, Chart, ColorBox, Flex, Icon, LabeledList, ProgressBar, Section, Table } from '../components';
|
|
import { Window } from '../layouts';
|
|
|
|
const PEAK_DRAW = 500000;
|
|
|
|
export const powerRank = (str) => {
|
|
const unit = String(str.split(' ')[1]).toLowerCase();
|
|
return ['w', 'kw', 'mw', 'gw'].indexOf(unit);
|
|
};
|
|
|
|
export const PowerMonitor = () => {
|
|
return (
|
|
<Window width={550} height={700} resizable>
|
|
<Window.Content scrollable>
|
|
<PowerMonitorContent />
|
|
</Window.Content>
|
|
</Window>
|
|
);
|
|
};
|
|
|
|
export const PowerMonitorContent = (props, context) => {
|
|
const { act, data } = useBackend(context);
|
|
|
|
const { map_levels, all_sensors, focus } = data;
|
|
|
|
if (focus) {
|
|
return <PowerMonitorFocus focus={focus} />;
|
|
}
|
|
|
|
let body = <Box color="bad">No sensors detected</Box>;
|
|
|
|
if (all_sensors) {
|
|
body = (
|
|
<Table>
|
|
{all_sensors.map((sensor) => (
|
|
<Table.Row key={sensor.name}>
|
|
<Table.Cell>
|
|
<Button
|
|
content={sensor.name}
|
|
icon={sensor.alarm ? 'bell' : 'sign-in-alt'}
|
|
onClick={() => act('setsensor', { id: sensor.name })}
|
|
/>
|
|
</Table.Cell>
|
|
</Table.Row>
|
|
))}
|
|
</Table>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<Section
|
|
title="No active sensor. Listing all."
|
|
buttons={
|
|
<Button
|
|
content="Scan For Sensors"
|
|
icon="undo"
|
|
onClick={() => act('refresh')}
|
|
/>
|
|
}>
|
|
{body}
|
|
</Section>
|
|
);
|
|
};
|
|
|
|
export const PowerMonitorFocus = (props, context) => {
|
|
const { act, data } = useBackend(context);
|
|
const { focus } = props;
|
|
const { history } = focus;
|
|
const [sortByField, setSortByField] = useLocalState(
|
|
context,
|
|
'sortByField',
|
|
null
|
|
);
|
|
const supply = history.supply[history.supply.length - 1] || 0;
|
|
const demand = history.demand[history.demand.length - 1] || 0;
|
|
const supplyData = history.supply.map((value, i) => [i, value]);
|
|
const demandData = history.demand.map((value, i) => [i, value]);
|
|
const maxValue = Math.max(PEAK_DRAW, ...history.supply, ...history.demand);
|
|
// Process area data
|
|
const areas = flow([
|
|
map((area, i) => ({
|
|
...area,
|
|
// Generate a unique id
|
|
id: area.name + i,
|
|
})),
|
|
sortByField === 'name' && sortBy((area) => area.name),
|
|
sortByField === 'charge' && sortBy((area) => -area.charge),
|
|
sortByField === 'draw' &&
|
|
sortBy(
|
|
(area) => -powerRank(area.load),
|
|
(area) => -parseFloat(area.load)
|
|
),
|
|
sortByField === 'problems' &&
|
|
sortBy(
|
|
(area) => area.eqp,
|
|
(area) => area.lgt,
|
|
(area) => area.env,
|
|
(area) => area.charge,
|
|
(area) => area.name
|
|
),
|
|
])(focus.areas);
|
|
return (
|
|
<Fragment>
|
|
<Section
|
|
title={focus.name}
|
|
buttons={
|
|
<Button
|
|
icon="sign-out-alt"
|
|
content="Back To Main"
|
|
onClick={() => act('clear')}
|
|
/>
|
|
}
|
|
/>
|
|
<Flex mx={-0.5} mb={1}>
|
|
<Flex.Item mx={0.5} width="200px">
|
|
<Section>
|
|
<LabeledList>
|
|
<LabeledList.Item label="Supply">
|
|
<ProgressBar
|
|
value={supply}
|
|
minValue={0}
|
|
maxValue={maxValue}
|
|
color="teal">
|
|
{toFixed(supply / 1000) + ' kW'}
|
|
</ProgressBar>
|
|
</LabeledList.Item>
|
|
<LabeledList.Item label="Draw">
|
|
<ProgressBar
|
|
value={demand}
|
|
minValue={0}
|
|
maxValue={maxValue}
|
|
color="pink">
|
|
{toFixed(demand / 1000) + ' kW'}
|
|
</ProgressBar>
|
|
</LabeledList.Item>
|
|
</LabeledList>
|
|
</Section>
|
|
</Flex.Item>
|
|
<Flex.Item mx={0.5} grow={1}>
|
|
<Section position="relative" height="100%">
|
|
<Chart.Line
|
|
fillPositionedParent
|
|
data={supplyData}
|
|
rangeX={[0, supplyData.length - 1]}
|
|
rangeY={[0, maxValue]}
|
|
strokeColor="rgba(0, 181, 173, 1)"
|
|
fillColor="rgba(0, 181, 173, 0.25)"
|
|
/>
|
|
<Chart.Line
|
|
fillPositionedParent
|
|
data={demandData}
|
|
rangeX={[0, demandData.length - 1]}
|
|
rangeY={[0, maxValue]}
|
|
strokeColor="rgba(224, 57, 151, 1)"
|
|
fillColor="rgba(224, 57, 151, 0.25)"
|
|
/>
|
|
</Section>
|
|
</Flex.Item>
|
|
</Flex>
|
|
<Section>
|
|
<Box mb={1}>
|
|
<Box inline mr={2} color="label">
|
|
Sort by:
|
|
</Box>
|
|
<Button.Checkbox
|
|
checked={sortByField === 'name'}
|
|
content="Name"
|
|
onClick={() => setSortByField(sortByField !== 'name' && 'name')}
|
|
/>
|
|
<Button.Checkbox
|
|
checked={sortByField === 'charge'}
|
|
content="Charge"
|
|
onClick={() => setSortByField(sortByField !== 'charge' && 'charge')}
|
|
/>
|
|
<Button.Checkbox
|
|
checked={sortByField === 'draw'}
|
|
content="Draw"
|
|
onClick={() => setSortByField(sortByField !== 'draw' && 'draw')}
|
|
/>
|
|
<Button.Checkbox
|
|
checked={sortByField === 'problems'}
|
|
content="Problems"
|
|
onClick={() =>
|
|
setSortByField(sortByField !== 'problems' && 'problems')
|
|
}
|
|
/>
|
|
</Box>
|
|
<Table>
|
|
<Table.Row header>
|
|
<Table.Cell>Area</Table.Cell>
|
|
<Table.Cell collapsing>Charge</Table.Cell>
|
|
<Table.Cell textAlign="right">Draw</Table.Cell>
|
|
<Table.Cell collapsing title="Equipment">
|
|
Eqp
|
|
</Table.Cell>
|
|
<Table.Cell collapsing title="Lighting">
|
|
Lgt
|
|
</Table.Cell>
|
|
<Table.Cell collapsing title="Environment">
|
|
Env
|
|
</Table.Cell>
|
|
</Table.Row>
|
|
{areas.map((area, i) => (
|
|
<tr key={area.id} className="Table__row candystripe">
|
|
<td>{area.name}</td>
|
|
<td className="Table__cell text-right text-nowrap">
|
|
<AreaCharge charging={area.charging} charge={area.charge} />
|
|
</td>
|
|
<td className="Table__cell text-right text-nowrap">
|
|
{area.load}
|
|
</td>
|
|
<td className="Table__cell text-center text-nowrap">
|
|
<AreaStatusColorBox status={area.eqp} />
|
|
</td>
|
|
<td className="Table__cell text-center text-nowrap">
|
|
<AreaStatusColorBox status={area.lgt} />
|
|
</td>
|
|
<td className="Table__cell text-center text-nowrap">
|
|
<AreaStatusColorBox status={area.env} />
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</Table>
|
|
</Section>
|
|
</Fragment>
|
|
);
|
|
};
|
|
|
|
export const AreaCharge = (props) => {
|
|
const { charging, charge } = props;
|
|
return (
|
|
<Fragment>
|
|
<Icon
|
|
width="18px"
|
|
textAlign="center"
|
|
name={
|
|
(charging === 0 &&
|
|
(charge > 50 ? 'battery-half' : 'battery-quarter')) ||
|
|
(charging === 1 && 'bolt') ||
|
|
(charging === 2 && 'battery-full')
|
|
}
|
|
color={
|
|
(charging === 0 && (charge > 50 ? 'yellow' : 'red')) ||
|
|
(charging === 1 && 'yellow') ||
|
|
(charging === 2 && 'green')
|
|
}
|
|
/>
|
|
<Box inline width="36px" textAlign="right">
|
|
{toFixed(charge) + '%'}
|
|
</Box>
|
|
</Fragment>
|
|
);
|
|
};
|
|
|
|
AreaCharge.defaultHooks = pureComponentHooks;
|
|
|
|
const AreaStatusColorBox = (props) => {
|
|
const { status } = props;
|
|
const power = Boolean(status & 2);
|
|
const mode = Boolean(status & 1);
|
|
const tooltipText = (power ? 'On' : 'Off') + ` [${mode ? 'auto' : 'manual'}]`;
|
|
return (
|
|
<ColorBox
|
|
color={power ? 'good' : 'bad'}
|
|
content={mode ? undefined : 'M'}
|
|
title={tooltipText}
|
|
/>
|
|
);
|
|
};
|
|
|
|
AreaStatusColorBox.defaultHooks = pureComponentHooks;
|