diff --git a/ui/components/SatisfactionForm/SatisfactionForm.tsx b/ui/components/SatisfactionForm/SatisfactionForm.tsx index a51e9aac69..539b57008a 100644 --- a/ui/components/SatisfactionForm/SatisfactionForm.tsx +++ b/ui/components/SatisfactionForm/SatisfactionForm.tsx @@ -1,16 +1,15 @@ -import { Box, Button, Flex, FormLabel, Input, Spacer, Text, Textarea } from "@chakra-ui/react" -import { useFormik } from "formik" -import { useRouter } from "next/router" +import { Box, Button, Flex, Text, Textarea } from "@chakra-ui/react" +import { Formik } from "formik" import { useEffect, useState } from "react" import { ApplicantIntention } from "shared/constants/application" import * as Yup from "yup" import { apiPost } from "../../utils/api.utils" -import { isNonEmptyString } from "../../utils/strutils" -import { testingParameters } from "../../utils/testingParameters" +import { CustomInput } from "../espace_pro" +import { CustomFormControl } from "../espace_pro/CustomFormControl" import SatisfactionFormNavigation from "./SatisfactionFormNavigation" -import SatisfactionFormSuccess from "./SatisfactionFormSuccess" +import { SatisfactionFormSuccess } from "./SatisfactionFormSuccess" const textAreaProperties = { border: "none", @@ -22,103 +21,86 @@ const textAreaProperties = { borderBottom: "2px solid", } -const inputProperties = { - border: "none", - background: "grey.200", - borderRadius: "4px 4px 0px 0px", - width: { base: "100%", md: "280px", lg: "380px" }, - height: "40px", - paddingLeft: 4, - borderBottom: "2px solid", -} - -const getFieldColor = (status) => { - return status === "is-valid-true" ? "#008941" : status === "is-valid-false" ? "#e10600" : "grey.600" -} - -const SatisfactionForm = () => { - const router = useRouter() - const { company_recruitment_intention, id, fn, ln, token } = router.query as { - company_recruitment_intention: string - id: string - fn: string - ln: string - token: string | undefined - } +export const SatisfactionForm = ({ + company_recruitment_intention, + id, + firstName, + lastName, + token, +}: { + company_recruitment_intention: ApplicantIntention + id: string + firstName: string + lastName: string + token: string | undefined +}) => { + const [sendingState, setSendingState] = useState<"not_sent" | "ok_sent" | "not_sent_because_of_errors">("not_sent") + useEffect(() => { + apiPost("/application/intention/:id", { + params: { id }, + body: { company_recruitment_intention }, + headers: { authorization: `Bearer ${token}` }, + }) + }, []) const isRefusedState = company_recruitment_intention === ApplicantIntention.REFUS - const getFeedbackText = () => { - const firstName = fn - const lastName = ln - const text = ( - - {company_recruitment_intention === ApplicantIntention.ENTRETIEN && ( - - Vous souhaitez rencontrer le/la candidat(e) ? - Répondez à {`${firstName} ${lastName}`} et proposez-lui une date de rencontre. - Rédigez votre réponse personnalisée. - - - Le candidat recevra votre commentaire ainsi que vos coordonnées directement sur sa boîte mail. - + const intentionVariants: Record = { + [ApplicantIntention.ENTRETIEN]: { + header: ( + <> + Vous souhaitez rencontrer le/la candidat(e) ? + Répondez à {`${firstName} ${lastName}`} et proposez-lui une date de rencontre. + Rédigez votre réponse personnalisée. + + + Le candidat recevra votre commentaire ainsi que vos coordonnées directement sur sa boîte mail. - - )} - {company_recruitment_intention === ApplicantIntention.NESAISPAS && ( - - La candidature de {`${firstName} ${lastName}`} vous intéresse, mais vous ne souhaitez pas prendre votre décision aujourd’hui ? - Indiquez-lui que vous lui apporterez une réponse prochainement. - Rédigez votre réponse personnalisée. - - - Le candidat recevra votre commentaire ainsi que vos coordonnées directement sur sa boîte mail. - + + > + ), + placeholderTextArea: + "Bonjour, Merci pour l'intérêt que vous portez à notre établissement. Votre candidature a retenu toute notre attention et nous souhaiterions échanger avec vous. Seriez-vous disponible le ... ", + }, + [ApplicantIntention.NESAISPAS]: { + header: ( + <> + La candidature de {`${firstName} ${lastName}`} vous intéresse, mais vous ne souhaitez pas prendre votre décision aujourd’hui ? + Indiquez-lui que vous lui apporterez une réponse prochainement. + Rédigez votre réponse personnalisée. + + + Le candidat recevra votre commentaire ainsi que vos coordonnées directement sur sa boîte mail. - - )} - {company_recruitment_intention === ApplicantIntention.REFUS && ( - - Vous souhaitez refuser la candidature ? - Indiquez au candidat {`${firstName} ${lastName}`} les raisons de ce refus. Une réponse personnalisée l’aidera pour ses futures recherches. - Rédigez votre réponse personnalisée. - - - Le candidat recevra votre message directement sur sa boîte mail. - + + > + ), + placeholderTextArea: "Bonjour, Merci pour l'intérêt que vous portez à notre établissement. Votre candidature a retenu toute notre attention et nous vous répondrons ...", + }, + [ApplicantIntention.REFUS]: { + header: ( + <> + Vous souhaitez refuser la candidature ? + Indiquez au candidat {`${firstName} ${lastName}`} les raisons de ce refus. Une réponse personnalisée l’aidera pour ses futures recherches. + Rédigez votre réponse personnalisée. + + + Le candidat recevra votre message directement sur sa boîte mail. - - )} - - ) - - return text + + > + ), + placeholderTextArea: + "Bonjour, Merci pour l'intérêt que vous portez à notre établissement. Nous ne sommes malheureusement pas en mesure de donner une suite favorable à votre candidature car ...", + }, } - useEffect(() => { - if (!router.isReady) return - const sendIntention = async () => { - await apiPost("/application/intention/:id", { - params: { id }, - body: { - company_recruitment_intention, - }, - headers: { - authorization: `Bearer ${token}`, - }, - }) - } - sendIntention() - }, [router.isReady]) - - const [sendingState, setSendingState] = useState("not_sent") - - const submitForm = async ({ email, phone, company_feedback }) => - await apiPost("/application/intentionComment/:id", { + const submitForm = async ({ email, phone, company_feedback }: { email: string; phone: string; company_feedback: string }) => { + apiPost("/application/intentionComment/:id", { params: { id }, body: { - phone: phone, - email: email, + phone, + email, company_feedback, company_recruitment_intention, }, @@ -126,147 +108,96 @@ const SatisfactionForm = () => { authorization: `Bearer ${token}`, }, }) - - const formik = useFormik({ - initialValues: { company_feedback: "", email: "", phone: "" }, - validationSchema: Yup.object().shape({ - company_feedback: Yup.string().nullable().required("Veuillez remplir le message"), - email: isRefusedState ? Yup.string() : Yup.string().email("⚠ Adresse e-mail invalide").required("⚠ L'adresse e-mail est obligatoire"), - phone: isRefusedState - ? Yup.string() - : Yup.string() - .matches(/^[0-9]{10}$/, "⚠ Le numéro de téléphone doit avoir exactement 10 chiffres") - .required("⚠ Le téléphone est obligatoire"), - }), - onSubmit: async (formikValues) => { - await submitForm(formikValues) - .then(() => setSendingState("ok_sent")) - .catch(() => setSendingState("not_sent_because_of_errors")) - }, - }) - - const getErrorForMessage = (message) => { - return ( - // @ts-expect-error: TODO - - {message || "pas d'erreur"} - - ) - } - - const getPlaceHolderText = () => { - switch (company_recruitment_intention) { - case ApplicantIntention.NESAISPAS: - return "Bonjour, Merci pour l'intérêt que vous portez à notre établissement. Votre candidature a retenu toute notre attention et nous vous répondrons ..." - - case ApplicantIntention.ENTRETIEN: - return "Bonjour, Merci pour l'intérêt que vous portez à notre établissement. Votre candidature a retenu toute notre attention et nous souhaiterions échanger avec vous. Seriez-vous disponible le ... " - - default: - return "Bonjour, Merci pour l'intérêt que vous portez à notre établissement. Nous ne sommes malheureusement pas en mesure de donner une suite favorable à votre candidature car ..." - } + .then(() => setSendingState("ok_sent")) + .catch(() => setSendingState("not_sent_because_of_errors")) } - const getFieldStatus = (formikObj, target) => { - let res = "is-not-validated" - if (formikObj.errors[target]) { - res = "is-valid-false" - } else if (formikObj.values[target]) { - res = "is-valid-true" - } - return res - } - - const commentFieldStatus = getFieldStatus(formik, "company_feedback") - const emailFieldStatus = getFieldStatus(formik, "email") - const phoneFieldStatus = getFieldStatus(formik, "phone") - return ( {sendingState !== "ok_sent" && ( - {getFeedbackText()} - {isNonEmptyString(company_recruitment_intention) && ( - - - - - - {formik.touched.company_feedback && formik.errors.company_feedback && getErrorForMessage(formik.errors.company_feedback)} - - {company_recruitment_intention !== ApplicantIntention.REFUS && ( + + {intentionVariants[company_recruitment_intention].header} + + + submitForm(formikValues)} + > + {({ values, handleChange, handleBlur, submitForm, dirty, isValid, isSubmitting }) => { + return ( <> - - Indiquez au candidat vos coordonnées, afin qu'il puisse vous recontacter. - - - - E-mail * - + + - {formik.touched.email && formik.errors.email && getErrorForMessage(formik.errors.email)} - {testingParameters?.simulatedRecipient ? Les emails seront envoyés à {testingParameters.simulatedRecipient} : ""} + + + + {!isRefusedState && ( + <> + + Indiquez au candidat vos coordonnées, afin qu'il puisse vous recontacter. + + + + + + + + + + > + )} + + {sendingState === "not_sent_because_of_errors" ? ( + + + Une erreur technique s'est produite + - - - Téléphone * - - {formik.touched.phone && formik.errors.phone && getErrorForMessage(formik.errors.phone)} - - + ) : ( + + + Envoyer le message + + + )} > - )} - - {sendingState === "not_sent_because_of_errors" ? ( - - {getErrorForMessage("Une erreur technique s'est produite")} - - ) : ( - - - Envoyer le message - - - )} - - - )} + ) + }} + + )} - {sendingState === "ok_sent" && } + {sendingState === "ok_sent" && } ) } - -export default SatisfactionForm diff --git a/ui/components/SatisfactionForm/SatisfactionFormSuccess.tsx b/ui/components/SatisfactionForm/SatisfactionFormSuccess.tsx index e4d033427b..df3f63adf8 100644 --- a/ui/components/SatisfactionForm/SatisfactionFormSuccess.tsx +++ b/ui/components/SatisfactionForm/SatisfactionFormSuccess.tsx @@ -1,21 +1,13 @@ import { Flex, Text } from "@chakra-ui/react" -import { useRouter } from "next/router" -import React from "react" - -const SatisfactionFormSuccess = () => { - const router = useRouter() - - const readIntention = () => { - const { intention } = router?.query ? router.query : { intention: "intention" } - return intention - } +import { ApplicantIntention } from "shared/constants/application" +export const SatisfactionFormSuccess = ({ intention }: { intention: ApplicantIntention }) => { return ( Merci d'avoir pris le temps d'envoyer un message au candidat. - {readIntention() === "ne_sais_pas" || readIntention() === "entretien" ? ( + {intention === ApplicantIntention.NESAISPAS || intention === ApplicantIntention.ENTRETIEN ? ( Il dispose désormais de vos coordonnées pour poursuivre l'échange. @@ -27,5 +19,3 @@ const SatisfactionFormSuccess = () => { ) } - -export default SatisfactionFormSuccess diff --git a/ui/pages/formulaire-intention.tsx b/ui/pages/formulaire-intention.tsx index b5a84257a1..4a3ff15924 100644 --- a/ui/pages/formulaire-intention.tsx +++ b/ui/pages/formulaire-intention.tsx @@ -1,12 +1,27 @@ +import { useRouter } from "next/router" import { NextSeo } from "next-seo" +import { parseEnum } from "shared" -import SatisfactionForm from "../components/SatisfactionForm/SatisfactionForm" +import { SatisfactionForm } from "@/components/SatisfactionForm/SatisfactionForm" + +import { ApplicantIntention } from "../../shared/constants/application" const FormulaireIntention = () => { + const router = useRouter() + const { company_recruitment_intention, id, fn, ln, token } = router.query as { + company_recruitment_intention: string + id: string + fn: string + ln: string + token: string | undefined + } + return ( <> - + {router.isReady && ( + + )} > ) }