source: src/sections/invoice/mail-compose.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: 8.1 KB
Line 
1import { useState, useMemo } from 'react';
2// @mui
3import Paper from '@mui/material/Paper';
4import Stack from '@mui/material/Stack';
5import Portal from '@mui/material/Portal';
6import Backdrop from '@mui/material/Backdrop';
7import IconButton from '@mui/material/IconButton';
8import Typography from '@mui/material/Typography';
9// hooks
10import { useBoolean } from 'src/hooks/use-boolean';
11import { useResponsive } from 'src/hooks/use-responsive';
12// components
13import Iconify from 'src/components/iconify';
14import Editor from 'src/components/editor';
15import { Invoice } from 'src/schemas';
16import FormProvider from 'src/components/hook-form/form-provider';
17import { RHFTextField } from 'src/components/hook-form';
18import { useForm } from 'react-hook-form';
19import { LoadingButton } from '@mui/lab';
20import { enqueueSnackbar } from 'notistack';
21import {
22 collectionRef,
23 collections,
24 createDocId,
25 createDocument,
26 firestoreBatch,
27} from 'src/lib/firestore';
28import { storage } from 'src/lib/firebase';
29import { getDownloadURL, ref } from 'firebase/storage';
30import { increment } from 'firebase/firestore';
31import { mutate } from 'swr';
32
33// ----------------------------------------------------------------------
34
35const ZINDEX = 1998;
36
37const POSITION = 24;
38
39type Props = {
40 invoice: Invoice | null;
41 onCloseCompose: VoidFunction;
42 invoiceMutationKey: string[];
43};
44
45export default function MailCompose({ invoice, onCloseCompose, invoiceMutationKey }: Props) {
46 const smUp = useResponsive('up', 'sm');
47
48 const fullScreen = useBoolean(false);
49
50 // const handleChangeMessage = useCallback((value: string) => {
51 // setMessage(value);
52 // }, []);
53
54 const defaultValues = useMemo(
55 () => ({
56 to: invoice?.invoiceTo?.email || '',
57 subject: `New Invoice for ${invoice?.month} - MVP Masters`,
58 message: `<p>Dear ${
59 invoice?.invoiceTo?.representative || ''
60 }, </p><p></p><p>We hope this email finds you well. Please find attached the invoice for ${invoice?.month}, detailing the work MVP Masters has completed during that period. We appreciate your prompt attention to this invoice. </p><p></p><p>Should you have any questions or require further clarification on any items, please don't hesitate to reply to this email.</p><p></p><p>Thank you for your continued partnership and trust in MVP Masters.</p><p><br></p><p>Warm regards,</p><p>Finance Team</p><p>MVP Masters</p>`,
61 }),
62 [invoice]
63 );
64
65 const methods = useForm({
66 defaultValues,
67 });
68
69 const {
70 reset,
71 handleSubmit,
72 watch,
73 setValue,
74 formState: { isSubmitting },
75 } = methods;
76
77 const values = watch();
78
79 const onSubmit = handleSubmit(async (data) => {
80 try {
81 if (!invoice) return;
82 // await new Promise((resolve) => setTimeout(resolve, 500));
83
84 const pdfFilename = invoice.pdfRef?.split('/').pop() as string;
85 const invoicePDFRef = ref(storage, invoice.pdfRef);
86 const url = await getDownloadURL(invoicePDFRef);
87
88 const mailData = {
89 from: '<MVP Masters - Finance Team> finance@mvpmasters.com',
90 to: invoice.invoiceTo.email,
91 message: {
92 subject: data.subject,
93 html: data.message,
94 attachments: [
95 {
96 filename: pdfFilename,
97 path: url,
98 },
99 ],
100 },
101 };
102
103 const invoiceData = {
104 sent: increment(1),
105 status: 'pending',
106 };
107
108 const mailId = createDocId(collections.mail);
109
110 // write to DB
111 await firestoreBatch<any>([
112 {
113 docPath: `${collections.mail}/${mailId}`,
114 type: 'set',
115 data: mailData,
116 },
117 {
118 docPath: `${collections.invoice}/${invoice.id}`,
119 type: 'set',
120 data: invoiceData,
121 },
122 ]);
123
124 mutate(invoiceMutationKey);
125
126 reset();
127 onCloseCompose();
128 enqueueSnackbar('Invoice sent!');
129 console.info('DATA', data);
130 } catch (error) {
131 console.error(error);
132 }
133 });
134
135 return (
136 <Portal>
137 {(fullScreen.value || !smUp) && <Backdrop open sx={{ zIndex: ZINDEX }} />}
138
139 <Paper
140 sx={{
141 right: 0,
142 bottom: 0,
143 borderRadius: 2,
144 display: 'flex',
145 position: 'fixed',
146 zIndex: ZINDEX + 1,
147 m: `${POSITION}px`,
148 overflow: 'hidden',
149 flexDirection: 'column',
150 boxShadow: (theme) => theme.customShadows.dropdown,
151 ...(fullScreen.value && {
152 m: 0,
153 right: POSITION / 2,
154 bottom: POSITION / 2,
155 width: `calc(100% - ${POSITION}px)`,
156 height: `calc(100% - ${POSITION}px)`,
157 }),
158 }}
159 >
160 <Stack
161 direction="row"
162 alignItems="center"
163 sx={{
164 bgcolor: 'background.neutral',
165 p: (theme) => theme.spacing(1.5, 1, 1.5, 2),
166 }}
167 >
168 <Typography variant="h6" sx={{ flexGrow: 1 }}>
169 Inv. {invoice?.invoiceNumber} - {invoice?.invoiceTo?.name}
170 </Typography>
171
172 <IconButton onClick={fullScreen.onToggle}>
173 <Iconify icon={fullScreen ? 'eva:collapse-fill' : 'eva:expand-fill'} />
174 </IconButton>
175
176 <IconButton onClick={onCloseCompose}>
177 <Iconify icon="mingcute:close-line" />
178 </IconButton>
179 </Stack>
180
181 <FormProvider methods={methods} onSubmit={onSubmit}>
182 <RHFTextField sx={{ px: 2, py: 2 }} variant="standard" name="to" placeholder="To" />
183 <RHFTextField
184 sx={{ px: 2, py: 1 }}
185 variant="standard"
186 name="subject"
187 placeholder="Subject"
188 />
189
190 {/* <InputBase
191 placeholder="To"
192 // endAdornment={
193 // <Stack direction="row" spacing={0.5} sx={{ typography: 'subtitle2' }}>
194 // <Box
195 // sx={{
196 // cursor: 'pointer',
197 // '&:hover': { textDecoration: 'underline' },
198 // }}
199 // >
200 // Cc
201 // </Box>
202 // <Box
203 // sx={{
204 // cursor: 'pointer',
205 // '&:hover': { textDecoration: 'underline' },
206 // }}
207 // >
208 // Bcc
209 // </Box>
210 // </Stack>
211 // }
212 sx={{
213 px: 2,
214 height: 48,
215 borderBottom: (theme) => `solid 1px ${alpha(theme.palette.grey[500], 0.08)}`,
216 }}
217 /> */}
218
219 {/* <InputBase
220 placeholder="Subject"
221 sx={{
222 px: 2,
223 height: 48,
224 borderBottom: (theme) => `solid 1px ${alpha(theme.palette.grey[500], 0.08)}`,
225 }}
226 /> */}
227
228 <Stack spacing={2} flexGrow={1} sx={{ p: 2 }}>
229 <Editor
230 simple
231 id="compose-mail"
232 // value={message}
233 // onChange={handleChangeMessage}
234 value={values.message}
235 onChange={(value) => setValue('message', value)}
236 placeholder="Type a message"
237 sx={{
238 '& .ql-editor': {},
239 ...(fullScreen.value && {
240 height: 1,
241 '& .quill': {
242 height: 1,
243 },
244 '& .ql-editor': {
245 maxHeight: 'unset',
246 },
247 }),
248 }}
249 />
250
251 <Stack direction="row" alignItems="center">
252 <Stack direction="row" alignItems="center" flexGrow={1}>
253 <IconButton>
254 <Iconify icon="solar:gallery-add-bold" />
255 </IconButton>
256
257 <IconButton>
258 <Iconify icon="eva:attach-2-fill" />
259 </IconButton>
260 </Stack>
261
262 <LoadingButton
263 type="submit"
264 variant="contained"
265 color="primary"
266 loading={isSubmitting}
267 endIcon={<Iconify icon="iconamoon:send-fill" />}
268 >
269 Send
270 </LoadingButton>
271 </Stack>
272 </Stack>
273 </FormProvider>
274 </Paper>
275 </Portal>
276 );
277}
Note: See TracBrowser for help on using the repository browser.