MkTable.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. import React, { useEffect } from 'react';
  2. import PropTypes from 'prop-types';
  3. import Table from '@material-ui/core/Table';
  4. import TableBody from '@material-ui/core/TableBody';
  5. import TableCell from '@material-ui/core/TableCell';
  6. import TableContainer from '@material-ui/core/TableContainer';
  7. import TableHead from '@material-ui/core/TableHead';
  8. import TablePagination from '@material-ui/core/TablePagination';
  9. import TableRow from '@material-ui/core/TableRow';
  10. import TableSortLabel from '@material-ui/core/TableSortLabel';
  11. import Typography from '@material-ui/core/Typography';
  12. import EditIcon from '@material-ui/icons/Edit';
  13. import ExpandMore from '@material-ui/icons/ExpandMore';
  14. import { Box, Button, createMuiTheme, Dialog, DialogActions, DialogTitle, Grid, IconButton, Menu, MenuItem, ThemeProvider } from '@material-ui/core';
  15. import CircularProgress from '@material-ui/core/CircularProgress';
  16. import { makeStyles } from '@material-ui/core/styles';
  17. function descendingComparator(a, b, _orderBy) {
  18. if (b[_orderBy] < a[_orderBy]) {
  19. return -1;
  20. }
  21. if (b[_orderBy] > a[_orderBy]) {
  22. return 1;
  23. }
  24. return 0;
  25. }
  26. function getComparator(_order, _orderBy) {
  27. return _order === 'desc'
  28. ? (a, b) => descendingComparator(a, b, _orderBy)
  29. : (a, b) => -descendingComparator(a, b, _orderBy);
  30. }
  31. function stableSort(array, comparator) {
  32. const stabilizedThis = array.map((el, index) => [el, index]);
  33. stabilizedThis.sort((a, b) => {
  34. const _order = comparator(a[0], b[0]);
  35. if (_order !== 0) return _order;
  36. return a[1] - b[1];
  37. });
  38. return stabilizedThis.map((el) => el[0]);
  39. }
  40. // const StyledTableCell = withStyles((theme) => ({
  41. // head: {
  42. // backgroundColor: '#0d47a1a8',
  43. // color: theme.palette.common.white,
  44. // },
  45. // body: {
  46. // fontSize: 14,
  47. // },
  48. // }))(TableCell);
  49. function EnhancedTableHead(props) {
  50. const { headerStyles, _order, _orderBy, onRequestSort, headCells } = props;
  51. const createSortHandler = (property) => (event) => {
  52. onRequestSort(event, property);
  53. // onGetData();
  54. };
  55. const sortedStyle = createMuiTheme({
  56. overrides: {
  57. MuiTableSortLabel: {
  58. root: {
  59. "&$active": {
  60. color: headerStyles.sortedHeader === undefined ? 'grey' : headerStyles.sortedHeader,
  61. "&& $icon": {
  62. color: headerStyles.sortedHeader === undefined ? 'grey' : headerStyles.sortedHeader
  63. }
  64. }
  65. }
  66. }
  67. }
  68. });
  69. return (
  70. <ThemeProvider theme={sortedStyle}>
  71. <TableHead>
  72. <TableRow>
  73. {headCells.map((headCell) => (
  74. <TableCell
  75. className={headerStyles.head}
  76. key={headCell.id}
  77. align={headCell.numeric ? 'right' : 'left'}
  78. padding={headCell.disablePadding ? 'none' : 'default'}
  79. sortDirection={_orderBy === headCell.id ? _order : false}
  80. style={{ width: headCell.id === 'sr' ? '15px' : headCell.width ? headCell.width : '' }}
  81. >
  82. {headCell.id !== 'sr' ?
  83. <TableSortLabel
  84. active={_orderBy === headCell.id}
  85. direction={_orderBy === headCell.id ? _order : 'asc'}
  86. onClick={createSortHandler(headCell.id)}
  87. style={{ whiteSpace: "nowrap" }}
  88. >
  89. {headCell.label}
  90. {_orderBy === headCell.id ? (
  91. <span className={headerStyles.visuallyHidden}>
  92. {_order === 'desc' ? 'sorted descending' : 'sorted ascending'}
  93. </span>
  94. ) : null}
  95. </TableSortLabel>
  96. : <TableSortLabel
  97. hideSortIcon="true"
  98. align="right"
  99. >
  100. {headCell.label}
  101. </TableSortLabel>
  102. }
  103. </TableCell>
  104. ))}
  105. <TableCell className={headerStyles.head} style={{ width: '100px' }} />
  106. </TableRow>
  107. </TableHead>
  108. </ThemeProvider>
  109. );
  110. }
  111. EnhancedTableHead.propTypes = {
  112. headerStyles: PropTypes.object.isRequired,
  113. onRequestSort: PropTypes.func.isRequired,
  114. onSelectAllClick: PropTypes.func.isRequired,
  115. _order: PropTypes.oneOf(['asc', 'desc']).isRequired,
  116. _orderBy: PropTypes.string.isRequired,
  117. rowCount: PropTypes.number.isRequired,
  118. };
  119. const useStyles = makeStyles({
  120. root: {
  121. width: '100%',
  122. },
  123. table: {
  124. // minWidth: 750,
  125. tableLayout: 'fixed'
  126. },
  127. visuallyHidden: {
  128. b_order: 0,
  129. clip: 'rect(0 0 0 0)',
  130. height: 1,
  131. margin: -1,
  132. overflow: 'hidden',
  133. padding: 0,
  134. position: 'absolute',
  135. top: 20,
  136. width: 1,
  137. },
  138. underline: {
  139. "&&&:before": {
  140. b_orderBottom: "none"
  141. },
  142. "&&:after": {
  143. b_orderBottom: "none"
  144. }
  145. },
  146. head: props => ({
  147. backgroundColor: props.headerBackgroundColor === undefined ? '#0d47a1a8' : props.headerBackgroundColor,
  148. color: props.headerTextColor === undefined ? 'white' : props.headerTextColor,
  149. }),
  150. sortedHeader: props => ({
  151. color: props.headerTextColor === undefined ? 'grey' : props.headerTextColor,
  152. }),
  153. styledTableRow: props => ({
  154. '&:nth-of-type(even)': {
  155. backgroundColor: props.primaryColor === undefined ? '#0d47a11c' : props.primaryColor,
  156. },
  157. }),
  158. });
  159. // const StyledTableRow = withStyles((theme) => ({
  160. // root: {
  161. // '&:nth-of-type(even)': {
  162. // backgroundColor: '#0d47a11c',
  163. // },
  164. // },
  165. // }))(TableRow);
  166. function getUpdatedDate(p) {
  167. var statusDate = p.updated_date;
  168. var day = '';
  169. if (statusDate !== undefined) {
  170. var convertDate = new Date(statusDate.toDate());
  171. var dd = String(convertDate.getDate()).padStart(2, '0');
  172. var mm = String(convertDate.getMonth() + 1).padStart(2, '0');
  173. var yyyy = convertDate.getFullYear();
  174. day = mm + '/' + dd + '/' + yyyy;
  175. }
  176. return day.toString();
  177. }
  178. export const formatDateToLocal = (date, withTime = true) => {
  179. if (!date) return "";
  180. var d = new Date(date);
  181. var ampm = d.getHours() >= 12 ? 'PM' : 'AM';
  182. var hours = d.getHours() > 12 ? d.getHours() - 12 : d.getHours();
  183. if (withTime) {
  184. return ("0" + (d.getMonth() + 1)).slice(-2) + "/" + ("0"
  185. + d.getDate()).slice(-2) + "/" + d.getFullYear()
  186. + " " + ("0" + (hours)).slice(-2)
  187. + ":" + ("0" + (d.getMinutes())).slice(-2)
  188. + ":" + ("0" + (d.getSeconds())).slice(-2)
  189. + " " + ampm;
  190. } else {
  191. return ("0" + (d.getMonth() + 1)).slice(-2) + "/" + ("0"
  192. + d.getDate()).slice(-2) + "/" + d.getFullYear();
  193. }
  194. }
  195. function RowMenu(props) {
  196. const { row, actions, onSelectedAction, onRowEdit } = props;
  197. const [anchorEl, setAnchorEl] = React.useState(null);
  198. const handleMenuClick = (event) => {
  199. event.stopPropagation();
  200. setAnchorEl(event.currentTarget);
  201. };
  202. const handleSelectMenu = (e, row, action) => {
  203. e.stopPropagation();
  204. onSelectedAction(row, action.action_name);
  205. setAnchorEl(null);
  206. }
  207. const handleClose = (e) => {
  208. setAnchorEl(null);
  209. e.stopPropagation();
  210. };
  211. const handleEdit = (e) => {
  212. onRowEdit(row);
  213. e.stopPropagation();
  214. }
  215. return (
  216. <div>
  217. <Menu
  218. id={`actions-${row.id}`}
  219. anchorEl={anchorEl}
  220. keepMounted
  221. open={Boolean(anchorEl)}
  222. onClose={handleClose}
  223. >
  224. {actions.map((action) => {
  225. return <MenuItem key={action.display_name}
  226. onClick={(e) => handleSelectMenu(e, row, action)}
  227. >
  228. {action.display_name}
  229. </MenuItem>
  230. }
  231. )}
  232. </Menu>
  233. <Grid
  234. container
  235. direction="row"
  236. justify="flex-end"
  237. alignItems="center"
  238. style={{ display: 'flex' }}>
  239. <IconButton
  240. id={`edit-${row.id}`}
  241. aria-label="more"
  242. aria-controls="long-menu"
  243. aria-haspopup="true"
  244. onClick={handleEdit}
  245. size="small"
  246. >
  247. <EditIcon />
  248. </IconButton>
  249. <Box style={{ width: '10px' }} />
  250. <IconButton
  251. id={`dropdown-${row.id}`}
  252. aria-label="more"
  253. aria-controls="long-menu"
  254. aria-haspopup="true"
  255. size="small"
  256. onClick={handleMenuClick}
  257. >
  258. <ExpandMore />
  259. </IconButton>
  260. </Grid>
  261. </div>
  262. );
  263. }
  264. RowMenu.propTypes = {
  265. row: PropTypes.object.isRequired,
  266. actions: PropTypes.array.isRequired,
  267. onSelectedAction: PropTypes.func.isRequired,
  268. onRowEdit: PropTypes.func
  269. }
  270. function ConfirmDialog(props) {
  271. const { type, itemName, openDialog, onCancel, onContinue } = props;
  272. const [open, setOpen] = React.useState(openDialog);
  273. const handleClose = () => {
  274. setOpen(false);
  275. onCancel(false);
  276. };
  277. const handleContinue = () => {
  278. onContinue(true);
  279. }
  280. return (
  281. <div>
  282. <Dialog
  283. open={open}
  284. onClose={handleClose}
  285. aria-labelledby="alert-dialog-title"
  286. aria-describedby="alert-dialog-description"
  287. >
  288. <DialogTitle id="alert-dialog-title">{"Delete this " + type + ' "' + itemName + '"?'}</DialogTitle>
  289. <DialogActions>
  290. <Button onClick={handleClose} color="primary">
  291. Cancel
  292. </Button>
  293. <Button onClick={handleContinue} color="primary" autoFocus>
  294. Delete
  295. </Button>
  296. </DialogActions>
  297. </Dialog>
  298. </div>
  299. );
  300. }
  301. ConfirmDialog.propTypes = {
  302. history: PropTypes.object,
  303. type: PropTypes.string,
  304. itemName: PropTypes.string,
  305. openDialog: PropTypes.bool,
  306. onCancel: PropTypes.func,
  307. onContinue: PropTypes.func
  308. };
  309. function MkTable(props) {
  310. console.log("props.styles:", props.styles)
  311. const classes = useStyles(props.styles);
  312. const { dispatch,
  313. data = [],
  314. headers = [],
  315. actions,
  316. onActions, title,
  317. page,
  318. rowsPerPage,
  319. noMoreToLoad = false,
  320. order,
  321. orderBy,
  322. isLoading = false,
  323. onChangePaginatePage,
  324. onGetData,
  325. onUpdateDataRow,
  326. onChangeRowPerPage,
  327. dense = true,
  328. } = props;
  329. const [_rowsPerPage, setRowsPerPage] = React.useState(rowsPerPage);
  330. const [_page, setPage] = React.useState(page);
  331. const [_noMoreToLoad, setNoMoreToLoad] = React.useState(noMoreToLoad);
  332. const [_order, setOrder] = React.useState(order);
  333. const [_orderBy, setOrderBy] = React.useState(orderBy);
  334. const [_isLoading, setIsLoading] = React.useState(isLoading);
  335. const [_isConfirm, setIsConfirm] = React.useState(false);
  336. const [itemName, setItemName] = React.useState('');
  337. const [row, setRow] = React.useState({});
  338. const [action, setAction] = React.useState('');
  339. const [_dense, setDense] = React.useState(dense);
  340. const handleSelectMenu = (row, action) => {
  341. if (action === 'delete') {
  342. setItemName(row.name === undefined ? row.product_desc : row.name);
  343. setIsConfirm(true);
  344. setRow(row);
  345. setAction(action);
  346. } else {
  347. onActions(row, action);
  348. }
  349. }
  350. const handleDelete = (v) => {
  351. setIsConfirm(false);
  352. onActions(row, action);
  353. }
  354. const handleCancel = (v) => {
  355. setIsConfirm(false);
  356. }
  357. const handleRowEdit = (row) => {
  358. onUpdateDataRow(row);
  359. }
  360. var offset = _page * _rowsPerPage;
  361. useEffect(() => {
  362. setNoMoreToLoad(noMoreToLoad);
  363. setPage(page);
  364. setOrder(order);
  365. setOrderBy(orderBy);
  366. setIsLoading(isLoading);
  367. setRowsPerPage(rowsPerPage);
  368. setDense(dense);
  369. }, []);
  370. const handleRequestSort = (event, property) => {
  371. const isAsc = _orderBy === property && _order === 'asc';
  372. setOrder(isAsc ? 'desc' : 'asc');
  373. setOrderBy(property);
  374. };
  375. const handleChangePage = (event, newPage) => {
  376. if (!_noMoreToLoad && (newPage + 1) * _rowsPerPage >= data.length) {
  377. onGetData();
  378. }
  379. setPage(newPage);
  380. onChangePaginatePage(newPage);
  381. };
  382. const handleChangeRowsPerPage = (event) => {
  383. setRowsPerPage(parseInt(event.target.value));
  384. onChangeRowPerPage(parseInt(event.target.value));
  385. setPage(0);
  386. };
  387. const getStatus = (data, header) => {
  388. var v = data[header.id];
  389. var _color = 'red';
  390. if (v === 'Pending') {
  391. _color = 'red';
  392. } else if (v === 'Started') {
  393. _color = 'orange';
  394. } else {
  395. _color = 'green';
  396. }
  397. return (<TableCell key={header.id} align={header.numeric ? 'right' : 'left'} ><Typography style={{ color: _color, fontWeight: '500' }}>{data[header.id]}</Typography></TableCell>);
  398. };
  399. return (
  400. <div className={classes.root}>
  401. <Grid container>
  402. <Grid item>
  403. <TableContainer>
  404. <Table
  405. className={classes.table}
  406. aria-labelledby="tableTitle"
  407. size={_dense ? 'small' : 'medium'}
  408. aria-label="enhanced table"
  409. >
  410. <EnhancedTableHead
  411. headerStyles={classes}
  412. headCells={headers}
  413. _order={_order}
  414. _orderBy={_orderBy}
  415. onRequestSort={handleRequestSort}
  416. rowCount={data.length !== undefined ? data.length : 0}
  417. dispatch={dispatch}
  418. />
  419. <TableBody>
  420. {_isLoading ? <TableRow className={classes.styledTableRow}>
  421. <TableCell colSpan={headers.length} align="center"> <CircularProgress /></TableCell>
  422. </TableRow> :
  423. (data.length !== 0 ? stableSort(data, getComparator(_order, _orderBy))
  424. .slice(_page * _rowsPerPage, _page * _rowsPerPage + _rowsPerPage)
  425. .map((row, index) => {
  426. return (
  427. <TableRow
  428. hover
  429. role="checkbox"
  430. tabIndex={-1}
  431. key={row.id}
  432. id={row.id}
  433. className={classes.styledTableRow}
  434. >
  435. {headers.map((h, i) => {
  436. if (h.id === 'sr') {
  437. return (<TableCell key={h.id} align="right" style={{ width: '15px' }}>{++offset}</TableCell>);
  438. }
  439. if (h.id === 'status') {
  440. getStatus(row, h);
  441. return (<TableCell key={h.id} align={h.numeric ? 'right' : 'left'}>{row[h.id]}</TableCell>);
  442. }
  443. if (h.id === 'updated_date') {
  444. return (<TableCell key={h.id} align={h.numeric ? 'right' : 'left'} style={{ width: h.width ? h.width : null }}>{getUpdatedDate(row)}</TableCell>);
  445. } else {
  446. return (<TableCell key={h.id} align={h.numeric ? 'right' : 'left'} style={{ width: h.width ? h.width : null }}
  447. >{row[h.id]}</TableCell>);
  448. }
  449. })}
  450. {actions ?
  451. <TableCell style={{ width: '150px' }} align='right'>
  452. <RowMenu
  453. actions={actions}
  454. row={row}
  455. onRowEdit={(data) => handleRowEdit(data)}
  456. onSelectedAction={(data, actionName) => handleSelectMenu(data, actionName)}
  457. />
  458. </TableCell>
  459. : <TableCell style={{ width: '150px' }} align='right'>
  460. <IconButton onClick={(event) => handleRowEdit(row)} size={dense ? "small" : "medium"}><EditIcon /></IconButton>
  461. </TableCell>}
  462. </TableRow>
  463. );
  464. }) : <TableRow className={classes.styledTableRow} style={{ width: '100%' }} />)}
  465. </TableBody>
  466. </Table>
  467. </TableContainer>
  468. <TablePagination
  469. rowsPerPageOptions={[10, 30, 50]}
  470. labelDisplayedRows={({ from, to, count }) => { console.log(from, to, count) }}
  471. component="div"
  472. count={data.length}
  473. rowsPerPage={_rowsPerPage}
  474. page={_page}
  475. onChangePage={handleChangePage}
  476. onChangeRowsPerPage={handleChangeRowsPerPage}
  477. />
  478. </Grid></Grid>
  479. {_isConfirm ? <ConfirmDialog
  480. type={title}
  481. itemName={itemName}
  482. openDialog={_isConfirm}
  483. onCancel={(v) => handleCancel(v)}
  484. onContinue={(v) => handleDelete(v)} /> : <div />}
  485. </div>
  486. );
  487. }
  488. MkTable.propTypes = {
  489. history: PropTypes.object,
  490. headers: PropTypes.array.isRequired,
  491. data: PropTypes.array.isRequired,
  492. query: PropTypes.object,
  493. onProductBOMClick: PropTypes.func,
  494. onActions: PropTypes.func,
  495. actions: PropTypes.array,
  496. title: PropTypes.string,
  497. order: PropTypes.any,
  498. orderBy: PropTypes.any,
  499. rowsPerPage: PropTypes.any,
  500. noMoreToLoad: PropTypes.any,
  501. isLoading: PropTypes.any,
  502. onChangePaginatePage: PropTypes.any,
  503. onGetData: PropTypes.any,
  504. onChangeRowPerPage: PropTypes.any,
  505. dense: PropTypes.any,
  506. styles: PropTypes.any
  507. };
  508. export default (MkTable);