index.js 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. import React, { useState, useEffect } from "react";
  2. import * as Yup from "yup";
  3. import { Formik, Form, Field } from "formik";
  4. import { toast } from "react-toastify";
  5. import { makeStyles } from "@material-ui/core/styles";
  6. import { green } from "@material-ui/core/colors";
  7. import Button from "@material-ui/core/Button";
  8. import TextField from "@material-ui/core/TextField";
  9. import Dialog from "@material-ui/core/Dialog";
  10. import DialogActions from "@material-ui/core/DialogActions";
  11. import DialogContent from "@material-ui/core/DialogContent";
  12. import DialogTitle from "@material-ui/core/DialogTitle";
  13. import CircularProgress from "@material-ui/core/CircularProgress";
  14. import { i18n } from "../../translate/i18n";
  15. import api from "../../services/api";
  16. import toastError from "../../errors/toastError";
  17. const useStyles = makeStyles((theme) => ({
  18. root: {
  19. display: "flex",
  20. flexWrap: "wrap",
  21. },
  22. multFieldLine: {
  23. display: "flex",
  24. "& > *:not(:last-child)": {
  25. marginRight: theme.spacing(1),
  26. },
  27. },
  28. btnWrapper: {
  29. position: "relative",
  30. },
  31. buttonProgress: {
  32. color: green[500],
  33. position: "absolute",
  34. top: "50%",
  35. left: "50%",
  36. marginTop: -12,
  37. marginLeft: -12,
  38. },
  39. formControl: {
  40. margin: theme.spacing(1),
  41. minWidth: 120,
  42. },
  43. }));
  44. const ContactListSchema = Yup.object().shape({
  45. name: Yup.string()
  46. .min(2, i18n.t("contactLists.dialog.nameShort"))
  47. .max(50, i18n.t("contactLists.dialog.nameLong"))
  48. .required(i18n.t("contactLists.dialog.nameRequired")),
  49. });
  50. const ContactListModal = ({ open, onClose, contactListId }) => {
  51. const classes = useStyles();
  52. const initialState = {
  53. name: "",
  54. };
  55. const [contactList, setContactList] = useState(initialState);
  56. useEffect(() => {
  57. const fetchContactList = async () => {
  58. if (!contactListId) return;
  59. try {
  60. const { data } = await api.get(`/contact-lists/${contactListId}`);
  61. setContactList((prevState) => {
  62. return { ...prevState, ...data };
  63. });
  64. } catch (err) {
  65. toastError(err);
  66. }
  67. };
  68. fetchContactList();
  69. }, [contactListId, open]);
  70. const handleClose = () => {
  71. onClose();
  72. setContactList(initialState);
  73. };
  74. const handleSaveContactList = async (values) => {
  75. const contactListData = { ...values };
  76. try {
  77. if (contactListId) {
  78. await api.put(`/contact-lists/${contactListId}`, contactListData);
  79. } else {
  80. await api.post("/contact-lists", contactListData);
  81. }
  82. toast.success(i18n.t("contactList.toasts.success"));
  83. } catch (err) {
  84. toastError(err);
  85. }
  86. handleClose();
  87. };
  88. return (
  89. <div className={classes.root}>
  90. <Dialog
  91. open={open}
  92. onClose={handleClose}
  93. maxWidth="xs"
  94. fullWidth
  95. scroll="paper"
  96. >
  97. <DialogTitle id="form-dialog-title">
  98. {contactListId
  99. ? `${i18n.t("contactLists.dialog.edit")}`
  100. : `${i18n.t("contactLists.dialog.add")}`}
  101. </DialogTitle>
  102. <Formik
  103. initialValues={contactList}
  104. enableReinitialize={true}
  105. validationSchema={ContactListSchema}
  106. onSubmit={(values, actions) => {
  107. setTimeout(() => {
  108. handleSaveContactList(values);
  109. actions.setSubmitting(false);
  110. }, 400);
  111. }}
  112. >
  113. {({ touched, errors, isSubmitting }) => (
  114. <Form>
  115. <DialogContent dividers>
  116. <div className={classes.multFieldLine}>
  117. <Field
  118. as={TextField}
  119. label={i18n.t("contactLists.dialog.name")}
  120. autoFocus
  121. name="name"
  122. error={touched.name && Boolean(errors.name)}
  123. helperText={touched.name && errors.name}
  124. variant="outlined"
  125. margin="dense"
  126. fullWidth
  127. />
  128. </div>
  129. </DialogContent>
  130. <DialogActions>
  131. <Button
  132. onClick={handleClose}
  133. color="secondary"
  134. disabled={isSubmitting}
  135. variant="outlined"
  136. >
  137. {i18n.t("contactLists.dialog.cancel")}
  138. </Button>
  139. <Button
  140. type="submit"
  141. color="primary"
  142. disabled={isSubmitting}
  143. variant="contained"
  144. className={classes.btnWrapper}
  145. >
  146. {contactListId
  147. ? `${i18n.t("contactLists.dialog.okEdit")}`
  148. : `${i18n.t("contactLists.dialog.okAdd")}`}
  149. {isSubmitting && (
  150. <CircularProgress
  151. size={24}
  152. className={classes.buttonProgress}
  153. />
  154. )}
  155. </Button>
  156. </DialogActions>
  157. </Form>
  158. )}
  159. </Formik>
  160. </Dialog>
  161. </div>
  162. );
  163. };
  164. export default ContactListModal;