Files
full-stack-fastapi-template/frontend/src/components/UserSettings/UserInformation.tsx

147 lines
4.0 KiB
TypeScript
Raw Normal View History

2024-03-08 14:58:36 +01:00
import React, { useState } from 'react'
import {
Box,
Button,
Container,
Flex,
FormControl,
FormErrorMessage,
FormLabel,
Heading,
Input,
Text,
useColorModeValue,
} from '@chakra-ui/react'
import { SubmitHandler, useForm } from 'react-hook-form'
import { useMutation, useQueryClient } from 'react-query'
2024-03-08 14:58:36 +01:00
import { ApiError, UserOut, UserUpdateMe, UsersService } from '../../client'
import useAuth from '../../hooks/useAuth'
import useCustomToast from '../../hooks/useCustomToast'
const UserInformation: React.FC = () => {
2024-03-08 14:58:36 +01:00
const queryClient = useQueryClient()
2024-03-11 16:50:46 +01:00
const color = useColorModeValue('inherit', 'ui.white')
2024-03-08 14:58:36 +01:00
const showToast = useCustomToast()
const [editMode, setEditMode] = useState(false)
const { user: currentUser } = useAuth()
const {
register,
handleSubmit,
reset,
getValues,
2024-03-08 14:58:36 +01:00
formState: { isSubmitting, errors, isDirty },
} = useForm<UserOut>({
mode: 'onBlur',
criteriaMode: 'all',
defaultValues: {
full_name: currentUser?.full_name,
email: currentUser?.email,
},
})
2024-03-08 14:58:36 +01:00
const toggleEditMode = () => {
setEditMode(!editMode)
}
2024-03-08 14:58:36 +01:00
const updateInfo = async (data: UserUpdateMe) => {
await UsersService.updateUserMe({ requestBody: data })
}
2024-03-08 14:58:36 +01:00
const mutation = useMutation(updateInfo, {
onSuccess: () => {
showToast('Success!', 'User updated successfully.', 'success')
},
onError: (err: ApiError) => {
const errDetail = err.body.detail
showToast('Something went wrong.', `${errDetail}`, 'error')
},
onSettled: () => {
queryClient.invalidateQueries('users')
queryClient.invalidateQueries('currentUser')
},
})
2024-03-08 14:58:36 +01:00
const onSubmit: SubmitHandler<UserUpdateMe> = async (data) => {
mutation.mutate(data)
}
2024-03-08 14:58:36 +01:00
const onCancel = () => {
reset()
toggleEditMode()
}
2024-03-08 14:58:36 +01:00
return (
<>
<Container maxW="full" as="form" onSubmit={handleSubmit(onSubmit)}>
<Heading size="sm" py={4}>
User Information
</Heading>
<Box w={{ sm: 'full', md: '50%' }}>
<FormControl>
<FormLabel color={color} htmlFor="name">
Full name
</FormLabel>
{editMode ? (
<Input
id="name"
{...register('full_name', { maxLength: 30 })}
type="text"
size="md"
/>
) : (
<Text size="md" py={2}>
{currentUser?.full_name || 'N/A'}
</Text>
)}
</FormControl>
<FormControl mt={4} isInvalid={!!errors.email}>
<FormLabel color={color} htmlFor="email">
Email
</FormLabel>
{editMode ? (
<Input
id="email"
{...register('email', {
required: 'Email is required',
pattern: {
value: /^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}$/i,
message: 'Invalid email address',
},
})}
2024-03-11 16:50:46 +01:00
type="email"
2024-03-08 14:58:36 +01:00
size="md"
/>
) : (
<Text size="md" py={2}>
{currentUser!.email}
</Text>
)}
{errors.email && (
<FormErrorMessage>{errors.email.message}</FormErrorMessage>
)}
</FormControl>
<Flex mt={4} gap={3}>
<Button
2024-03-11 16:50:46 +01:00
variant="primary"
2024-03-08 14:58:36 +01:00
onClick={toggleEditMode}
type={editMode ? 'button' : 'submit'}
isLoading={editMode ? isSubmitting : false}
isDisabled={editMode ? !isDirty || !getValues('email') : false}
2024-03-08 14:58:36 +01:00
>
{editMode ? 'Save' : 'Edit'}
</Button>
{editMode && (
<Button onClick={onCancel} isDisabled={isSubmitting}>
Cancel
</Button>
)}
</Flex>
</Box>
</Container>
</>
)
}
2024-03-08 14:58:36 +01:00
export default UserInformation