mirror of
				https://github.com/hibiken/asynqmon.git
				synced 2025-10-26 16:26:12 +08:00 
			
		
		
		
	(ui): Convert all tasks tables to use TasksTable component
This commit is contained in:
		| @@ -1,23 +1,13 @@ | ||||
| import Checkbox from "@material-ui/core/Checkbox"; | ||||
| import IconButton from "@material-ui/core/IconButton"; | ||||
| import Paper from "@material-ui/core/Paper"; | ||||
| import { makeStyles } from "@material-ui/core/styles"; | ||||
| import Table from "@material-ui/core/Table"; | ||||
| import TableBody from "@material-ui/core/TableBody"; | ||||
| import TableCell from "@material-ui/core/TableCell"; | ||||
| import TableContainer from "@material-ui/core/TableContainer"; | ||||
| import TableFooter from "@material-ui/core/TableFooter"; | ||||
| import TableHead from "@material-ui/core/TableHead"; | ||||
| import TablePagination from "@material-ui/core/TablePagination"; | ||||
| import TableRow from "@material-ui/core/TableRow"; | ||||
| import Tooltip from "@material-ui/core/Tooltip"; | ||||
| import ArchiveIcon from "@material-ui/icons/Archive"; | ||||
| import DeleteIcon from "@material-ui/icons/Delete"; | ||||
| import FileCopyOutlinedIcon from "@material-ui/icons/FileCopyOutlined"; | ||||
| import MoreHorizIcon from "@material-ui/icons/MoreHoriz"; | ||||
| import Alert from "@material-ui/lab/Alert"; | ||||
| import AlertTitle from "@material-ui/lab/AlertTitle"; | ||||
| import React, { useCallback, useState } from "react"; | ||||
| import React from "react"; | ||||
| import { connect, ConnectedProps } from "react-redux"; | ||||
| import { useHistory } from "react-router-dom"; | ||||
| import { taskRowsPerPageChange } from "../actions/settingsActions"; | ||||
| @@ -30,33 +20,12 @@ import { | ||||
|   deletePendingTaskAsync, | ||||
|   listPendingTasksAsync, | ||||
| } from "../actions/tasksActions"; | ||||
| import { usePolling } from "../hooks"; | ||||
| import { taskDetailsPath } from "../paths"; | ||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||
| import { AppState } from "../store"; | ||||
| import { TableColumn } from "../types/table"; | ||||
| import { prettifyPayload, uuidPrefix } from "../utils"; | ||||
| import SyntaxHighlighter from "./SyntaxHighlighter"; | ||||
| import TableActions from "./TableActions"; | ||||
| import TablePaginationActions, { | ||||
|   rowsPerPageOptions, | ||||
| } from "./TablePaginationActions"; | ||||
|  | ||||
| const useStyles = makeStyles((theme) => ({ | ||||
|   table: { | ||||
|     minWidth: 650, | ||||
|   }, | ||||
|   stickyHeaderCell: { | ||||
|     background: theme.palette.background.paper, | ||||
|   }, | ||||
|   alert: { | ||||
|     borderTopLeftRadius: 0, | ||||
|     borderTopRightRadius: 0, | ||||
|   }, | ||||
|   pagination: { | ||||
|     border: "none", | ||||
|   }, | ||||
| })); | ||||
| import TasksTable, { RowProps, useRowStyles } from "./TasksTable"; | ||||
|  | ||||
| function mapStateToProps(state: AppState) { | ||||
|   return { | ||||
| @@ -71,13 +40,13 @@ function mapStateToProps(state: AppState) { | ||||
| } | ||||
|  | ||||
| const mapDispatchToProps = { | ||||
|   listPendingTasksAsync, | ||||
|   deletePendingTaskAsync, | ||||
|   batchDeletePendingTasksAsync, | ||||
|   deleteAllPendingTasksAsync, | ||||
|   archivePendingTaskAsync, | ||||
|   batchArchivePendingTasksAsync, | ||||
|   archiveAllPendingTasksAsync, | ||||
|   listTasks: listPendingTasksAsync, | ||||
|   deleteTask: deletePendingTaskAsync, | ||||
|   batchDeleteTasks: batchDeletePendingTasksAsync, | ||||
|   deleteAllTasks: deleteAllPendingTasksAsync, | ||||
|   archiveTask: archivePendingTaskAsync, | ||||
|   batchArchiveTasks: batchArchivePendingTasksAsync, | ||||
|   archiveAllTasks: archiveAllPendingTasksAsync, | ||||
|   taskRowsPerPageChange, | ||||
| }; | ||||
|  | ||||
| @@ -90,268 +59,14 @@ interface Props { | ||||
|   totalTaskCount: number; // total number of pending tasks | ||||
| } | ||||
|  | ||||
| function PendingTasksTable(props: Props & ReduxProps) { | ||||
|   const { pollInterval, listPendingTasksAsync, queue, pageSize } = props; | ||||
|   const classes = useStyles(); | ||||
|   const [page, setPage] = useState(0); | ||||
|   const [selectedIds, setSelectedIds] = useState<string[]>([]); | ||||
|   const [activeTaskId, setActiveTaskId] = useState<string>(""); | ||||
|   const handlePageChange = ( | ||||
|     event: React.MouseEvent<HTMLButtonElement> | null, | ||||
|     newPage: number | ||||
|   ) => { | ||||
|     setPage(newPage); | ||||
|   }; | ||||
|  | ||||
|   const handleRowsPerPageChange = ( | ||||
|     event: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement> | ||||
|   ) => { | ||||
|     props.taskRowsPerPageChange(parseInt(event.target.value, 10)); | ||||
|     setPage(0); | ||||
|   }; | ||||
|  | ||||
|   const handleSelectAllClick = (event: React.ChangeEvent<HTMLInputElement>) => { | ||||
|     if (event.target.checked) { | ||||
|       const newSelected = props.tasks.map((t) => t.id); | ||||
|       setSelectedIds(newSelected); | ||||
|     } else { | ||||
|       setSelectedIds([]); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const handleDeleteAllClick = () => { | ||||
|     props.deleteAllPendingTasksAsync(queue); | ||||
|   }; | ||||
|  | ||||
|   const handleArchiveAllClick = () => { | ||||
|     props.archiveAllPendingTasksAsync(queue); | ||||
|   }; | ||||
|  | ||||
|   const handleBatchDeleteClick = () => { | ||||
|     props | ||||
|       .batchDeletePendingTasksAsync(queue, selectedIds) | ||||
|       .then(() => setSelectedIds([])); | ||||
|   }; | ||||
|  | ||||
|   const handleBatchArchiveClick = () => { | ||||
|     props | ||||
|       .batchArchivePendingTasksAsync(queue, selectedIds) | ||||
|       .then(() => setSelectedIds([])); | ||||
|   }; | ||||
|  | ||||
|   const fetchData = useCallback(() => { | ||||
|     const pageOpts = { page: page + 1, size: pageSize }; | ||||
|     listPendingTasksAsync(queue, pageOpts); | ||||
|   }, [page, pageSize, queue, listPendingTasksAsync]); | ||||
|  | ||||
|   usePolling(fetchData, pollInterval); | ||||
|  | ||||
|   if (props.error.length > 0) { | ||||
|     return ( | ||||
|       <Alert severity="error" className={classes.alert}> | ||||
|         <AlertTitle>Error</AlertTitle> | ||||
|         {props.error} | ||||
|       </Alert> | ||||
|     ); | ||||
|   } | ||||
|   if (props.tasks.length === 0) { | ||||
|     return ( | ||||
|       <Alert severity="info" className={classes.alert}> | ||||
|         <AlertTitle>Info</AlertTitle> | ||||
|         No pending tasks at this time. | ||||
|       </Alert> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   const columns: TableColumn[] = [ | ||||
|     { key: "id", label: "ID", align: "left" }, | ||||
|     { key: "type", label: "Type", align: "left" }, | ||||
|     { key: "paylod", label: "Payload", align: "left" }, | ||||
|     { key: "retried", label: "Retried", align: "right" }, | ||||
|     { key: "max_retry", label: "Max Retry", align: "right" }, | ||||
|     { key: "actions", label: "Actions", align: "center" }, | ||||
|   ]; | ||||
|  | ||||
|   const rowCount = props.tasks.length; | ||||
|   const numSelected = selectedIds.length; | ||||
|   return ( | ||||
|     <div> | ||||
|       {!window.READ_ONLY && ( | ||||
|         <TableActions | ||||
|           showIconButtons={numSelected > 0} | ||||
|           iconButtonActions={[ | ||||
|             { | ||||
|               tooltip: "Delete", | ||||
|               icon: <DeleteIcon />, | ||||
|               onClick: handleBatchDeleteClick, | ||||
|               disabled: props.batchActionPending, | ||||
|             }, | ||||
|             { | ||||
|               tooltip: "Archive", | ||||
|               icon: <ArchiveIcon />, | ||||
|               onClick: handleBatchArchiveClick, | ||||
|               disabled: props.batchActionPending, | ||||
|             }, | ||||
|           ]} | ||||
|           menuItemActions={[ | ||||
|             { | ||||
|               label: "Delete All", | ||||
|               onClick: handleDeleteAllClick, | ||||
|               disabled: props.allActionPending, | ||||
|             }, | ||||
|             { | ||||
|               label: "Archive All", | ||||
|               onClick: handleArchiveAllClick, | ||||
|               disabled: props.allActionPending, | ||||
|             }, | ||||
|           ]} | ||||
|         /> | ||||
|       )} | ||||
|       <TableContainer component={Paper}> | ||||
|         <Table | ||||
|           stickyHeader={true} | ||||
|           className={classes.table} | ||||
|           aria-label="pending tasks table" | ||||
|           size="small" | ||||
|         > | ||||
|           <TableHead> | ||||
|             <TableRow> | ||||
|               {!window.READ_ONLY && ( | ||||
|                 <TableCell | ||||
|                   padding="checkbox" | ||||
|                   classes={{ stickyHeader: classes.stickyHeaderCell }} | ||||
|                 > | ||||
|                   <IconButton> | ||||
|                     <Checkbox | ||||
|                       indeterminate={numSelected > 0 && numSelected < rowCount} | ||||
|                       checked={rowCount > 0 && numSelected === rowCount} | ||||
|                       onChange={handleSelectAllClick} | ||||
|                       inputProps={{ | ||||
|                         "aria-label": "select all tasks shown in the table", | ||||
|                       }} | ||||
|                     /> | ||||
|                   </IconButton> | ||||
|                 </TableCell> | ||||
|               )} | ||||
|               {columns | ||||
|                 .filter((col) => { | ||||
|                   // Filter out actions column in readonly mode. | ||||
|                   return !window.READ_ONLY || col.key !== "actions"; | ||||
|                 }) | ||||
|                 .map((col) => ( | ||||
|                   <TableCell | ||||
|                     key={col.key} | ||||
|                     align={col.align} | ||||
|                     classes={{ | ||||
|                       stickyHeader: classes.stickyHeaderCell, | ||||
|                     }} | ||||
|                   > | ||||
|                     {col.label} | ||||
|                   </TableCell> | ||||
|                 ))} | ||||
|             </TableRow> | ||||
|           </TableHead> | ||||
|           <TableBody> | ||||
|             {props.tasks.map((task) => ( | ||||
|               <Row | ||||
|                 key={task.id} | ||||
|                 task={task} | ||||
|                 isSelected={selectedIds.includes(task.id)} | ||||
|                 onSelectChange={(checked: boolean) => { | ||||
|                   if (checked) { | ||||
|                     setSelectedIds(selectedIds.concat(task.id)); | ||||
|                   } else { | ||||
|                     setSelectedIds(selectedIds.filter((id) => id !== task.id)); | ||||
|                   } | ||||
|                 }} | ||||
|                 allActionPending={props.allActionPending} | ||||
|                 onDeleteClick={() => | ||||
|                   props.deletePendingTaskAsync(queue, task.id) | ||||
|                 } | ||||
|                 onArchiveClick={() => { | ||||
|                   props.archivePendingTaskAsync(queue, task.id); | ||||
|                 }} | ||||
|                 onActionCellEnter={() => setActiveTaskId(task.id)} | ||||
|                 onActionCellLeave={() => setActiveTaskId("")} | ||||
|                 showActions={activeTaskId === task.id} | ||||
|               /> | ||||
|             ))} | ||||
|           </TableBody> | ||||
|           <TableFooter> | ||||
|             <TableRow> | ||||
|               <TablePagination | ||||
|                 rowsPerPageOptions={rowsPerPageOptions} | ||||
|                 colSpan={columns.length + 1} | ||||
|                 count={props.totalTaskCount} | ||||
|                 rowsPerPage={pageSize} | ||||
|                 page={page} | ||||
|                 SelectProps={{ | ||||
|                   inputProps: { "aria-label": "rows per page" }, | ||||
|                   native: true, | ||||
|                 }} | ||||
|                 onPageChange={handlePageChange} | ||||
|                 onRowsPerPageChange={handleRowsPerPageChange} | ||||
|                 ActionsComponent={TablePaginationActions} | ||||
|                 className={classes.pagination} | ||||
|               /> | ||||
|             </TableRow> | ||||
|           </TableFooter> | ||||
|         </Table> | ||||
|       </TableContainer> | ||||
|     </div> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| const useRowStyles = makeStyles((theme) => ({ | ||||
|   root: { | ||||
|     cursor: "pointer", | ||||
|     "& #copy-button": { | ||||
|       display: "none", | ||||
|     }, | ||||
|     "&:hover": { | ||||
|       boxShadow: theme.shadows[2], | ||||
|       "& #copy-button": { | ||||
|         display: "inline-block", | ||||
|       }, | ||||
|     }, | ||||
|     "&:hover $copyButton": { | ||||
|       display: "inline-block", | ||||
|     }, | ||||
|     "&:hover .MuiTableCell-root": { | ||||
|       borderBottomColor: theme.palette.background.paper, | ||||
|     }, | ||||
|   }, | ||||
|  | ||||
|   actionCell: { | ||||
|     width: "96px", | ||||
|   }, | ||||
|   actionButton: { | ||||
|     marginLeft: 3, | ||||
|     marginRight: 3, | ||||
|   }, | ||||
|   idCell: { | ||||
|     width: "200px", | ||||
|   }, | ||||
|   copyButton: { | ||||
|     display: "none", | ||||
|   }, | ||||
|   IdGroup: { | ||||
|     display: "flex", | ||||
|     alignItems: "center", | ||||
|   }, | ||||
| })); | ||||
|  | ||||
| interface RowProps { | ||||
|   task: TaskInfoExtended; | ||||
|   isSelected: boolean; | ||||
|   onSelectChange: (checked: boolean) => void; | ||||
|   onDeleteClick: () => void; | ||||
|   onArchiveClick: () => void; | ||||
|   allActionPending: boolean; | ||||
|   showActions: boolean; | ||||
|   onActionCellEnter: () => void; | ||||
|   onActionCellLeave: () => void; | ||||
| } | ||||
| const columns: TableColumn[] = [ | ||||
|   { key: "id", label: "ID", align: "left" }, | ||||
|   { key: "type", label: "Type", align: "left" }, | ||||
|   { key: "paylod", label: "Payload", align: "left" }, | ||||
|   { key: "retried", label: "Retried", align: "right" }, | ||||
|   { key: "max_retry", label: "Max Retry", align: "right" }, | ||||
|   { key: "actions", label: "Actions", align: "center" }, | ||||
| ]; | ||||
|  | ||||
| function Row(props: RowProps) { | ||||
|   const { task } = props; | ||||
| @@ -446,4 +161,15 @@ function Row(props: RowProps) { | ||||
|   ); | ||||
| } | ||||
|  | ||||
| function PendingTasksTable(props: Props & ReduxProps) { | ||||
|   return ( | ||||
|     <TasksTable | ||||
|       taskState="pending" | ||||
|       columns={columns} | ||||
|       renderRow={(rowProps: RowProps) => <Row {...rowProps} />} | ||||
|       {...props} | ||||
|     /> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| export default connector(PendingTasksTable); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user