diff --git a/api/src/controllers/user.js b/api/src/controllers/user.js index 23172fe41e..13cebb81b7 100644 --- a/api/src/controllers/user.js +++ b/api/src/controllers/user.js @@ -330,7 +330,7 @@ router.post( }; const prevUser = await User.findOne({ where: { email: newUser.email } }); - if (prevUser) return res.status(400).send({ ok: false, error: "A user already exists with this email" }); + if (prevUser) return res.status(400).send({ ok: false, error: "Un utilisateur existe déjà avec cet email" }); const data = await User.create(newUser, { returning: true }); diff --git a/dashboard/src/components/createWrapper.js b/dashboard/src/components/createWrapper.js deleted file mode 100644 index cb98f71f16..0000000000 --- a/dashboard/src/components/createWrapper.js +++ /dev/null @@ -1,10 +0,0 @@ -import styled from 'styled-components'; - -const CreateWrapper = styled.div` - margin-bottom: 40px; - width: 100%; - display: flex; - justify-content: flex-end; -` - -export default CreateWrapper; diff --git a/dashboard/src/scenes/place/list.js b/dashboard/src/scenes/place/list.js index e7f14c5588..f0667f28e3 100644 --- a/dashboard/src/scenes/place/list.js +++ b/dashboard/src/scenes/place/list.js @@ -6,7 +6,6 @@ import { toast } from 'react-toastify'; import { SmallHeader } from '../../components/header'; import ButtonCustom from '../../components/ButtonCustom'; import Loading from '../../components/loading'; -import CreateWrapper from '../../components/createWrapper'; import Table from '../../components/table'; import Search from '../../components/search'; import Page from '../../components/pagination'; @@ -110,7 +109,7 @@ const Create = () => { const setPlaces = useSetRecoilState(placesState); return ( - +
setOpen(true)} @@ -152,7 +151,7 @@ const Create = () => { - +
); }; diff --git a/dashboard/src/scenes/team/list.js b/dashboard/src/scenes/team/list.js index d70e311960..f0ed77eefc 100644 --- a/dashboard/src/scenes/team/list.js +++ b/dashboard/src/scenes/team/list.js @@ -7,7 +7,6 @@ import { useRecoilState, useRecoilValue, useSetRecoilState } from 'recoil'; import { SmallHeader } from '../../components/header'; import ButtonCustom from '../../components/ButtonCustom'; -import CreateWrapper from '../../components/createWrapper'; import Table from '../../components/table'; import NightSessionModale from '../../components/NightSessionModale'; import { currentTeamState, organisationState, teamsState, userState } from '../../recoil/auth'; @@ -99,7 +98,7 @@ const Create = () => { const onboardingForTeams = !teams.length; return ( - +
setOpen(true)} title="Créer une nouvelle équipe" padding="12px 24px" /> setOpen(false)} size="lg" backdrop="static"> : null} toggle={() => setOpen(false)}> @@ -201,7 +200,7 @@ const Create = () => { - +
); }; diff --git a/dashboard/src/scenes/user/list.js b/dashboard/src/scenes/user/list.js index 7da4f1d932..672422e733 100644 --- a/dashboard/src/scenes/user/list.js +++ b/dashboard/src/scenes/user/list.js @@ -1,23 +1,19 @@ import React, { useEffect, useMemo, useState } from 'react'; -import { Col, FormGroup, Input, Modal, ModalBody, ModalHeader, Row, Label } from 'reactstrap'; import { useHistory } from 'react-router-dom'; -import { Formik } from 'formik'; import { toast } from 'react-toastify'; -import styled from 'styled-components'; import { useRecoilValue } from 'recoil'; +import { teamsState, userState } from '../../recoil/auth'; +import API from '../../services/api'; +import { formatDateWithFullMonth } from '../../services/date'; +import useTitle from '../../services/useTitle'; +import { useLocalStorage } from '../../services/useLocalStorage'; +import { ModalBody, ModalContainer, ModalHeader, ModalFooter } from '../../components/tailwind/Modal'; import { SmallHeader } from '../../components/header'; import SelectTeamMultiple from '../../components/SelectTeamMultiple'; import Loading from '../../components/loading'; -import ButtonCustom from '../../components/ButtonCustom'; import Table from '../../components/table'; -import CreateWrapper from '../../components/createWrapper'; import TagTeam from '../../components/TagTeam'; -import { userState } from '../../recoil/auth'; -import API from '../../services/api'; -import { formatDateWithFullMonth } from '../../services/date'; -import useTitle from '../../services/useTitle'; import SelectRole from '../../components/SelectRole'; -import { useLocalStorage } from '../../services/useLocalStorage'; import { emailRegex } from '../../utils'; const defaultSort = (a, b, sortOrder) => (sortOrder === 'ASC' ? (a.name || '').localeCompare(b.name) : (b.name || '').localeCompare(a.name)); @@ -57,25 +53,24 @@ const List = () => { const [sortBy, setSortBy] = useLocalStorage('users-sortBy', 'createdAt'); const [sortOrder, setSortOrder] = useLocalStorage('users-sortOrder', 'ASC'); - const getUsers = async () => { - const response = await API.get({ path: '/user' }); - if (response.error) return toast.error(response.error); - setUsers(response.data); - setRefresh(false); - }; - const data = useMemo(() => users.sort(sortUsers(sortBy, sortOrder)), [users, sortBy, sortOrder]); useEffect(() => { - getUsers(); - // eslint-disable-next-line react-hooks/exhaustive-deps + API.get({ path: '/user' }).then((response) => { + if (response.error) { + toast.error(response.error); + return false; + } + setUsers(response.data); + setRefresh(false); + }); }, [refresh]); - if (!!refresh) return ; + if (!users.length) return ; return ( <> - {['superadmin', 'admin'].includes(user.role) && setRefresh(true)} />} + {['admin'].includes(user.role) && setRefresh(true)} />} { dataKey: 'teams', render: (user) => { return ( - +
{user.teams.map((t) => ( ))} - +
); }, }, @@ -150,115 +145,199 @@ const List = () => { ); }; -const TeamWrapper = styled.div` - display: grid; - grid-auto-flow: row; - row-gap: 5px; -`; - -const Create = ({ onChange }) => { +const Create = ({ onChange, users }) => { const [open, setOpen] = useState(false); + const [isSubmitting, setIsSubmitting] = useState(false); + const teams = useRecoilValue(teamsState); + const initialState = useMemo(() => { + return { + name: '', + email: '', + role: 'normal', + team: teams.map((t) => t._id), + healthcareProfessional: false, + }; + }, [teams]); + useEffect(() => { + if (!open) setData(initialState); + }, [open, initialState]); + const [data, setData] = useState(initialState); + const handleChange = (event) => { + const target = event.currentTarget || event.target; + const { name, value } = target; + setData((data) => ({ ...data, [name]: value })); + }; + + const handleSubmit = async () => { + try { + if (data.role === 'restricted-access') data.healthcareProfessional = false; + if (!data.email) { + toast.error("L'email est obligatoire"); + return false; + } + if (!emailRegex.test(data.email)) { + toast.error("L'email est invalide"); + return false; + } + if (!data.name) { + toast.error('Le nom est obligatoire'); + return false; + } + if (!data.role) { + toast.error('Le rôle est obligatoire'); + return false; + } + if (!data.team?.length) { + toast.error('Veuillez sélectionner une équipe'); + return false; + } + setIsSubmitting(true); + const { ok } = await API.post({ path: '/user', body: data }); + setIsSubmitting(false); + if (!ok) { + return false; + } + toast.success('Création réussie !'); + onChange(); + setData(initialState); + return true; + } catch (errorCreatingUser) { + console.log('error in creating user', errorCreatingUser); + toast.error(errorCreatingUser.message); + setIsSubmitting(false); + return false; + } + }; + return ( - - setOpen(true)} color="primary" title="Créer un nouvel utilisateur" padding="12px 24px" /> - setOpen(false)} size="lg" backdrop="static"> - setOpen(false)}>Créer un nouvel utilisateur +
+ + setOpen(false)} size="full"> + setOpen(false)} title="Créer de nouveaux utilisateurs" /> - { - const errors = {}; - if (values.role === 'restricted-access') values.healthcareProfessional = false; - if (!values.name) errors.name = 'Le nom est obligatoire'; - if (!values.email) errors.email = "L'email est obligatoire"; - else if (!emailRegex.test(values.email)) errors.email = "L'email est invalide"; - if (!values.role) errors.role = 'Le rôle est obligatoire'; - if (!values.team?.length) errors.team = 'Veuillez sélectionner une équipe'; - return errors; - }} - onSubmit={async (body, actions) => { - try { - const { ok } = await API.post({ path: '/user', body }); - actions.setSubmitting(false); - if (!ok) return; - toast.success('Création réussie !'); - onChange(); - setOpen(false); - } catch (errorCreatingUser) { - console.log('error in creating user', errorCreatingUser); - toast.error(errorCreatingUser.message); - } + className="tw-p-4" + onSubmit={async (e) => { + e.preventDefault(); + await handleSubmit(); }}> - {({ values, handleChange, handleSubmit, isSubmitting, errors, touched }) => ( - - -
- - - - {touched.name && errors.name &&
{errors.name}
} -
- - - - - - {touched.email && errors.email &&
{errors.email}
} -
- - - - - - {touched.role && errors.role &&
{errors.role}
} -
- - - - -
- handleChange({ target: { value: teamIds, name: 'team' } })} - value={values.team || []} - colored - inputId="team" - /> - {touched.team && errors.team &&
{errors.team}
} -
-
- - {values.role !== 'restricted-access' && ( - - -
- Un professionnel·le de santé a accès au dossier médical complet des personnes. -
- - )} - -
- - - - - - - )} - +
+
+ + { + const target = event.currentTarget || event.target; + const { value } = target; + if (data.email === data.name) { + setData((data) => ({ ...data, email: value, name: value })); + } else { + setData((data) => ({ ...data, email: value })); + } + }} + /> +
+
+ +
+ +
+
+
+ +
+ handleChange({ target: { value: teamIds, name: 'team' } })} + value={data.team} + inputId="team" + name="team" + /> +
+
+
+ + +
+ {data.role !== 'restricted-access' && ( +
+ +
+ Un professionnel·le de santé a accès au dossier médical complet des personnes. +
+
+ )} +
+
+ Utilisateurs existants ({users?.length ?? 0}) +
    + {users.map((user) => { + return ( + +
    {user.name}
    +
    {user.email}
    +
    + ); + })} +
+
+ - - + + + + + + + ); };