import React, { useEffect, useState } from 'react'; import { useRef } from 'react'; import { useOutletContext } from 'react-router-dom'; import { Alert, Button, FormControl, FormControlLabel, FormLabel, Grid, Radio, RadioGroup, Snackbar, TextField, } from '@mui/material'; import { useForm, Controller, useFieldArray } from 'react-hook-form'; import { useUpdateGuestMutation } from '../slices/apiSlice'; import type { Guest } from '../slices/apiSlice'; interface StatusProps { isError: boolean; setOpen: (open: boolean) => void; } const Status = ({ isError, setOpen }: StatusProps) => { return isError ? ( setOpen(false)}> RSVP failed ) : ( setOpen(false)}> RSVP updated ); }; function RsvpForm() { const [updateGuest, { isLoading, isSuccess, isError, error }] = useUpdateGuestMutation(); const guest: Guest = useOutletContext(); const previousPartySize = useRef(guest?.partySize - 1); const [open, setOpen] = useState(false); useEffect(() => { setOpen(isSuccess || isError); }, [isSuccess, isError]); const { register, handleSubmit, control, watch, formState: { errors }, } = useForm({ defaultValues: { id: guest?.id, firstName: guest?.firstName, lastName: guest?.lastName, attendance: guest?.attendance, email: guest?.email, message: guest?.message, partySize: guest?.partySize, partyList: guest?.partyList, }, }); const onSubmit = async (data: Guest) => { updateGuest({ ...data }); }; const { fields, append, remove } = useFieldArray({ control, name: 'partyList', }); const handleParty = () => { const partySize = Number(watch('partySize')) - 1; if ( partySize > previousPartySize.current && partySize > 0 && partySize < 10 ) { append( new Array(partySize - previousPartySize.current).fill({ firstName: '', lastName: '', }) ); previousPartySize.current = partySize; } else if (partySize < previousPartySize.current && partySize >= 0) { remove( [...Array(previousPartySize.current - partySize).keys()].map( (_, i) => partySize - 1 + i ) ); previousPartySize.current = partySize; } }; return ( Please RSVP for the wedding by March 10, 2025. The ceremony will commence at 2 PM on April 26 in Divine Shepherd. The reception will follow at 5 PM in A Venue on the Ridge. Attendance ( } label="Accept" /> } label="Decline" /> )} /> value.trim(), required: 'This field is required', pattern: { value: /\S+@\S+\.\S+/, message: 'Please enter a valid email address', }, })} /> { event.currentTarget.blur(); }} error={!!errors.partySize} helperText={errors.partySize?.message} required {...register('partySize', { valueAsNumber: true, onChange: handleParty, required: 'This field is required', min: { value: 1, message: 'Please enter a positive integer' }, max: { value: 9, message: 'Please enter an integer less than 10', }, })} /> {fields.map((field, index) => { return ( value.trim(), required: 'This field is required', })} /> value.trim(), required: 'This field is required', })} /> ); })} RSVP setOpen(false)} autoHideDuration={5000} > ); } export default RsvpForm;
Please RSVP for the wedding by March 10, 2025. The ceremony will commence at 2 PM on April 26 in Divine Shepherd. The reception will follow at 5 PM in A Venue on the Ridge.