source: src/sections/invoice/invoice-new-edit-details.tsx@ 057453c

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

feat: implement employees

  • Property mode set to 100644
File size: 10.0 KB
Line 
1import sum from 'lodash/sum';
2import { useCallback, useEffect } from 'react';
3import { useFormContext, useFieldArray } from 'react-hook-form';
4// @mui
5import Box from '@mui/material/Box';
6import Stack from '@mui/material/Stack';
7import Button from '@mui/material/Button';
8import Divider from '@mui/material/Divider';
9import MenuItem from '@mui/material/MenuItem';
10import Typography from '@mui/material/Typography';
11import { inputBaseClasses } from '@mui/material/InputBase';
12import InputAdornment from '@mui/material/InputAdornment';
13// utils
14import { fCurrency } from 'src/utils/format-number';
15// types
16import { InvoiceItem } from 'src/schemas';
17// components
18import Iconify from 'src/components/iconify';
19import { RHFSelect, RHFTextField } from 'src/components/hook-form';
20import { useGetServices } from 'src/api/service';
21
22// ----------------------------------------------------------------------
23
24export default function InvoiceNewEditDetails() {
25 const { control, setValue, watch, resetField } = useFormContext();
26
27 const { fields, append, remove } = useFieldArray({
28 control,
29 name: 'items',
30 });
31
32 const { services: invoiceServices } = useGetServices();
33
34 const values = watch();
35
36 const currencySign = values.currency === 'EUR' ? '€' : '$';
37
38 const totalOnRow = values.items.map((item: InvoiceItem) => item.quantity * item.price);
39 const subTotal = sum(totalOnRow);
40 const taxAmount = (values.taxes / 100) * subTotal;
41 const totalAmount = subTotal - values.discount + taxAmount;
42
43 useEffect(() => {
44 setValue('subTotal', subTotal);
45 }, [setValue, subTotal]);
46
47 useEffect(() => {
48 setValue('totalAmount', totalAmount);
49 }, [setValue, totalAmount]);
50
51 const handleAdd = () => {
52 append({
53 title: '',
54 description: '',
55 service: '',
56 quantity: 1,
57 price: 0,
58 total: 0,
59 });
60 };
61
62 const handleRemove = (index: number) => {
63 remove(index);
64 };
65
66 const handleClearService = useCallback(
67 (index: number) => {
68 resetField(`items[${index}].quantity`);
69 resetField(`items[${index}]`);
70 resetField(`items[${index}].total`);
71 },
72 [resetField]
73 );
74
75 const handleSelectService = useCallback(
76 (index: number, option: string) => {
77 const service = invoiceServices.find((service) => service.id === option);
78 if (!service) return;
79
80 const quantityType = values.quantityType?.toLowerCase();
81 let price = 0;
82
83 switch (quantityType) {
84 case 'sprint':
85 price = service.sprint;
86 break;
87 case 'hour':
88 price = service.hour;
89 break;
90 case 'month':
91 price = service.month;
92 break;
93 default:
94 price = 0;
95 }
96
97 setValue(`items[${index}].price`, price);
98 setValue(`items[${index}].total`, values.items[index].quantity * price);
99 },
100 [setValue, values.items, values.quantityType, invoiceServices]
101 );
102
103 const handleChangeQuantity = useCallback(
104 (event: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement>, index: number) => {
105 setValue(`items[${index}].quantity`, Number(event.target.value));
106 setValue(
107 `items[${index}].total`,
108 values.items.map((item: InvoiceItem) => item.quantity * item.price)[index]
109 );
110 },
111 [setValue, values.items]
112 );
113
114 const handleChangePrice = useCallback(
115 (event: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement>, index: number) => {
116 setValue(`items[${index}].price`, Number(event.target.value));
117 setValue(
118 `items[${index}].total`,
119 values.items.map((item: InvoiceItem) => item.quantity * item.price)[index]
120 );
121 },
122 [setValue, values.items]
123 );
124
125 const renderTotal = (
126 <Stack
127 spacing={2}
128 alignItems="flex-end"
129 sx={{ mt: 3, textAlign: 'right', typography: 'body2' }}
130 >
131 <Stack direction="row">
132 <Box sx={{ color: 'text.secondary' }}>Subtotal</Box>
133 <Box sx={{ width: 160, typography: 'subtitle2' }}>
134 {fCurrency(subTotal, values.currency) || '-'}
135 </Box>
136 </Stack>
137
138 {/* <Stack direction="row">
139 <Box sx={{ color: 'text.secondary' }}>Discount</Box>
140 <Box
141 sx={{
142 width: 160,
143 ...(values.discount && { color: 'error.main' }),
144 }}
145 >
146 {values.discount ? `- ${fCurrency(values.discount, values.currency)}` : '-'}
147 </Box>
148 </Stack>
149
150 <Stack direction="row">
151 <Box sx={{ color: 'text.secondary' }}>Taxes</Box>
152 <Box sx={{ width: 160 }}>
153 {values.taxes ? fCurrency(values.taxes, values.currency) : '-'}
154 </Box>
155 </Stack> */}
156
157 <Stack direction="row" sx={{ typography: 'subtitle1' }}>
158 <Box>Total</Box>
159 <Box sx={{ width: 160 }}>{fCurrency(totalAmount, values.currency) || '-'}</Box>
160 </Stack>
161 </Stack>
162 );
163
164 return (
165 <Box sx={{ p: 3 }}>
166 <Typography variant="h6" sx={{ color: 'text.disabled', mb: 3 }}>
167 Details:
168 </Typography>
169
170 <Stack divider={<Divider flexItem sx={{ borderStyle: 'dashed' }} />} spacing={3}>
171 {fields.map((item, index) => (
172 <Stack key={item.id} alignItems="flex-end" spacing={1.5}>
173 <Stack direction={{ xs: 'column', md: 'row' }} spacing={2} sx={{ width: 1 }}>
174 <RHFTextField
175 size="small"
176 name={`items[${index}].title`}
177 label="Title"
178 InputLabelProps={{ shrink: true }}
179 />
180
181 <RHFTextField
182 size="small"
183 name={`items[${index}].description`}
184 label="Description"
185 InputLabelProps={{ shrink: true }}
186 />
187
188 {invoiceServices?.length && (
189 <RHFSelect
190 name={`items[${index}].service`}
191 size="small"
192 label="Service"
193 InputLabelProps={{ shrink: true }}
194 sx={{
195 maxWidth: { md: 160 },
196 }}
197 >
198 <MenuItem
199 value=""
200 onClick={() => handleClearService(index)}
201 sx={{ fontStyle: 'italic', color: 'text.secondary' }}
202 >
203 None
204 </MenuItem>
205
206 <Divider sx={{ borderStyle: 'dashed' }} />
207
208 {invoiceServices.map((service) => (
209 <MenuItem
210 key={service.id}
211 value={service.id}
212 onClick={() => handleSelectService(index, service.id)}
213 >
214 {service.name}
215 </MenuItem>
216 ))}
217 </RHFSelect>
218 )}
219
220 <RHFTextField
221 size="small"
222 type="number"
223 name={`items[${index}].quantity`}
224 label="Quantity"
225 placeholder="0"
226 onChange={(event) => handleChangeQuantity(event, index)}
227 InputLabelProps={{ shrink: true }}
228 sx={{ maxWidth: { md: 96 } }}
229 />
230
231 <RHFTextField
232 size="small"
233 type="number"
234 name={`items[${index}].price`}
235 label="Price"
236 placeholder="0.00"
237 onChange={(event) => handleChangePrice(event, index)}
238 InputProps={{
239 startAdornment: (
240 <InputAdornment position="start">
241 <Box sx={{ typography: 'subtitle2', color: 'text.disabled' }}>
242 {currencySign}
243 </Box>
244 </InputAdornment>
245 ),
246 }}
247 sx={{ maxWidth: { md: 96 } }}
248 />
249
250 <RHFTextField
251 disabled
252 size="small"
253 type="number"
254 name={`items[${index}].total`}
255 label="Total"
256 placeholder="0.00"
257 value={values.items[index].total === 0 ? '' : values.items[index].total.toFixed(2)}
258 onChange={(event) => handleChangePrice(event, index)}
259 InputProps={{
260 startAdornment: (
261 <InputAdornment position="start">
262 <Box sx={{ typography: 'subtitle2', color: 'text.disabled' }}>
263 {currencySign}
264 </Box>
265 </InputAdornment>
266 ),
267 }}
268 sx={{
269 maxWidth: { md: 104 },
270 [`& .${inputBaseClasses.input}`]: {
271 textAlign: { md: 'right' },
272 },
273 }}
274 />
275 </Stack>
276
277 <Button
278 size="small"
279 color="error"
280 startIcon={<Iconify icon="solar:trash-bin-trash-bold" />}
281 onClick={() => handleRemove(index)}
282 >
283 Remove
284 </Button>
285 </Stack>
286 ))}
287 </Stack>
288
289 <Divider sx={{ my: 3, borderStyle: 'dashed' }} />
290
291 <Stack
292 spacing={3}
293 direction={{ xs: 'column', md: 'row' }}
294 alignItems={{ xs: 'flex-end', md: 'center' }}
295 >
296 <Button
297 size="small"
298 color="primary"
299 startIcon={<Iconify icon="mingcute:add-line" />}
300 onClick={handleAdd}
301 sx={{ flexShrink: 0 }}
302 >
303 Add Item
304 </Button>
305
306 <Stack
307 spacing={2}
308 justifyContent="flex-end"
309 direction={{ xs: 'column', md: 'row' }}
310 sx={{ width: 1 }}
311 >
312 <RHFTextField
313 size="small"
314 label={`Discount(${currencySign})`}
315 name="discount"
316 type="number"
317 sx={{ maxWidth: { md: 120 } }}
318 />
319
320 <RHFTextField
321 size="small"
322 label="Taxes(%)"
323 name="taxes"
324 type="number"
325 sx={{ maxWidth: { md: 120 } }}
326 />
327 </Stack>
328 </Stack>
329
330 {renderTotal}
331 </Box>
332 );
333}
Note: See TracBrowser for help on using the repository browser.