source: src/sections/user/customer-new-edit-form.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.3 KB
Line 
1import * as Yup from 'yup';
2import { useCallback, useMemo } from 'react';
3import { useForm, Controller } from 'react-hook-form';
4import { zodResolver } from '@hookform/resolvers/zod';
5// @mui
6import LoadingButton from '@mui/lab/LoadingButton';
7import Box from '@mui/material/Box';
8import Card from '@mui/material/Card';
9import Stack from '@mui/material/Stack';
10import Switch from '@mui/material/Switch';
11import Grid from '@mui/material/Unstable_Grid2';
12import Typography from '@mui/material/Typography';
13import FormControlLabel from '@mui/material/FormControlLabel';
14// utils
15import { fData } from 'src/utils/format-number';
16// routes
17import { paths } from 'src/routes/paths';
18import { useRouter } from 'src/routes/hooks';
19// types
20import { Customer, NewCustomer, newCustomerSchema } from 'mvpmasters-shared';
21// components
22import Label from 'src/components/label';
23import { useSnackbar } from 'src/components/snackbar';
24import FormProvider, { RHFTextField, RHFUploadAvatar } from 'src/components/hook-form';
25import uploadToFirebaseStorage from 'src/utils/upload-to-firebase-storage';
26import { addDoc, collection } from 'firebase/firestore';
27import { db } from 'src/lib/firebase';
28import { createCustomer } from 'src/api/customer';
29
30// ----------------------------------------------------------------------
31
32type Props = {
33 currentUser?: Customer;
34};
35
36export default function CustomerNewEditForm({ currentUser: currentCustomer }: Props) {
37 const router = useRouter();
38
39 const { enqueueSnackbar } = useSnackbar();
40
41 const defaultValues: NewCustomer = useMemo(
42 () => ({
43 name: currentCustomer?.name || '',
44 representative: currentCustomer?.representative || '',
45 email: currentCustomer?.email || '',
46 logoUrl: currentCustomer?.logoUrl || null,
47 address: currentCustomer?.address || {
48 country: '',
49 state: '',
50 street: '',
51 zip: '',
52 city: '',
53 },
54 vatNumber: currentCustomer?.vatNumber || '',
55 companyId: currentCustomer?.companyId || '',
56 companyNumber: currentCustomer?.companyNumber || '',
57 id: currentCustomer?.id || '',
58 phoneNumber: currentCustomer?.phoneNumber || '',
59 status: 'active',
60 }),
61 [currentCustomer]
62 );
63
64 const methods = useForm({
65 resolver: zodResolver(newCustomerSchema),
66 defaultValues,
67 });
68
69 const {
70 reset,
71 watch,
72 control,
73 setValue,
74 handleSubmit,
75 formState: { isSubmitting },
76 } = methods;
77
78 const values = watch();
79
80 const onSubmit = handleSubmit(async (data) => {
81 try {
82 // await new Promise((resolve) => setTimeout(resolve, 500));
83 const logoFile = data.logoUrl as File & { preview: string };
84
85 const storagePath: string = `customers/${logoFile.name}`;
86 const logoUrl = await uploadToFirebaseStorage(logoFile, storagePath);
87
88 // const customersRef = collection(db, 'customers');
89 // await addDoc(customersRef, { ...data, logoUrl });
90 await createCustomer({ ...data, logoUrl });
91
92 reset();
93 enqueueSnackbar(currentCustomer ? 'Update success!' : 'Create success!');
94 router.push(paths.dashboard.customer.list);
95 console.info('DATA', data);
96 } catch (error) {
97 console.error(error);
98 }
99 });
100
101 const handleDrop = useCallback(
102 (acceptedFiles: File[]) => {
103 const file = acceptedFiles[0];
104
105 const newFile = Object.assign(file, {
106 preview: URL.createObjectURL(file),
107 });
108
109 if (file) {
110 setValue('logoUrl', newFile, { shouldValidate: true });
111 }
112 },
113 [setValue]
114 );
115
116 return (
117 <FormProvider methods={methods} onSubmit={onSubmit}>
118 <Grid container spacing={3}>
119 <Grid xs={12} md={4}>
120 <Card sx={{ pt: 10, pb: 5, px: 3 }}>
121 {currentCustomer && (
122 <Label
123 color={
124 (values.status === 'active' && 'success') ||
125 (values.status === 'banned' && 'error') ||
126 'warning'
127 }
128 sx={{ position: 'absolute', top: 24, right: 24 }}
129 >
130 {values.status}
131 </Label>
132 )}
133
134 <Box sx={{ mb: 5 }}>
135 <RHFUploadAvatar
136 name="logoUrl"
137 maxSize={3145728}
138 onDrop={handleDrop}
139 helperText={
140 <Typography
141 variant="caption"
142 sx={{
143 mt: 3,
144 mx: 'auto',
145 display: 'block',
146 textAlign: 'center',
147 color: 'text.disabled',
148 }}
149 >
150 Allowed *.jpeg, *.jpg, *.png, *.gif
151 <br /> max size of {fData(3145728)}
152 </Typography>
153 }
154 />
155 </Box>
156
157 {currentCustomer && (
158 <FormControlLabel
159 labelPlacement="start"
160 control={
161 <Controller
162 name="status"
163 control={control}
164 render={({ field }) => (
165 <Switch
166 {...field}
167 checked={field.value !== 'active'}
168 onChange={(event) =>
169 field.onChange(event.target.checked ? 'banned' : 'active')
170 }
171 />
172 )}
173 />
174 }
175 label={
176 <>
177 <Typography variant="subtitle2" sx={{ mb: 0.5 }}>
178 Banned
179 </Typography>
180 <Typography variant="body2" sx={{ color: 'text.secondary' }}>
181 Apply disable account
182 </Typography>
183 </>
184 }
185 sx={{ mx: 0, mb: 3, width: 1, justifyContent: 'space-between' }}
186 />
187 )}
188 </Card>
189 </Grid>
190
191 <Grid xs={12} md={8}>
192 <Card sx={{ p: 3 }}>
193 <Box
194 rowGap={3}
195 columnGap={2}
196 display="grid"
197 gridTemplateColumns={{
198 xs: 'repeat(1, 1fr)',
199 sm: 'repeat(2, 1fr)',
200 }}
201 >
202 <RHFTextField name="name" label="Name" />
203 <RHFTextField name="email" label="Email Address" />
204 <RHFTextField name="representative" label="Representative" />
205 <RHFTextField name="phoneNumber" label="Phone Number" />
206
207 <RHFTextField name="address.country" label="Country" />
208
209 {/* <RHFAutocomplete
210 name="address.country"
211 label="Country"
212 options={countries.map((country) => country.label)}
213 getOptionLabel={(option) => option}
214 isOptionEqualToValue={(option, value) => option === value}
215 renderOption={(props, option) => {
216 const { code, label, phone } = countries.filter(
217 (country) => country.label === option
218 )[0];
219
220 if (!label) {
221 return null;
222 }
223
224 return (
225 <li {...props} key={label}>
226 <Iconify
227 key={label}
228 icon={`circle-flags:${code.toLowerCase()}`}
229 width={28}
230 sx={{ mr: 1 }}
231 />
232 {label} ({code}) +{phone}
233 </li>
234 );
235 }}
236 /> */}
237
238 <RHFTextField name="address.state" label="State/Region" />
239 <RHFTextField name="address.city" label="City" />
240 <RHFTextField name="address.street" label="Street" />
241 <RHFTextField name="address.zip" label="Zip/Code" />
242 <RHFTextField name="vatNumber" label="VAT" />
243 <RHFTextField name="companyNumber" label="Company Number" />
244 </Box>
245
246 <Stack alignItems="flex-end" sx={{ mt: 3 }}>
247 <LoadingButton type="submit" variant="contained" loading={isSubmitting}>
248 {!currentCustomer ? 'Create Customer' : 'Save Changes'}
249 </LoadingButton>
250 </Stack>
251 </Card>
252 </Grid>
253 </Grid>
254 </FormProvider>
255 );
256}
Note: See TracBrowser for help on using the repository browser.