source: src/sections/invoice/invoice-details.tsx@ 5d6f37a

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

add customer

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