mirror of
https://github.com/bluewave-labs/Checkmate.git
synced 2026-01-24 19:01:01 -06:00
290 lines
8.3 KiB
JavaScript
290 lines
8.3 KiB
JavaScript
import PropTypes from "prop-types";
|
|
import {
|
|
TableContainer,
|
|
Table,
|
|
TableHead,
|
|
TableRow,
|
|
TableCell,
|
|
TableBody,
|
|
Paper,
|
|
Box,
|
|
TablePagination,
|
|
Stack,
|
|
Typography,
|
|
Button,
|
|
} from "@mui/material";
|
|
import ArrowDownwardRoundedIcon from "@mui/icons-material/ArrowDownwardRounded";
|
|
import ArrowUpwardRoundedIcon from "@mui/icons-material/ArrowUpwardRounded";
|
|
|
|
import { useState, useEffect, memo, useCallback, useRef } from "react";
|
|
import { useDispatch } from "react-redux";
|
|
import { useTheme } from "@emotion/react";
|
|
import LeftArrowDouble from "../../../assets/icons/left-arrow-double.svg?react";
|
|
import RightArrowDouble from "../../../assets/icons/right-arrow-double.svg?react";
|
|
import LeftArrow from "../../../assets/icons/left-arrow.svg?react";
|
|
import RightArrow from "../../../assets/icons/right-arrow.svg?react";
|
|
import SelectorVertical from "../../../assets/icons/selector-vertical.svg?react";
|
|
// import ActionsMenu from "../actionsMenu";
|
|
|
|
/**
|
|
* Component for pagination actions (first, previous, next, last).
|
|
*
|
|
* @component
|
|
* @param {Object} props
|
|
* @param {number} props.count - Total number of items.
|
|
* @param {number} props.page - Current page number.
|
|
* @param {number} props.rowsPerPage - Number of rows per page.
|
|
* @param {function} props.onPageChange - Callback function to handle page change.
|
|
*
|
|
* @returns {JSX.Element} Pagination actions component.
|
|
*/
|
|
const TablePaginationActions = (props) => {
|
|
const { count, page, rowsPerPage, onPageChange } = props;
|
|
const handleFirstPageButtonClick = (event) => {
|
|
onPageChange(event, 0);
|
|
};
|
|
const handleBackButtonClick = (event) => {
|
|
onPageChange(event, page - 1);
|
|
};
|
|
const handleNextButtonClick = (event) => {
|
|
onPageChange(event, page + 1);
|
|
};
|
|
const handleLastPageButtonClick = (event) => {
|
|
onPageChange(event, Math.max(0, Math.ceil(count / rowsPerPage) - 1));
|
|
};
|
|
|
|
return (
|
|
<Box sx={{ flexShrink: 0, ml: "24px" }}>
|
|
<Button
|
|
variant="group"
|
|
onClick={handleFirstPageButtonClick}
|
|
disabled={page === 0}
|
|
aria-label="first page"
|
|
>
|
|
<LeftArrowDouble />
|
|
</Button>
|
|
<Button
|
|
variant="group"
|
|
onClick={handleBackButtonClick}
|
|
disabled={page === 0}
|
|
aria-label="previous page"
|
|
>
|
|
<LeftArrow />
|
|
</Button>
|
|
<Button
|
|
variant="group"
|
|
onClick={handleNextButtonClick}
|
|
disabled={page >= Math.ceil(count / rowsPerPage) - 1}
|
|
aria-label="next page"
|
|
>
|
|
<RightArrow />
|
|
</Button>
|
|
<Button
|
|
variant="group"
|
|
onClick={handleLastPageButtonClick}
|
|
disabled={page >= Math.ceil(count / rowsPerPage) - 1}
|
|
aria-label="last page"
|
|
>
|
|
<RightArrowDouble />
|
|
</Button>
|
|
</Box>
|
|
);
|
|
};
|
|
|
|
TablePaginationActions.propTypes = {
|
|
count: PropTypes.number.isRequired,
|
|
page: PropTypes.number.isRequired,
|
|
rowsPerPage: PropTypes.number.isRequired,
|
|
onPageChange: PropTypes.func.isRequired,
|
|
};
|
|
|
|
const MaintenanceTable = ({
|
|
page,
|
|
setPage,
|
|
rowsPerPage,
|
|
setRowsPerPage,
|
|
sort,
|
|
setSort,
|
|
maintenanceWindows,
|
|
maintenanceWindowCount,
|
|
}) => {
|
|
const theme = useTheme();
|
|
const dispatch = useDispatch();
|
|
|
|
const handleChangePage = (event, newPage) => {
|
|
setPage(newPage);
|
|
};
|
|
|
|
const handleChangeRowsPerPage = (event) => {
|
|
dispatch(
|
|
setRowsPerPage({
|
|
value: parseInt(event.target.value, 10),
|
|
table: "monitors",
|
|
})
|
|
);
|
|
setPage(0);
|
|
};
|
|
|
|
/**
|
|
* Helper function to calculate the range of displayed rows.
|
|
* @returns {string}
|
|
*/
|
|
const getRange = () => {
|
|
let start = page * rowsPerPage + 1;
|
|
let end = Math.min(
|
|
page * rowsPerPage + rowsPerPage,
|
|
maintenanceWindowCount
|
|
);
|
|
return `${start} - ${end}`;
|
|
};
|
|
|
|
const handleSort = async (field) => {
|
|
let order = "";
|
|
if (sort.field !== field) {
|
|
order = "desc";
|
|
} else {
|
|
order = sort.order === "asc" ? "desc" : "asc";
|
|
}
|
|
setSort({ field, order });
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<TableContainer component={Paper}>
|
|
<Table>
|
|
<TableHead>
|
|
<TableRow>
|
|
<TableCell
|
|
sx={{ cursor: "pointer" }}
|
|
onClick={() => handleSort("name")}
|
|
>
|
|
<Box>
|
|
Host
|
|
<span
|
|
style={{
|
|
visibility: sort.field === "name" ? "visible" : "hidden",
|
|
}}
|
|
>
|
|
{sort.order === "asc" ? (
|
|
<ArrowUpwardRoundedIcon />
|
|
) : (
|
|
<ArrowDownwardRoundedIcon />
|
|
)}
|
|
</span>
|
|
</Box>
|
|
</TableCell>
|
|
<TableCell
|
|
sx={{ cursor: "pointer" }}
|
|
onClick={() => handleSort("status")}
|
|
>
|
|
{" "}
|
|
<Box width="max-content">
|
|
{" "}
|
|
Status
|
|
<span
|
|
style={{
|
|
visibility:
|
|
sort.field === "status" ? "visible" : "hidden",
|
|
}}
|
|
>
|
|
{sort.order === "asc" ? (
|
|
<ArrowUpwardRoundedIcon />
|
|
) : (
|
|
<ArrowDownwardRoundedIcon />
|
|
)}
|
|
</span>
|
|
</Box>
|
|
</TableCell>
|
|
<TableCell>Response Time</TableCell>
|
|
<TableCell>Type</TableCell>
|
|
<TableCell>Actions</TableCell>
|
|
</TableRow>
|
|
</TableHead>
|
|
<TableBody></TableBody>
|
|
</Table>
|
|
</TableContainer>
|
|
<Stack
|
|
direction="row"
|
|
alignItems="center"
|
|
justifyContent="space-between"
|
|
px={theme.spacing(4)}
|
|
>
|
|
<Typography px={theme.spacing(2)} variant="body2" sx={{ opacity: 0.7 }}>
|
|
Showing {getRange()} of {maintenanceWindowCount} maintenance window(s)
|
|
</Typography>
|
|
<TablePagination
|
|
component="div"
|
|
count={maintenanceWindowCount}
|
|
page={page}
|
|
onPageChange={handleChangePage}
|
|
rowsPerPage={rowsPerPage}
|
|
rowsPerPageOptions={[5, 10, 15, 25]}
|
|
onRowsPerPageChange={handleChangeRowsPerPage}
|
|
ActionsComponent={TablePaginationActions}
|
|
labelRowsPerPage="Rows per page"
|
|
labelDisplayedRows={({ page, count }) =>
|
|
`Page ${page + 1} of ${Math.max(0, Math.ceil(count / rowsPerPage))}`
|
|
}
|
|
slotProps={{
|
|
select: {
|
|
MenuProps: {
|
|
keepMounted: true,
|
|
disableScrollLock: true,
|
|
PaperProps: {
|
|
className: "pagination-dropdown",
|
|
sx: {
|
|
mt: 0,
|
|
mb: theme.spacing(2),
|
|
},
|
|
},
|
|
transformOrigin: { vertical: "bottom", horizontal: "left" },
|
|
anchorOrigin: { vertical: "top", horizontal: "left" },
|
|
sx: { mt: theme.spacing(-2) },
|
|
},
|
|
inputProps: { id: "pagination-dropdown" },
|
|
IconComponent: SelectorVertical,
|
|
sx: {
|
|
ml: theme.spacing(4),
|
|
mr: theme.spacing(12),
|
|
minWidth: theme.spacing(20),
|
|
textAlign: "left",
|
|
"&.Mui-focused > div": {
|
|
backgroundColor: theme.palette.background.main,
|
|
},
|
|
},
|
|
},
|
|
}}
|
|
sx={{
|
|
mt: theme.spacing(6),
|
|
color: theme.palette.text.secondary,
|
|
"& svg path": {
|
|
stroke: theme.palette.text.tertiary,
|
|
strokeWidth: 1.3,
|
|
},
|
|
"& .MuiSelect-select": {
|
|
border: 1,
|
|
borderColor: theme.palette.border.light,
|
|
borderRadius: theme.shape.borderRadius,
|
|
},
|
|
}}
|
|
/>
|
|
</Stack>
|
|
</>
|
|
);
|
|
};
|
|
|
|
MaintenanceTable.propTypes = {
|
|
isAdmin: PropTypes.bool,
|
|
page: PropTypes.number,
|
|
setPage: PropTypes.func,
|
|
rowsPerPage: PropTypes.number,
|
|
setRowsPerPage: PropTypes.func,
|
|
sort: PropTypes.object,
|
|
setSort: PropTypes.func,
|
|
maintenanceWindows: PropTypes.array,
|
|
maintenanceWindowCount: PropTypes.number,
|
|
};
|
|
|
|
const MemoizedMaintenanceTable = memo(MaintenanceTable);
|
|
export default MemoizedMaintenanceTable;
|