[5d6f37a] | 1 | import { format } from 'date-fns';
|
---|
| 2 | // @mui
|
---|
| 3 | import Link from '@mui/material/Link';
|
---|
| 4 | import Button from '@mui/material/Button';
|
---|
| 5 | import Avatar from '@mui/material/Avatar';
|
---|
| 6 | import Divider from '@mui/material/Divider';
|
---|
| 7 | import MenuItem from '@mui/material/MenuItem';
|
---|
| 8 | import TableRow from '@mui/material/TableRow';
|
---|
| 9 | import TableCell from '@mui/material/TableCell';
|
---|
| 10 | import IconButton from '@mui/material/IconButton';
|
---|
| 11 | import Typography from '@mui/material/Typography';
|
---|
| 12 | import ListItemText from '@mui/material/ListItemText';
|
---|
| 13 | // hooks
|
---|
| 14 | import { useBoolean } from 'src/hooks/use-boolean';
|
---|
| 15 | // utils
|
---|
| 16 | import { fNumber } from 'src/utils/format-number';
|
---|
| 17 | // types
|
---|
| 18 | import { Invoice } from 'mvpmasters-shared';
|
---|
| 19 | // components
|
---|
| 20 | import Label from 'src/components/label';
|
---|
| 21 | import Iconify from 'src/components/iconify';
|
---|
| 22 | import { ConfirmDialog } from 'src/components/custom-dialog';
|
---|
| 23 | import CustomPopover, { usePopover } from 'src/components/custom-popover';
|
---|
| 24 |
|
---|
| 25 | // ----------------------------------------------------------------------
|
---|
| 26 |
|
---|
| 27 | type Props = {
|
---|
| 28 | row: Invoice;
|
---|
| 29 | selected: boolean;
|
---|
| 30 | onSelectRow: VoidFunction;
|
---|
| 31 | onViewRow: VoidFunction;
|
---|
| 32 | onEditRow: VoidFunction;
|
---|
| 33 | onCopyRow: VoidFunction;
|
---|
| 34 | onDeleteRow: () => Promise<void>;
|
---|
| 35 | onSendRow: VoidFunction;
|
---|
| 36 | onToggleCompose: VoidFunction;
|
---|
| 37 | };
|
---|
| 38 |
|
---|
| 39 | export default function InvoiceTableRow({
|
---|
| 40 | row,
|
---|
| 41 | selected,
|
---|
| 42 | onSelectRow,
|
---|
| 43 | onViewRow,
|
---|
| 44 | onEditRow,
|
---|
| 45 | onCopyRow,
|
---|
| 46 | onDeleteRow,
|
---|
| 47 | onSendRow,
|
---|
| 48 | onToggleCompose,
|
---|
| 49 | }: Props) {
|
---|
| 50 | const {
|
---|
| 51 | sent,
|
---|
| 52 | invoiceNumber,
|
---|
| 53 | createDate,
|
---|
| 54 | dueDate,
|
---|
| 55 | status,
|
---|
| 56 | invoiceTo,
|
---|
| 57 | currency,
|
---|
| 58 | totalAmount,
|
---|
| 59 | month,
|
---|
| 60 | } = row;
|
---|
| 61 |
|
---|
| 62 | const confirmSend = useBoolean();
|
---|
| 63 | const confirmDelete = useBoolean();
|
---|
| 64 |
|
---|
| 65 | const popover = usePopover();
|
---|
| 66 |
|
---|
| 67 | return (
|
---|
| 68 | <>
|
---|
| 69 | <TableRow hover selected={selected}>
|
---|
| 70 | {/* <TableCell padding="checkbox">
|
---|
| 71 | <Checkbox checked={selected} onClick={onSelectRow} />
|
---|
| 72 | </TableCell> */}
|
---|
| 73 |
|
---|
| 74 | <TableCell sx={{ display: 'flex', alignItems: 'center' }}>
|
---|
| 75 | <Avatar alt={invoiceTo.name} src={invoiceTo.logoUrl} sx={{ mr: 2 }}>
|
---|
| 76 | {invoiceTo.name.charAt(0).toUpperCase()}
|
---|
| 77 | </Avatar>
|
---|
| 78 |
|
---|
| 79 | <ListItemText
|
---|
| 80 | disableTypography
|
---|
| 81 | primary={
|
---|
| 82 | <Typography variant="body2" noWrap>
|
---|
| 83 | {invoiceTo.name}
|
---|
| 84 | </Typography>
|
---|
| 85 | }
|
---|
| 86 | secondary={
|
---|
| 87 | <Link
|
---|
| 88 | noWrap
|
---|
| 89 | variant="body2"
|
---|
| 90 | onClick={onViewRow}
|
---|
| 91 | sx={{ color: 'text.disabled', cursor: 'pointer' }}
|
---|
| 92 | >
|
---|
| 93 | {invoiceNumber}
|
---|
| 94 | </Link>
|
---|
| 95 | }
|
---|
| 96 | />
|
---|
| 97 | </TableCell>
|
---|
| 98 |
|
---|
| 99 | <TableCell>{fNumber(totalAmount)}</TableCell>
|
---|
| 100 |
|
---|
| 101 | <TableCell>{currency}</TableCell>
|
---|
| 102 |
|
---|
| 103 | <TableCell>{month}</TableCell>
|
---|
| 104 |
|
---|
| 105 | <TableCell>
|
---|
| 106 | <ListItemText
|
---|
| 107 | primary={format(createDate.toDate(), 'dd MMM yyyy')}
|
---|
| 108 | secondary={format(createDate.toDate(), 'p')}
|
---|
| 109 | primaryTypographyProps={{ typography: 'body2', noWrap: true }}
|
---|
| 110 | secondaryTypographyProps={{
|
---|
| 111 | mt: 0.5,
|
---|
| 112 | component: 'span',
|
---|
| 113 | typography: 'caption',
|
---|
| 114 | }}
|
---|
| 115 | />
|
---|
| 116 | </TableCell>
|
---|
| 117 |
|
---|
| 118 | <TableCell>
|
---|
| 119 | <ListItemText
|
---|
| 120 | primary={format(dueDate.toDate(), 'dd MMM yyyy')}
|
---|
| 121 | secondary={format(dueDate.toDate(), 'p')}
|
---|
| 122 | primaryTypographyProps={{ typography: 'body2', noWrap: true }}
|
---|
| 123 | secondaryTypographyProps={{
|
---|
| 124 | mt: 0.5,
|
---|
| 125 | component: 'span',
|
---|
| 126 | typography: 'caption',
|
---|
| 127 | }}
|
---|
| 128 | />
|
---|
| 129 | </TableCell>
|
---|
| 130 |
|
---|
| 131 | <TableCell align="center">{sent}</TableCell>
|
---|
| 132 |
|
---|
| 133 | <TableCell>
|
---|
| 134 | <Label
|
---|
| 135 | variant="soft"
|
---|
| 136 | color={
|
---|
| 137 | (status === 'paid' && 'success') ||
|
---|
| 138 | (status === 'processing' && 'info') ||
|
---|
| 139 | (status === 'pending' && 'warning') ||
|
---|
| 140 | (status === 'overdue' && 'error') ||
|
---|
| 141 | 'default'
|
---|
| 142 | }
|
---|
| 143 | >
|
---|
| 144 | {status}
|
---|
| 145 | </Label>
|
---|
| 146 | </TableCell>
|
---|
| 147 |
|
---|
| 148 | <TableCell align="right" sx={{ px: 1 }}>
|
---|
| 149 | <IconButton color={popover.open ? 'inherit' : 'default'} onClick={popover.onOpen}>
|
---|
| 150 | <Iconify icon="eva:more-vertical-fill" />
|
---|
| 151 | </IconButton>
|
---|
| 152 | </TableCell>
|
---|
| 153 | </TableRow>
|
---|
| 154 |
|
---|
| 155 | <CustomPopover
|
---|
| 156 | open={popover.open}
|
---|
| 157 | onClose={popover.onClose}
|
---|
| 158 | arrow="right-top"
|
---|
| 159 | sx={{ width: 160 }}
|
---|
| 160 | >
|
---|
| 161 | <MenuItem
|
---|
| 162 | onClick={() => {
|
---|
| 163 | onCopyRow();
|
---|
| 164 | popover.onClose();
|
---|
| 165 | }}
|
---|
| 166 | >
|
---|
| 167 | <Iconify icon="solar:copy-bold" />
|
---|
| 168 | Copy
|
---|
| 169 | </MenuItem>
|
---|
| 170 |
|
---|
| 171 | <MenuItem
|
---|
| 172 | onClick={() => {
|
---|
| 173 | onViewRow();
|
---|
| 174 | popover.onClose();
|
---|
| 175 | }}
|
---|
| 176 | >
|
---|
| 177 | <Iconify icon="solar:eye-bold" />
|
---|
| 178 | View
|
---|
| 179 | </MenuItem>
|
---|
| 180 |
|
---|
| 181 | <MenuItem
|
---|
| 182 | onClick={() => {
|
---|
| 183 | onEditRow();
|
---|
| 184 | popover.onClose();
|
---|
| 185 | }}
|
---|
| 186 | disabled={status === 'processing'}
|
---|
| 187 | >
|
---|
| 188 | <Iconify icon="solar:pen-bold" />
|
---|
| 189 | Edit
|
---|
| 190 | </MenuItem>
|
---|
| 191 |
|
---|
| 192 | <MenuItem
|
---|
| 193 | onClick={() => {
|
---|
| 194 | popover.onClose();
|
---|
| 195 | onToggleCompose();
|
---|
| 196 | }}
|
---|
| 197 | sx={{ color: 'info.main' }}
|
---|
| 198 | disabled={status === 'processing' || status === 'paid'}
|
---|
| 199 | >
|
---|
| 200 | <Iconify icon="iconamoon:send-fill" />
|
---|
| 201 | Send
|
---|
| 202 | </MenuItem>
|
---|
| 203 |
|
---|
| 204 | <Divider sx={{ borderStyle: 'dashed' }} />
|
---|
| 205 |
|
---|
| 206 | <MenuItem
|
---|
| 207 | onClick={() => {
|
---|
| 208 | confirmDelete.onTrue();
|
---|
| 209 | popover.onClose();
|
---|
| 210 | }}
|
---|
| 211 | disabled={status === 'processing'}
|
---|
| 212 | sx={{ color: 'error.main' }}
|
---|
| 213 | >
|
---|
| 214 | <Iconify icon="solar:trash-bin-trash-bold" />
|
---|
| 215 | Delete
|
---|
| 216 | </MenuItem>
|
---|
| 217 | </CustomPopover>
|
---|
| 218 |
|
---|
| 219 | <ConfirmDialog
|
---|
| 220 | open={confirmSend.value}
|
---|
| 221 | onClose={confirmSend.onFalse}
|
---|
| 222 | title="Confirm"
|
---|
| 223 | content="Are you sure you want to send?"
|
---|
| 224 | action={
|
---|
| 225 | <Button
|
---|
| 226 | variant="contained"
|
---|
| 227 | color="info"
|
---|
| 228 | onClick={() => {
|
---|
| 229 | onSendRow();
|
---|
| 230 | confirmSend.onFalse();
|
---|
| 231 | }}
|
---|
| 232 | >
|
---|
| 233 | Send
|
---|
| 234 | </Button>
|
---|
| 235 | }
|
---|
| 236 | />
|
---|
| 237 |
|
---|
| 238 | <ConfirmDialog
|
---|
| 239 | open={confirmDelete.value}
|
---|
| 240 | onClose={confirmDelete.onFalse}
|
---|
| 241 | title="Delete"
|
---|
| 242 | content="Are you sure you want to delete?"
|
---|
| 243 | action={
|
---|
| 244 | <Button
|
---|
| 245 | variant="contained"
|
---|
| 246 | color="error"
|
---|
| 247 | onClick={async () => {
|
---|
| 248 | await onDeleteRow();
|
---|
| 249 | confirmDelete.onToggle();
|
---|
| 250 | }}
|
---|
| 251 | >
|
---|
| 252 | Delete
|
---|
| 253 | </Button>
|
---|
| 254 | }
|
---|
| 255 | />
|
---|
| 256 | </>
|
---|
| 257 | );
|
---|
| 258 | }
|
---|