diff --git a/frontend/src/component/executiveDashboard/ExecutiveDashboard.tsx b/frontend/src/component/executiveDashboard/ExecutiveDashboard.tsx index 6632407906..f623492e07 100644 --- a/frontend/src/component/executiveDashboard/ExecutiveDashboard.tsx +++ b/frontend/src/component/executiveDashboard/ExecutiveDashboard.tsx @@ -1,5 +1,33 @@ +import { Box, Paper, styled, Typography } from '@mui/material'; +import { PageHeader } from 'component/common/PageHeader/PageHeader'; import { VFC } from 'react'; +import { UsersChart } from './UsersChart/UsersChart'; + +const StyledGrid = styled(Box)(({ theme }) => ({ + display: 'grid', + gridTemplateColumns: `repeat(auto-fill, minmax(600px, 1fr))`, + gridAutoRows: '1fr', + gap: theme.spacing(2), +})); export const ExecutiveDashboard: VFC = () => { - return <>test; + return ( + <> + ({ paddingBottom: theme.spacing(4) })}> + + Dashboard + + } + // subtitle='Succesfully synchronized: 01 Sep 2023 - 07:05:07' + /> + + {/* Dashboard */} + + Stats + + + + ); }; diff --git a/frontend/src/component/executiveDashboard/UsersChart/UsersChart.tsx b/frontend/src/component/executiveDashboard/UsersChart/UsersChart.tsx new file mode 100644 index 0000000000..ee5a468d68 --- /dev/null +++ b/frontend/src/component/executiveDashboard/UsersChart/UsersChart.tsx @@ -0,0 +1,3 @@ +import { lazy } from 'react'; + +export const UsersChart = lazy(() => import('./UsersChartComponent')); diff --git a/frontend/src/component/executiveDashboard/UsersChart/UsersChartComponent.tsx b/frontend/src/component/executiveDashboard/UsersChart/UsersChartComponent.tsx new file mode 100644 index 0000000000..c1ef0659b5 --- /dev/null +++ b/frontend/src/component/executiveDashboard/UsersChart/UsersChartComponent.tsx @@ -0,0 +1,407 @@ +import { type VFC } from 'react'; +import { + Chart as ChartJS, + CategoryScale, + LinearScale, + PointElement, + LineElement, + Title, + Tooltip, + Legend, + TimeScale, +} from 'chart.js'; +import { Line } from 'react-chartjs-2'; +import 'chartjs-adapter-date-fns'; +import { Paper, Theme, useTheme } from '@mui/material'; +import { + useLocationSettings, + type ILocationSettings, +} from 'hooks/useLocationSettings'; +import { formatDateYMD } from 'utils/formatDate'; + +type Data = { + date: string | Date; + total?: number; + active?: number; + inactive?: number; +}[]; + +const mockData: Data = [ + { + date: '2023-01-21', + }, + { + date: '2023-01-28', + }, + { + date: '2023-02-04', + }, + { + date: '2023-02-11', + }, + { + date: '2023-02-18', + }, + { + date: '2023-02-25', + }, + { + date: '2023-03-04', + }, + { + date: '2023-03-11', + }, + { + date: '2023-03-18', + }, + { + date: '2023-03-25', + }, + { + date: '2023-04-01', + }, + { + date: '2023-04-08', + }, + { + date: '2023-04-15', + }, + { + date: '2023-04-22', + total: 43, + active: 0, + inactive: 0, + }, + { + date: '2023-04-29', + total: 54, + active: 54, + inactive: 0, + }, + { + date: '2023-05-06', + total: 63, + active: 63, + inactive: 0, + }, + { + date: '2023-05-13', + total: 81, + active: 81, + inactive: 0, + }, + { + date: '2023-05-20', + total: 80, + active: 80, + inactive: 0, + }, + { + date: '2023-05-27', + total: 95, + active: 95, + inactive: 0, + }, + { + date: '2023-06-03', + total: 108, + active: 108, + inactive: 0, + }, + { + date: '2023-06-10', + total: 101, + active: 101, + inactive: 0, + }, + { + date: '2023-06-17', + total: 104, + active: 104, + inactive: 0, + }, + { + date: '2023-06-24', + total: 114, + active: 114, + inactive: 0, + }, + { + date: '2023-07-01', + total: 108, + active: 106, + inactive: 2, + }, + { + date: '2023-07-08', + total: 103, + active: 102, + inactive: 1, + }, + { + date: '2023-07-15', + total: 106, + active: 105, + inactive: 1, + }, + { + date: '2023-07-22', + total: 112, + active: 106, + inactive: 6, + }, + { + date: '2023-07-29', + total: 113, + active: 107, + inactive: 6, + }, + { + date: '2023-08-05', + total: 109, + active: 98, + inactive: 11, + }, + { + date: '2023-08-12', + total: 110, + active: 96, + inactive: 14, + }, + { + date: '2023-08-19', + total: 127, + active: 111, + inactive: 16, + }, + { + date: '2023-08-26', + total: 140, + active: 124, + inactive: 16, + }, + { + date: '2023-09-02', + total: 150, + active: 130, + inactive: 20, + }, + { + date: '2023-09-09', + total: 168, + active: 148, + inactive: 20, + }, + { + date: '2023-09-16', + total: 171, + active: 154, + inactive: 17, + }, + { + date: '2023-09-23', + total: 190, + active: 174, + inactive: 16, + }, + { + date: '2023-09-30', + total: 186, + active: 169, + inactive: 17, + }, + { + date: '2023-10-07', + total: 188, + active: 173, + inactive: 15, + }, + { + date: '2023-10-14', + total: 181, + active: 166, + inactive: 15, + }, + { + date: '2023-10-21', + total: 192, + active: 177, + inactive: 15, + }, + { + date: '2023-10-28', + total: 183, + active: 164, + inactive: 19, + }, + { + date: '2023-11-04', + total: 200, + active: 180, + inactive: 20, + }, + { + date: '2023-11-11', + total: 212, + active: 189, + inactive: 23, + }, + { + date: '2023-11-18', + total: 204, + active: 177, + inactive: 27, + }, + { + date: '2023-11-25', + total: 200, + active: 173, + inactive: 27, + }, + { + date: '2023-12-02', + total: 200, + active: 175, + inactive: 25, + }, + { + date: '2023-12-09', + total: 200, + active: 176, + inactive: 24, + }, + { + date: '2023-12-16', + total: 215, + active: 186, + inactive: 29, + }, + { + date: '2023-12-23', + total: 221, + active: 195, + inactive: 26, + }, + { + date: '2023-12-30', + total: 214, + active: 184, + inactive: 30, + }, + { + date: '2024-01-06', + total: 204, + active: 173, + inactive: 31, + }, + { + date: '2024-01-13', + total: 215, + active: 181, + inactive: 34, + }, +]; + +const createData = (theme: Theme) => ({ + labels: mockData.map((item) => item.date), + datasets: [ + { + label: 'Total users', + data: mockData.map((item) => item.total), + borderColor: theme.palette.primary.main, + backgroundColor: theme.palette.primary.main, + fill: true, + }, + { + label: 'Inactive users', + data: mockData.map((item) => item.inactive), + borderColor: theme.palette.error.main, + backgroundColor: theme.palette.error.main, + fill: true, + }, + { + label: 'Active users', + data: mockData.map((item) => item.active), + borderColor: theme.palette.success.main, + backgroundColor: theme.palette.success.main, + fill: true, + }, + ], +}); + +const createOptions = (theme: Theme, locationSettings: ILocationSettings) => + ({ + responsive: true, + plugins: { + legend: { + position: 'bottom', + }, + tooltip: { + callbacks: { + title: (tooltipItems: any) => { + const item = tooltipItems?.[0]; + const date = + item?.chart?.data?.labels?.[item.dataIndex]; + return date + ? formatDateYMD(date, locationSettings.locale) + : ''; + }, + }, + }, + }, + locale: locationSettings.locale, + interaction: { + intersect: false, + axis: 'x', + }, + color: theme.palette.text.secondary, + scales: { + y: { + type: 'linear', + grid: { + color: theme.palette.divider, + borderColor: theme.palette.divider, + }, + ticks: { color: theme.palette.text.secondary }, + }, + x: { + type: 'time', + time: { + unit: 'month', + }, + grid: { + color: theme.palette.divider, + borderColor: theme.palette.divider, + }, + ticks: { + color: theme.palette.text.secondary, + }, + }, + }, + }) as const; + +const UsersChartComponent: VFC = () => { + const theme = useTheme(); + const { locationSettings } = useLocationSettings(); + const data = createData(theme); + const options = createOptions(theme, locationSettings); + + return ( + ({ padding: theme.spacing(4) })}> + + + ); +}; + +ChartJS.register( + CategoryScale, + LinearScale, + PointElement, + LineElement, + TimeScale, + Title, + Tooltip, + Legend, +); + +export default UsersChartComponent;