mirror of
https://github.com/bluewave-labs/Checkmate.git
synced 2026-05-16 22:48:40 -05:00
149 lines
3.9 KiB
React
149 lines
3.9 KiB
React
import {
|
|
Paper,
|
|
Table,
|
|
TableBody,
|
|
TableCell,
|
|
TableContainer,
|
|
TableHead,
|
|
TableRow,
|
|
} from "@mui/material";
|
|
import SkeletonLayout from "./skeleton";
|
|
import PropTypes from "prop-types";
|
|
import { useTheme } from "@emotion/react";
|
|
|
|
/**
|
|
* @typedef {Object} Header
|
|
* @property {number|string} id - The unique identifier for the header.
|
|
* @property {React.ReactNode} content - The content to display in the header cell.
|
|
* @property {Function} onClick - A function to be called when this cell is clicked, receiving the event and row data as an argument.
|
|
* @property {Object} getCellSx - Function that takes a row and returns a style object for the table cell.
|
|
* @property {Function} render - A function to render the cell content for a given row.
|
|
*/
|
|
|
|
/**
|
|
* @typedef {Object} Config
|
|
* @property {Function} onRowClick - A function to be called when a row is clicked, receiving the row data as an argument.
|
|
* @property {Object} rowSX - Style object for the table row.
|
|
*/
|
|
|
|
/**
|
|
* DataTable component renders a table with headers and data.
|
|
*
|
|
* @param {Object} props - The component props.
|
|
* @param {Header[]} props.headers - An array of header objects, each containing an `id`, `content`, and `render` function.
|
|
* @param {Array} props.data - An array of data objects, each representing a row.
|
|
* @returns {JSX.Element} The rendered table component.
|
|
*/
|
|
|
|
const DataTable = ({
|
|
shouldRender = true,
|
|
headers = [],
|
|
data = [],
|
|
config = {
|
|
emptyView: "No data",
|
|
onRowClick: () => {},
|
|
},
|
|
}) => {
|
|
const theme = useTheme();
|
|
if (!shouldRender) {
|
|
return <SkeletonLayout />;
|
|
}
|
|
|
|
if ((headers?.length ?? 0) === 0) {
|
|
return "No data";
|
|
}
|
|
|
|
return (
|
|
<TableContainer component={Paper}>
|
|
<Table
|
|
stickyHeader
|
|
sx={{
|
|
"&.MuiTable-root :is(.MuiTableHead-root, .MuiTableBody-root) :is(th, td)": {
|
|
paddingLeft: theme.spacing(8),
|
|
},
|
|
"& :is(th)": {
|
|
backgroundColor: theme.palette.secondary.main,
|
|
color: theme.palette.secondary.contrastText,
|
|
fontWeight: 600,
|
|
},
|
|
"& :is(td)": {
|
|
backgroundColor: theme.palette.primary.main,
|
|
color: theme.palette.primary.contrastTextSecondary,
|
|
},
|
|
"& .MuiTableBody-root .MuiTableRow-root:last-child .MuiTableCell-root": {
|
|
borderBottom: "none",
|
|
},
|
|
}}
|
|
>
|
|
<TableHead>
|
|
<TableRow>
|
|
{headers.map((header, index) => (
|
|
<TableCell
|
|
key={header.id}
|
|
align={index === 0 ? "left" : "center"}
|
|
>
|
|
{header.content}
|
|
</TableCell>
|
|
))}
|
|
</TableRow>
|
|
</TableHead>
|
|
<TableBody>
|
|
{(data?.length ?? 0) === 0 ? (
|
|
<TableRow>
|
|
<TableCell
|
|
colSpan={headers.length}
|
|
align="center"
|
|
>
|
|
{config.emptyView}
|
|
</TableCell>
|
|
</TableRow>
|
|
) : (
|
|
data.map((row) => {
|
|
const key = row.id || row._id || Math.random();
|
|
return (
|
|
<TableRow
|
|
key={key}
|
|
sx={config?.rowSX ?? {}}
|
|
onClick={config?.onRowClick ? () => config.onRowClick(row) : null}
|
|
>
|
|
{headers.map((header, index) => {
|
|
return (
|
|
<TableCell
|
|
align={index === 0 ? "left" : "center"}
|
|
key={header.id}
|
|
onClick={header.onClick ? (e) => header.onClick(e, row) : null}
|
|
sx={header.getCellSx ? header.getCellSx(row) : {}}
|
|
>
|
|
{header.render(row)}
|
|
</TableCell>
|
|
);
|
|
})}
|
|
</TableRow>
|
|
);
|
|
})
|
|
)}
|
|
</TableBody>
|
|
</Table>
|
|
</TableContainer>
|
|
);
|
|
};
|
|
|
|
DataTable.propTypes = {
|
|
shouldRender: PropTypes.bool,
|
|
headers: PropTypes.arrayOf(
|
|
PropTypes.shape({
|
|
id: PropTypes.oneOfType([PropTypes.number, PropTypes.string]).isRequired,
|
|
content: PropTypes.node.isRequired,
|
|
render: PropTypes.func.isRequired,
|
|
})
|
|
).isRequired,
|
|
data: PropTypes.array,
|
|
config: PropTypes.shape({
|
|
onRowClick: PropTypes.func,
|
|
rowSX: PropTypes.object,
|
|
emptyView: PropTypes.node,
|
|
}),
|
|
};
|
|
|
|
export default DataTable;
|