[5d6f37a] | 1 | import sum from 'lodash/sum';
|
---|
| 2 | import { useCallback, useEffect } from 'react';
|
---|
| 3 | import { useFormContext, useFieldArray } from 'react-hook-form';
|
---|
| 4 | // @mui
|
---|
| 5 | import Box from '@mui/material/Box';
|
---|
| 6 | import Stack from '@mui/material/Stack';
|
---|
| 7 | import Button from '@mui/material/Button';
|
---|
| 8 | import Divider from '@mui/material/Divider';
|
---|
| 9 | import MenuItem from '@mui/material/MenuItem';
|
---|
| 10 | import Typography from '@mui/material/Typography';
|
---|
| 11 | import { inputBaseClasses } from '@mui/material/InputBase';
|
---|
| 12 | import InputAdornment from '@mui/material/InputAdornment';
|
---|
| 13 | // utils
|
---|
| 14 | import { fCurrency } from 'src/utils/format-number';
|
---|
| 15 | // types
|
---|
[057453c] | 16 | import { InvoiceItem } from 'src/schemas';
|
---|
[5d6f37a] | 17 | // components
|
---|
| 18 | import Iconify from 'src/components/iconify';
|
---|
| 19 | import { RHFSelect, RHFTextField } from 'src/components/hook-form';
|
---|
| 20 | import { useGetServices } from 'src/api/service';
|
---|
| 21 |
|
---|
| 22 | // ----------------------------------------------------------------------
|
---|
| 23 |
|
---|
| 24 | export 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) => {
|
---|
[057453c] | 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);
|
---|
[5d6f37a] | 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 | }
|
---|