mirror of
				https://github.com/hibiken/asynqmon.git
				synced 2025-10-25 07:46:12 +08:00 
			
		
		
		
	(ui): Add renderRow prop to TasksTable component
This commit is contained in:
		| @@ -1,6 +1,18 @@ | |||||||
| import React from "react"; | import React from "react"; | ||||||
| import { connect, ConnectedProps } from "react-redux"; | import { connect, ConnectedProps } from "react-redux"; | ||||||
| import TasksTable from "./TasksTable"; | import { useHistory } from "react-router-dom"; | ||||||
|  | import TableRow from "@material-ui/core/TableRow"; | ||||||
|  | import TableCell from "@material-ui/core/TableCell"; | ||||||
|  | import Checkbox from "@material-ui/core/Checkbox"; | ||||||
|  | import IconButton from "@material-ui/core/IconButton"; | ||||||
|  | import Tooltip from "@material-ui/core/Tooltip"; | ||||||
|  | import FileCopyOutlinedIcon from "@material-ui/icons/FileCopyOutlined"; | ||||||
|  | import DeleteIcon from "@material-ui/icons/Delete"; | ||||||
|  | import ArchiveIcon from "@material-ui/icons/Archive"; | ||||||
|  | import MoreHorizIcon from "@material-ui/icons/MoreHoriz"; | ||||||
|  | import PlayArrowIcon from "@material-ui/icons/PlayArrow"; | ||||||
|  | import SyntaxHighlighter from "./SyntaxHighlighter"; | ||||||
|  | import TasksTable, { RowProps, useRowStyles } from "./TasksTable"; | ||||||
| import { | import { | ||||||
|   batchDeleteScheduledTasksAsync, |   batchDeleteScheduledTasksAsync, | ||||||
|   batchRunScheduledTasksAsync, |   batchRunScheduledTasksAsync, | ||||||
| @@ -16,6 +28,8 @@ import { | |||||||
| import { taskRowsPerPageChange } from "../actions/settingsActions"; | import { taskRowsPerPageChange } from "../actions/settingsActions"; | ||||||
| import { AppState } from "../store"; | import { AppState } from "../store"; | ||||||
| import { TableColumn } from "../types/table"; | import { TableColumn } from "../types/table"; | ||||||
|  | import { durationBefore, prettifyPayload, uuidPrefix } from "../utils"; | ||||||
|  | import { taskDetailsPath } from "../paths"; | ||||||
|  |  | ||||||
| function mapStateToProps(state: AppState) { | function mapStateToProps(state: AppState) { | ||||||
|   return { |   return { | ||||||
| @@ -61,7 +75,116 @@ const columns: TableColumn[] = [ | |||||||
| ]; | ]; | ||||||
|  |  | ||||||
| function ScheduledTasksTable(props: Props & ReduxProps) { | function ScheduledTasksTable(props: Props & ReduxProps) { | ||||||
|   return <TasksTable taskState="scheduled" columns={columns} {...props} />; |   return ( | ||||||
|  |     <TasksTable | ||||||
|  |       taskState="scheduled" | ||||||
|  |       columns={columns} | ||||||
|  |       renderRow={(rowProps: RowProps) => <Row {...rowProps} />} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function Row(props: RowProps) { | ||||||
|  |   const { task } = props; | ||||||
|  |   const classes = useRowStyles(); | ||||||
|  |   const history = useHistory(); | ||||||
|  |   return ( | ||||||
|  |     <TableRow | ||||||
|  |       key={task.id} | ||||||
|  |       className={classes.root} | ||||||
|  |       selected={props.isSelected} | ||||||
|  |       onClick={() => history.push(taskDetailsPath(task.queue, task.id))} | ||||||
|  |     > | ||||||
|  |       {!window.READ_ONLY && ( | ||||||
|  |         <TableCell padding="checkbox" onClick={(e) => e.stopPropagation()}> | ||||||
|  |           <IconButton> | ||||||
|  |             <Checkbox | ||||||
|  |               onChange={(event: React.ChangeEvent<HTMLInputElement>) => | ||||||
|  |                 props.onSelectChange(event.target.checked) | ||||||
|  |               } | ||||||
|  |               checked={props.isSelected} | ||||||
|  |             /> | ||||||
|  |           </IconButton> | ||||||
|  |         </TableCell> | ||||||
|  |       )} | ||||||
|  |       <TableCell component="th" scope="row" className={classes.idCell}> | ||||||
|  |         <div className={classes.IdGroup}> | ||||||
|  |           {uuidPrefix(task.id)} | ||||||
|  |           <Tooltip title="Copy full ID to clipboard"> | ||||||
|  |             <IconButton | ||||||
|  |               onClick={(e) => { | ||||||
|  |                 e.stopPropagation(); | ||||||
|  |                 navigator.clipboard.writeText(task.id); | ||||||
|  |               }} | ||||||
|  |               size="small" | ||||||
|  |               className={classes.copyButton} | ||||||
|  |             > | ||||||
|  |               <FileCopyOutlinedIcon fontSize="small" /> | ||||||
|  |             </IconButton> | ||||||
|  |           </Tooltip> | ||||||
|  |         </div> | ||||||
|  |       </TableCell> | ||||||
|  |       <TableCell>{task.type}</TableCell> | ||||||
|  |       <TableCell> | ||||||
|  |         <SyntaxHighlighter | ||||||
|  |           language="json" | ||||||
|  |           customStyle={{ margin: 0, maxWidth: 400 }} | ||||||
|  |         > | ||||||
|  |           {prettifyPayload(task.payload)} | ||||||
|  |         </SyntaxHighlighter> | ||||||
|  |       </TableCell> | ||||||
|  |       <TableCell>{durationBefore(task.next_process_at)}</TableCell> | ||||||
|  |       {!window.READ_ONLY && ( | ||||||
|  |         <TableCell | ||||||
|  |           align="center" | ||||||
|  |           className={classes.actionCell} | ||||||
|  |           onMouseEnter={props.onActionCellEnter} | ||||||
|  |           onMouseLeave={props.onActionCellLeave} | ||||||
|  |           onClick={(e) => e.stopPropagation()} | ||||||
|  |         > | ||||||
|  |           {props.showActions ? ( | ||||||
|  |             <React.Fragment> | ||||||
|  |               <Tooltip title="Delete"> | ||||||
|  |                 <IconButton | ||||||
|  |                   onClick={props.onDeleteClick} | ||||||
|  |                   disabled={task.requestPending || props.allActionPending} | ||||||
|  |                   size="small" | ||||||
|  |                   className={classes.actionButton} | ||||||
|  |                 > | ||||||
|  |                   <DeleteIcon fontSize="small" /> | ||||||
|  |                 </IconButton> | ||||||
|  |               </Tooltip> | ||||||
|  |               <Tooltip title="Archive"> | ||||||
|  |                 <IconButton | ||||||
|  |                   onClick={props.onArchiveClick} | ||||||
|  |                   disabled={task.requestPending || props.allActionPending} | ||||||
|  |                   size="small" | ||||||
|  |                   className={classes.actionButton} | ||||||
|  |                 > | ||||||
|  |                   <ArchiveIcon fontSize="small" /> | ||||||
|  |                 </IconButton> | ||||||
|  |               </Tooltip> | ||||||
|  |               <Tooltip title="Run"> | ||||||
|  |                 <IconButton | ||||||
|  |                   onClick={props.onRunClick} | ||||||
|  |                   disabled={task.requestPending || props.allActionPending} | ||||||
|  |                   size="small" | ||||||
|  |                   className={classes.actionButton} | ||||||
|  |                 > | ||||||
|  |                   <PlayArrowIcon fontSize="small" /> | ||||||
|  |                 </IconButton> | ||||||
|  |               </Tooltip> | ||||||
|  |             </React.Fragment> | ||||||
|  |           ) : ( | ||||||
|  |             <IconButton size="small" onClick={props.onActionCellEnter}> | ||||||
|  |               <MoreHorizIcon fontSize="small" /> | ||||||
|  |             </IconButton> | ||||||
|  |           )} | ||||||
|  |         </TableCell> | ||||||
|  |       )} | ||||||
|  |     </TableRow> | ||||||
|  |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| export default connector(ScheduledTasksTable); | export default connector(ScheduledTasksTable); | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| import React, { useState, useCallback } from "react"; | import React, { useState, useCallback } from "react"; | ||||||
| import { useHistory } from "react-router-dom"; |  | ||||||
| import { makeStyles } from "@material-ui/core/styles"; | import { makeStyles } from "@material-ui/core/styles"; | ||||||
| import Table from "@material-ui/core/Table"; | import Table from "@material-ui/core/Table"; | ||||||
| import TableBody from "@material-ui/core/TableBody"; | import TableBody from "@material-ui/core/TableBody"; | ||||||
| @@ -10,26 +9,20 @@ import TableRow from "@material-ui/core/TableRow"; | |||||||
| import TableFooter from "@material-ui/core/TableFooter"; | import TableFooter from "@material-ui/core/TableFooter"; | ||||||
| import TablePagination from "@material-ui/core/TablePagination"; | import TablePagination from "@material-ui/core/TablePagination"; | ||||||
| import Paper from "@material-ui/core/Paper"; | import Paper from "@material-ui/core/Paper"; | ||||||
| import Tooltip from "@material-ui/core/Tooltip"; |  | ||||||
| import Checkbox from "@material-ui/core/Checkbox"; | import Checkbox from "@material-ui/core/Checkbox"; | ||||||
| import IconButton from "@material-ui/core/IconButton"; | import IconButton from "@material-ui/core/IconButton"; | ||||||
| import PlayArrowIcon from "@material-ui/icons/PlayArrow"; | import PlayArrowIcon from "@material-ui/icons/PlayArrow"; | ||||||
| import DeleteIcon from "@material-ui/icons/Delete"; | import DeleteIcon from "@material-ui/icons/Delete"; | ||||||
| import ArchiveIcon from "@material-ui/icons/Archive"; | import ArchiveIcon from "@material-ui/icons/Archive"; | ||||||
| import MoreHorizIcon from "@material-ui/icons/MoreHoriz"; |  | ||||||
| import Alert from "@material-ui/lab/Alert"; | import Alert from "@material-ui/lab/Alert"; | ||||||
| import AlertTitle from "@material-ui/lab/AlertTitle"; | import AlertTitle from "@material-ui/lab/AlertTitle"; | ||||||
| import SyntaxHighlighter from "./SyntaxHighlighter"; |  | ||||||
| import TablePaginationActions, { | import TablePaginationActions, { | ||||||
|   rowsPerPageOptions, |   rowsPerPageOptions, | ||||||
| } from "./TablePaginationActions"; | } from "./TablePaginationActions"; | ||||||
| import TableActions from "./TableActions"; | import TableActions from "./TableActions"; | ||||||
| import { durationBefore, uuidPrefix, prettifyPayload } from "../utils"; |  | ||||||
| import { usePolling } from "../hooks"; | import { usePolling } from "../hooks"; | ||||||
| import { TaskInfoExtended } from "../reducers/tasksReducer"; | import { TaskInfoExtended } from "../reducers/tasksReducer"; | ||||||
| import { TableColumn } from "../types/table"; | import { TableColumn } from "../types/table"; | ||||||
| import { taskDetailsPath } from "../paths"; |  | ||||||
| import FileCopyOutlinedIcon from "@material-ui/icons/FileCopyOutlined"; |  | ||||||
| import { PaginationOptions } from "../api"; | import { PaginationOptions } from "../api"; | ||||||
| import { TaskState } from "../types/taskState"; | import { TaskState } from "../types/taskState"; | ||||||
|  |  | ||||||
| @@ -74,6 +67,8 @@ interface Props { | |||||||
|   runTask?: (qname: string, taskId: string) => Promise<void>; |   runTask?: (qname: string, taskId: string) => Promise<void>; | ||||||
|   archiveTask?: (qname: string, taskId: string) => Promise<void>; |   archiveTask?: (qname: string, taskId: string) => Promise<void>; | ||||||
|   taskRowsPerPageChange: (n: number) => void; |   taskRowsPerPageChange: (n: number) => void; | ||||||
|  |  | ||||||
|  |   renderRow: (rowProps: RowProps) => JSX.Element; | ||||||
| } | } | ||||||
|  |  | ||||||
| export default function TasksTable(props: Props) { | export default function TasksTable(props: Props) { | ||||||
| @@ -250,39 +245,33 @@ export default function TasksTable(props: Props) { | |||||||
|             </TableRow> |             </TableRow> | ||||||
|           </TableHead> |           </TableHead> | ||||||
|           <TableBody> |           <TableBody> | ||||||
|             {props.tasks.map((task) => ( |             {props.tasks.map((task) => { | ||||||
|               <Row |               return props.renderRow({ | ||||||
|                 key={task.id} |                 key: task.id, | ||||||
|                 task={task} |                 task: task, | ||||||
|                 allActionPending={props.allActionPending} |                 allActionPending: props.allActionPending, | ||||||
|                 isSelected={selectedIds.includes(task.id)} |                 isSelected: selectedIds.includes(task.id), | ||||||
|                 onSelectChange={(checked: boolean) => { |                 onSelectChange: (checked: boolean) => { | ||||||
|                   if (checked) { |                   if (checked) { | ||||||
|                     setSelectedIds(selectedIds.concat(task.id)); |                     setSelectedIds(selectedIds.concat(task.id)); | ||||||
|                   } else { |                   } else { | ||||||
|                     setSelectedIds(selectedIds.filter((id) => id !== task.id)); |                     setSelectedIds(selectedIds.filter((id) => id !== task.id)); | ||||||
|                   } |                   } | ||||||
|                 }} |                 }, | ||||||
|                 onRunClick={ |                 onRunClick: props.runTask | ||||||
|                   props.runTask |                   ? createTaskAction(props.runTask, task.id) | ||||||
|                     ? createTaskAction(props.runTask, task.id) |                   : undefined, | ||||||
|                     : undefined |                 onDeleteClick: props.deleteTask | ||||||
|                 } |                   ? createTaskAction(props.deleteTask, task.id) | ||||||
|                 onDeleteClick={ |                   : undefined, | ||||||
|                   props.deleteTask |                 onArchiveClick: props.archiveTask | ||||||
|                     ? createTaskAction(props.deleteTask, task.id) |                   ? createTaskAction(props.archiveTask, task.id) | ||||||
|                     : undefined |                   : undefined, | ||||||
|                 } |                 onActionCellEnter: () => setActiveTaskId(task.id), | ||||||
|                 onArchiveClick={ |                 onActionCellLeave: () => setActiveTaskId(""), | ||||||
|                   props.archiveTask |                 showActions: activeTaskId === task.id, | ||||||
|                     ? createTaskAction(props.archiveTask, task.id) |               }); | ||||||
|                     : undefined |             })} | ||||||
|                 } |  | ||||||
|                 onActionCellEnter={() => setActiveTaskId(task.id)} |  | ||||||
|                 onActionCellLeave={() => setActiveTaskId("")} |  | ||||||
|                 showActions={activeTaskId === task.id} |  | ||||||
|               /> |  | ||||||
|             ))} |  | ||||||
|           </TableBody> |           </TableBody> | ||||||
|           <TableFooter> |           <TableFooter> | ||||||
|             <TableRow> |             <TableRow> | ||||||
| @@ -309,7 +298,7 @@ export default function TasksTable(props: Props) { | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| const useRowStyles = makeStyles((theme) => ({ | export const useRowStyles = makeStyles((theme) => ({ | ||||||
|   root: { |   root: { | ||||||
|     cursor: "pointer", |     cursor: "pointer", | ||||||
|     "& #copy-button": { |     "& #copy-button": { | ||||||
| @@ -347,7 +336,8 @@ const useRowStyles = makeStyles((theme) => ({ | |||||||
|   }, |   }, | ||||||
| })); | })); | ||||||
|  |  | ||||||
| interface RowProps { | export interface RowProps { | ||||||
|  |   key: string; | ||||||
|   task: TaskInfoExtended; |   task: TaskInfoExtended; | ||||||
|   isSelected: boolean; |   isSelected: boolean; | ||||||
|   onSelectChange: (checked: boolean) => void; |   onSelectChange: (checked: boolean) => void; | ||||||
| @@ -359,105 +349,3 @@ interface RowProps { | |||||||
|   onActionCellEnter: () => void; |   onActionCellEnter: () => void; | ||||||
|   onActionCellLeave: () => void; |   onActionCellLeave: () => void; | ||||||
| } | } | ||||||
|  |  | ||||||
| function Row(props: RowProps) { |  | ||||||
|   const { task } = props; |  | ||||||
|   const classes = useRowStyles(); |  | ||||||
|   const history = useHistory(); |  | ||||||
|   return ( |  | ||||||
|     <TableRow |  | ||||||
|       key={task.id} |  | ||||||
|       className={classes.root} |  | ||||||
|       selected={props.isSelected} |  | ||||||
|       onClick={() => history.push(taskDetailsPath(task.queue, task.id))} |  | ||||||
|     > |  | ||||||
|       {!window.READ_ONLY && ( |  | ||||||
|         <TableCell padding="checkbox" onClick={(e) => e.stopPropagation()}> |  | ||||||
|           <IconButton> |  | ||||||
|             <Checkbox |  | ||||||
|               onChange={(event: React.ChangeEvent<HTMLInputElement>) => |  | ||||||
|                 props.onSelectChange(event.target.checked) |  | ||||||
|               } |  | ||||||
|               checked={props.isSelected} |  | ||||||
|             /> |  | ||||||
|           </IconButton> |  | ||||||
|         </TableCell> |  | ||||||
|       )} |  | ||||||
|       <TableCell component="th" scope="row" className={classes.idCell}> |  | ||||||
|         <div className={classes.IdGroup}> |  | ||||||
|           {uuidPrefix(task.id)} |  | ||||||
|           <Tooltip title="Copy full ID to clipboard"> |  | ||||||
|             <IconButton |  | ||||||
|               onClick={(e) => { |  | ||||||
|                 e.stopPropagation(); |  | ||||||
|                 navigator.clipboard.writeText(task.id); |  | ||||||
|               }} |  | ||||||
|               size="small" |  | ||||||
|               className={classes.copyButton} |  | ||||||
|             > |  | ||||||
|               <FileCopyOutlinedIcon fontSize="small" /> |  | ||||||
|             </IconButton> |  | ||||||
|           </Tooltip> |  | ||||||
|         </div> |  | ||||||
|       </TableCell> |  | ||||||
|       <TableCell>{task.type}</TableCell> |  | ||||||
|       <TableCell> |  | ||||||
|         <SyntaxHighlighter |  | ||||||
|           language="json" |  | ||||||
|           customStyle={{ margin: 0, maxWidth: 400 }} |  | ||||||
|         > |  | ||||||
|           {prettifyPayload(task.payload)} |  | ||||||
|         </SyntaxHighlighter> |  | ||||||
|       </TableCell> |  | ||||||
|       <TableCell>{durationBefore(task.next_process_at)}</TableCell> |  | ||||||
|       {!window.READ_ONLY && ( |  | ||||||
|         <TableCell |  | ||||||
|           align="center" |  | ||||||
|           className={classes.actionCell} |  | ||||||
|           onMouseEnter={props.onActionCellEnter} |  | ||||||
|           onMouseLeave={props.onActionCellLeave} |  | ||||||
|           onClick={(e) => e.stopPropagation()} |  | ||||||
|         > |  | ||||||
|           {props.showActions ? ( |  | ||||||
|             <React.Fragment> |  | ||||||
|               <Tooltip title="Delete"> |  | ||||||
|                 <IconButton |  | ||||||
|                   onClick={props.onDeleteClick} |  | ||||||
|                   disabled={task.requestPending || props.allActionPending} |  | ||||||
|                   size="small" |  | ||||||
|                   className={classes.actionButton} |  | ||||||
|                 > |  | ||||||
|                   <DeleteIcon fontSize="small" /> |  | ||||||
|                 </IconButton> |  | ||||||
|               </Tooltip> |  | ||||||
|               <Tooltip title="Archive"> |  | ||||||
|                 <IconButton |  | ||||||
|                   onClick={props.onArchiveClick} |  | ||||||
|                   disabled={task.requestPending || props.allActionPending} |  | ||||||
|                   size="small" |  | ||||||
|                   className={classes.actionButton} |  | ||||||
|                 > |  | ||||||
|                   <ArchiveIcon fontSize="small" /> |  | ||||||
|                 </IconButton> |  | ||||||
|               </Tooltip> |  | ||||||
|               <Tooltip title="Run"> |  | ||||||
|                 <IconButton |  | ||||||
|                   onClick={props.onRunClick} |  | ||||||
|                   disabled={task.requestPending || props.allActionPending} |  | ||||||
|                   size="small" |  | ||||||
|                   className={classes.actionButton} |  | ||||||
|                 > |  | ||||||
|                   <PlayArrowIcon fontSize="small" /> |  | ||||||
|                 </IconButton> |  | ||||||
|               </Tooltip> |  | ||||||
|             </React.Fragment> |  | ||||||
|           ) : ( |  | ||||||
|             <IconButton size="small" onClick={props.onActionCellEnter}> |  | ||||||
|               <MoreHorizIcon fontSize="small" /> |  | ||||||
|             </IconButton> |  | ||||||
|           )} |  | ||||||
|         </TableCell> |  | ||||||
|       )} |  | ||||||
|     </TableRow> |  | ||||||
|   ); |  | ||||||
| } |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user