source: src/sections/invoice/invoice-details.tsx@ 87c9f1e

main
Last change on this file since 87c9f1e was 87c9f1e, checked in by Naum Shapkarovski <naumshapkarovski@…>, 5 weeks ago

update the seed script. update the prisma schema, use mapping

  • Property mode set to 100644
File size: 8.6 KB
Line 
1import { useCallback } from 'react';
2// @mui
3import { styled } from '@mui/material/styles';
4import Box from '@mui/material/Box';
5import Card from '@mui/material/Card';
6import Table from '@mui/material/Table';
7import Stack from '@mui/material/Stack';
8import Divider from '@mui/material/Divider';
9import TableRow from '@mui/material/TableRow';
10import TableHead from '@mui/material/TableHead';
11import TableCell from '@mui/material/TableCell';
12import TableBody from '@mui/material/TableBody';
13import Grid from '@mui/material/Unstable_Grid2';
14import Typography from '@mui/material/Typography';
15import TableContainer from '@mui/material/TableContainer';
16// utils
17import { fDate } from 'src/utils/format-time';
18import { fCurrency } from 'src/utils/format-number';
19// types
20import { Invoice } from 'src/schemas';
21// components
22import Label from 'src/components/label';
23import Scrollbar from 'src/components/scrollbar';
24//
25import { mutate } from 'swr';
26import { collections, updateDocument } from 'src/lib/firestore';
27import { createFullAddress } from 'src/utils/create-full-address';
28import { getQuantityType } from 'src/utils/get-invoice-quantity-type';
29import InvoiceToolbar from './invoice-toolbar';
30import { updateInvoice } from 'src/api/invoice';
31
32// ----------------------------------------------------------------------
33
34const StyledTableRow = styled(TableRow)(({ theme }) => ({
35 '& td': {
36 textAlign: 'right',
37 borderBottom: 'none',
38 paddingTop: theme.spacing(1),
39 paddingBottom: theme.spacing(1),
40 },
41}));
42
43// ----------------------------------------------------------------------
44
45type Props = {
46 invoice: Invoice;
47};
48
49export const INVOICE_STATUS_OPTIONS = [
50 { value: 'draft', label: 'Draft' },
51 // { value: 'processing', label: 'Processing' },
52 { value: 'paid', label: 'Paid' },
53 { value: 'pending', label: 'Pending' },
54 { value: 'overdue', label: 'Overdue' },
55];
56
57export default function InvoiceDetails({ invoice }: Props) {
58 const currentStatus = invoice.status;
59
60 const handleChangeStatus = useCallback(
61 async (event: React.ChangeEvent<HTMLInputElement>) => {
62 // await updateDocument(collections.invoice, invoice.id, { status: event.target.value });
63 await updateInvoice(invoice.id, {
64 status: event.target.value as 'draft' | 'processing' | 'pending' | 'overdue' | 'paid',
65 });
66 mutate([collections.invoice, invoice.id]);
67 },
68 [invoice]
69 );
70
71 const renderTotal = (
72 <>
73 <StyledTableRow>
74 <TableCell colSpan={4} />
75
76 <TableCell sx={{ color: 'text.secondary' }}>
77 <Box sx={{ mt: 2 }} />
78 Subtotal
79 </TableCell>
80 <TableCell width={120} sx={{ typography: 'subtitle2' }}>
81 <Box sx={{ mt: 2 }} />
82 {fCurrency(invoice.subTotal, invoice.currency)}
83 </TableCell>
84 </StyledTableRow>
85
86 {!!invoice.discount && (
87 <StyledTableRow>
88 <TableCell colSpan={4} />
89 <TableCell sx={{ color: 'text.secondary' }}>Discount</TableCell>
90 <TableCell width={120} sx={{ color: 'error.main', typography: 'body2' }}>
91 {fCurrency(-invoice.discount, invoice.currency)}
92 </TableCell>
93 </StyledTableRow>
94 )}
95
96 {/* <StyledTableRow>
97 <TableCell colSpan={3} />
98 <TableCell sx={{ color: 'text.secondary' }}>Taxes</TableCell>
99 <TableCell width={120}>{fCurrency(invoice.taxes)}</TableCell>
100 </StyledTableRow> */}
101
102 <StyledTableRow>
103 <TableCell colSpan={4} />
104 <TableCell sx={{ typography: 'subtitle1' }}>Total</TableCell>
105 <TableCell width={140} sx={{ typography: 'subtitle1' }}>
106 {fCurrency(invoice.totalAmount, invoice.currency)}
107 </TableCell>
108 </StyledTableRow>
109 </>
110 );
111
112 const renderFooter = (
113 <Grid container>
114 <Grid xs={12} md={9} sx={{ py: 3 }}>
115 <Typography variant="subtitle2">NOTES</Typography>
116
117 <Typography variant="body2">
118 VAT is not calculated according to article 14, paragraph 3, point 5 from the VAT act.
119 </Typography>
120 </Grid>
121
122 <Grid xs={12} md={3} sx={{ py: 3, textAlign: 'right' }}>
123 <Typography variant="subtitle2">Have a Question?</Typography>
124
125 <Typography variant="body2">{invoice.invoiceFrom.email}</Typography>
126 </Grid>
127 </Grid>
128 );
129
130 const renderList = (
131 <TableContainer sx={{ overflow: 'unset', mt: 5 }}>
132 <Scrollbar>
133 <Table sx={{ minWidth: 960 }}>
134 <TableHead>
135 <TableRow>
136 <TableCell width={40}>#</TableCell>
137
138 <TableCell sx={{ typography: 'subtitle2' }}>Description</TableCell>
139
140 <TableCell align="right">{getQuantityType(invoice)}</TableCell>
141
142 <TableCell align="right">Currency</TableCell>
143
144 <TableCell align="right">Unit price</TableCell>
145
146 <TableCell align="right">Total</TableCell>
147 </TableRow>
148 </TableHead>
149
150 <TableBody>
151 {invoice.items.map((row, index) => (
152 <TableRow key={index}>
153 <TableCell>{index + 1}</TableCell>
154
155 <TableCell>
156 <Box sx={{ maxWidth: 560 }}>
157 <Typography variant="subtitle2">{row.title}</Typography>
158
159 <Typography variant="body2" sx={{ color: 'text.secondary' }} noWrap>
160 {row.description}
161 </Typography>
162 </Box>
163 </TableCell>
164
165 <TableCell align="right">{row.quantity}</TableCell>
166
167 <TableCell align="right">{invoice.currency}</TableCell>
168
169 <TableCell align="right">{fCurrency(row.price, invoice.currency)}</TableCell>
170
171 <TableCell align="right">
172 {fCurrency(row.price * row.quantity, invoice.currency)}
173 </TableCell>
174 </TableRow>
175 ))}
176
177 {renderTotal}
178 </TableBody>
179 </Table>
180 </Scrollbar>
181 </TableContainer>
182 );
183
184 return (
185 <>
186 <InvoiceToolbar
187 invoice={invoice}
188 currentStatus={currentStatus || ''}
189 onChangeStatus={handleChangeStatus}
190 statusOptions={INVOICE_STATUS_OPTIONS}
191 />
192
193 <Card sx={{ pt: 5, px: 5 }}>
194 <Box
195 rowGap={5}
196 display="grid"
197 alignItems="center"
198 gridTemplateColumns={{
199 xs: 'repeat(1, 1fr)',
200 sm: 'repeat(2, 1fr)',
201 }}
202 >
203 <Box
204 component="img"
205 alt="logo"
206 src="/logo/logo_single.svg"
207 sx={{ width: 48, height: 48 }}
208 />
209
210 <Stack spacing={1} alignItems={{ xs: 'flex-start', md: 'flex-end' }}>
211 <Label
212 variant="soft"
213 color={
214 (currentStatus === 'paid' && 'success') ||
215 (currentStatus === 'processing' && 'info') ||
216 (currentStatus === 'pending' && 'warning') ||
217 (currentStatus === 'overdue' && 'error') ||
218 'default'
219 }
220 >
221 {currentStatus}
222 </Label>
223
224 <Typography variant="h6">{invoice.invoiceNumber}</Typography>
225 </Stack>
226
227 <Stack sx={{ typography: 'body2', whiteSpace: 'pre-line' }}>
228 <Typography variant="subtitle2" sx={{ mb: 1 }}>
229 Invoice From
230 </Typography>
231 {invoice.invoiceFrom.name}
232 <br />
233 {createFullAddress(invoice.invoiceFrom.address)}
234 <br />
235 <br />
236 {invoice.invoiceFrom.email}
237 <br />
238 </Stack>
239
240 <Stack sx={{ typography: 'body2', whiteSpace: 'pre-line' }}>
241 <Typography variant="subtitle2" sx={{ mb: 1 }}>
242 Invoice To
243 </Typography>
244 {invoice.invoiceTo.name}
245 <br />
246 {!!invoice.invoiceTo.companyNumber && (
247 <>
248 Company ID: {invoice.invoiceTo.companyNumber}
249 <br />
250 </>
251 )}
252 {createFullAddress(invoice.invoiceTo.address)}
253 <br />
254 <br />
255 {invoice.invoiceTo.email}
256 <br />
257 </Stack>
258
259 <Stack sx={{ typography: 'body2' }}>
260 <Typography variant="subtitle2" sx={{ mb: 1 }}>
261 Date Issued
262 </Typography>
263 {fDate(invoice.issueDate)}
264 </Stack>
265
266 <Stack sx={{ typography: 'body2' }}>
267 <Typography variant="subtitle2" sx={{ mb: 1 }}>
268 Due Date
269 </Typography>
270 {fDate(invoice.dueDate)}
271 </Stack>
272 </Box>
273
274 {renderList}
275
276 <Divider sx={{ mt: 5, borderStyle: 'dashed' }} />
277
278 {renderFooter}
279 </Card>
280 </>
281 );
282}
Note: See TracBrowser for help on using the repository browser.