mirror of
https://github.com/emsesp/EMS-ESP32.git
synced 2025-12-06 15:59:52 +03:00
288 lines
8.5 KiB
TypeScript
288 lines
8.5 KiB
TypeScript
import { FC, useState, useContext, useEffect } from 'react';
|
|
import { useSnackbar } from 'notistack';
|
|
import {
|
|
Avatar,
|
|
Button,
|
|
List,
|
|
ListItem,
|
|
ListItemAvatar,
|
|
ListItemText,
|
|
Box,
|
|
Dialog,
|
|
DialogActions,
|
|
DialogContent,
|
|
DialogTitle,
|
|
Theme,
|
|
useTheme
|
|
} from '@mui/material';
|
|
|
|
import { Table } from '@table-library/react-table-library/table';
|
|
import { useTheme as tableTheme } from '@table-library/react-table-library/theme';
|
|
import { Header, HeaderRow, HeaderCell, Body, Row, Cell } from '@table-library/react-table-library/table';
|
|
|
|
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
|
import RefreshIcon from '@mui/icons-material/Refresh';
|
|
import PermScanWifiIcon from '@mui/icons-material/PermScanWifi';
|
|
import CancelIcon from '@mui/icons-material/Cancel';
|
|
import DirectionsBusIcon from '@mui/icons-material/DirectionsBus';
|
|
|
|
import { AuthenticatedContext } from '../contexts/authentication';
|
|
|
|
import { ButtonRow, FormLoader, SectionContent } from '../components';
|
|
|
|
import { Status, busConnectionStatus, Stat } from './types';
|
|
|
|
import { extractErrorMessage, useRest } from '../utils';
|
|
|
|
import * as EMSESP from './api';
|
|
|
|
import type { Translation } from '../i18n/i18n-types';
|
|
import { useI18nContext } from '../i18n/i18n-react';
|
|
|
|
export const isConnected = ({ status }: Status) => status !== busConnectionStatus.BUS_STATUS_OFFLINE;
|
|
|
|
const busStatusHighlight = ({ status }: Status, theme: Theme) => {
|
|
switch (status) {
|
|
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
|
return theme.palette.warning.main;
|
|
case busConnectionStatus.BUS_STATUS_CONNECTED:
|
|
return theme.palette.success.main;
|
|
case busConnectionStatus.BUS_STATUS_OFFLINE:
|
|
return theme.palette.error.main;
|
|
default:
|
|
return theme.palette.warning.main;
|
|
}
|
|
};
|
|
|
|
const showQuality = (stat: Stat) => {
|
|
if (stat.q === 0 || stat.s + stat.f === 0) {
|
|
return;
|
|
}
|
|
if (stat.q === 100) {
|
|
return <div style={{ color: '#00FF7F' }}>{stat.q}%</div>;
|
|
}
|
|
if (stat.q >= 95) {
|
|
return <div style={{ color: 'orange' }}>{stat.q}%</div>;
|
|
} else {
|
|
return <div style={{ color: 'red' }}>{stat.q}%</div>;
|
|
}
|
|
};
|
|
|
|
const DashboardStatus: FC = () => {
|
|
const { loadData, data, errorMessage } = useRest<Status>({ read: EMSESP.readStatus });
|
|
|
|
const { LL } = useI18nContext();
|
|
|
|
const theme = useTheme();
|
|
const [confirmScan, setConfirmScan] = useState<boolean>(false);
|
|
const { enqueueSnackbar } = useSnackbar();
|
|
|
|
const { me } = useContext(AuthenticatedContext);
|
|
|
|
const showName = (id: any) => {
|
|
let name: keyof Translation['STATUS_NAMES'] = id;
|
|
return LL.STATUS_NAMES[name]();
|
|
};
|
|
|
|
const busStatus = ({ status }: Status) => {
|
|
switch (status) {
|
|
case busConnectionStatus.BUS_STATUS_CONNECTED:
|
|
return LL.CONNECTED(0);
|
|
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
|
return LL.TX_ISSUES();
|
|
case busConnectionStatus.BUS_STATUS_OFFLINE:
|
|
return LL.DISCONNECTED();
|
|
default:
|
|
return 'Unknown';
|
|
}
|
|
};
|
|
|
|
const stats_theme = tableTheme({
|
|
Table: `
|
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 90px 90px 80px;
|
|
`,
|
|
BaseRow: `
|
|
font-size: 14px;
|
|
`,
|
|
HeaderRow: `
|
|
text-transform: uppercase;
|
|
background-color: black;
|
|
color: #90CAF9;
|
|
|
|
.th {
|
|
height: 42px;
|
|
font-weight: 500;
|
|
border-bottom: 1px solid #565656;
|
|
}
|
|
`,
|
|
Row: `
|
|
.td {
|
|
padding: 8px;
|
|
border-top: 1px solid #565656;
|
|
border-bottom: 1px solid #565656;
|
|
}
|
|
|
|
&:nth-of-type(odd) .td {
|
|
background-color: #303030;
|
|
}
|
|
&:nth-of-type(even) .td {
|
|
background-color: #1e1e1e;
|
|
}
|
|
`,
|
|
BaseCell: `
|
|
&:not(:first-of-type) {
|
|
text-align: center;
|
|
}
|
|
`
|
|
});
|
|
|
|
useEffect(() => {
|
|
const timer = setInterval(() => loadData(), 30000);
|
|
return () => {
|
|
clearInterval(timer);
|
|
};
|
|
// eslint-disable-next-line
|
|
}, []);
|
|
|
|
const scan = async () => {
|
|
try {
|
|
await EMSESP.scanDevices();
|
|
enqueueSnackbar(LL.SCANNING() + '...', { variant: 'info' });
|
|
} catch (error) {
|
|
enqueueSnackbar(extractErrorMessage(error, LL.PROBLEM_UPDATING()), { variant: 'error' });
|
|
} finally {
|
|
setConfirmScan(false);
|
|
}
|
|
};
|
|
|
|
const formatDurationSec = (duration_sec: number) => {
|
|
const days = Math.trunc((duration_sec * 1000) / 86400000);
|
|
const hours = Math.trunc((duration_sec * 1000) / 3600000) % 24;
|
|
const minutes = Math.trunc((duration_sec * 1000) / 60000) % 60;
|
|
const seconds = Math.trunc((duration_sec * 1000) / 1000) % 60;
|
|
|
|
let formatted = '';
|
|
if (days) {
|
|
formatted += LL.NUM_DAYS({ num: days }) + ' ';
|
|
}
|
|
if (hours) {
|
|
formatted += LL.NUM_HOURS({ num: hours }) + ' ';
|
|
}
|
|
if (minutes) {
|
|
formatted += LL.NUM_MINUTES({ num: minutes }) + ' ';
|
|
}
|
|
formatted += LL.NUM_SECONDS({ num: seconds });
|
|
return formatted;
|
|
};
|
|
|
|
const renderScanDialog = () => (
|
|
<Dialog open={confirmScan} onClose={() => setConfirmScan(false)}>
|
|
<DialogTitle>{LL.SCAN_DEVICES()}</DialogTitle>
|
|
<DialogContent dividers>{LL.EMS_SCAN()}</DialogContent>
|
|
<DialogActions>
|
|
<Button startIcon={<CancelIcon />} variant="outlined" onClick={() => setConfirmScan(false)} color="secondary">
|
|
{LL.CANCEL()}
|
|
</Button>
|
|
<Button startIcon={<PermScanWifiIcon />} variant="outlined" onClick={scan} color="primary" autoFocus>
|
|
{LL.SCAN()}
|
|
</Button>
|
|
</DialogActions>
|
|
</Dialog>
|
|
);
|
|
|
|
const content = () => {
|
|
if (!data) {
|
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<List>
|
|
<ListItem>
|
|
<ListItemAvatar>
|
|
<Avatar sx={{ bgcolor: busStatusHighlight(data, theme) }}>
|
|
<DirectionsBusIcon />
|
|
</Avatar>
|
|
</ListItemAvatar>
|
|
<ListItemText
|
|
primary={LL.EMS_BUS_STATUS()}
|
|
secondary={busStatus(data) + ' (' + formatDurationSec(data.uptime) + ')'}
|
|
/>
|
|
</ListItem>
|
|
<ListItem>
|
|
<ListItemAvatar>
|
|
<Avatar>
|
|
<DeviceHubIcon />
|
|
</Avatar>
|
|
</ListItemAvatar>
|
|
<ListItemText
|
|
primary={LL.ACTIVE_DEVICES()}
|
|
secondary={
|
|
LL.NUM_DEVICES({ num: data.num_devices }) +
|
|
', ' +
|
|
LL.NUM_TEMP_SENSORS({ num: data.num_sensors }) +
|
|
', ' +
|
|
LL.NUM_ANALOG_SENSORS({ num: data.num_analogs })
|
|
}
|
|
/>
|
|
</ListItem>
|
|
<Box m={3}></Box>
|
|
<Table data={{ nodes: data.stats }} theme={stats_theme} layout={{ custom: true }}>
|
|
{(tableList: any) => (
|
|
<>
|
|
<Header>
|
|
<HeaderRow>
|
|
<HeaderCell resize></HeaderCell>
|
|
<HeaderCell stiff>{LL.SUCCESS()}</HeaderCell>
|
|
<HeaderCell stiff>{LL.FAIL()}</HeaderCell>
|
|
<HeaderCell stiff>{LL.QUALITY()}</HeaderCell>
|
|
</HeaderRow>
|
|
</Header>
|
|
<Body>
|
|
{tableList.map((stat: Stat) => (
|
|
<Row key={stat.id} item={stat}>
|
|
<Cell>{showName(stat.id)}</Cell>
|
|
<Cell stiff>{Intl.NumberFormat().format(stat.s)}</Cell>
|
|
<Cell stiff>{Intl.NumberFormat().format(stat.f)}</Cell>
|
|
<Cell stiff>{showQuality(stat)}</Cell>
|
|
</Row>
|
|
))}
|
|
</Body>
|
|
</>
|
|
)}
|
|
</Table>
|
|
</List>
|
|
{renderScanDialog()}
|
|
<Box display="flex" flexWrap="wrap">
|
|
<Box flexGrow={1} sx={{ '& button': { mt: 2 } }}>
|
|
<Button startIcon={<RefreshIcon />} variant="outlined" color="secondary" onClick={loadData}>
|
|
{LL.REFRESH()}
|
|
</Button>
|
|
</Box>
|
|
<Box flexWrap="nowrap" whiteSpace="nowrap">
|
|
<ButtonRow>
|
|
<Button
|
|
startIcon={<PermScanWifiIcon />}
|
|
variant="outlined"
|
|
color="primary"
|
|
disabled={!me.admin}
|
|
onClick={() => setConfirmScan(true)}
|
|
>
|
|
{LL.SCAN_DEVICES()}
|
|
</Button>
|
|
</ButtonRow>
|
|
</Box>
|
|
</Box>
|
|
</>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<SectionContent title={LL.EMS_BUS_STATUS_TITLE()} titleGutter>
|
|
{content()}
|
|
</SectionContent>
|
|
);
|
|
};
|
|
|
|
export default DashboardStatus;
|