diff options
author | eug-vs <eug-vs@keemail.me> | 2020-08-29 14:29:28 +0300 |
---|---|---|
committer | eug-vs <eug-vs@keemail.me> | 2020-08-29 14:29:28 +0300 |
commit | 868e380262dcef15d9645ceb912d18701ecb61fb (patch) | |
tree | 2cef481a57733d814b1b0a1e5dffc926db668179 /src/containers | |
parent | 935daf265d21f0c2ec34ce79ff468474c0a728d1 (diff) | |
download | which-ui-868e380262dcef15d9645ceb912d18701ecb61fb.tar.gz |
feat: use yup and formik
Diffstat (limited to 'src/containers')
-rw-r--r-- | src/containers/Registration/Registration.tsx | 203 |
1 files changed, 84 insertions, 119 deletions
diff --git a/src/containers/Registration/Registration.tsx b/src/containers/Registration/Registration.tsx index 422cf92..b5c56e6 100644 --- a/src/containers/Registration/Registration.tsx +++ b/src/containers/Registration/Registration.tsx @@ -1,10 +1,7 @@ -import React, { - useState, - useRef, - FormEvent, - useMemo -} from 'react'; +import React, { useState } from 'react'; import { useHistory } from 'react-router-dom'; +import { Formik, Form, Field } from 'formik'; +import * as Yup from 'yup'; import { makeStyles } from '@material-ui/core/styles'; import { TextField, @@ -16,6 +13,22 @@ import { Visibility, VisibilityOff } from '@material-ui/icons'; import { post } from '../../requests'; import { useAuth } from '../../hooks/useAuth'; +interface Fields { + username: string; + email: string; + password: string; +} + +const validationSchema = Yup.object({ + username: Yup.string() + .required('This field is required'), + email: Yup.string() + .email('Invalid email address') + .required('This field is required'), + password: Yup.string() + .min(6, 'Should be at least 6 characters') + .required('This field is required'), +}); const useStyles = makeStyles(theme => ({ root: { @@ -42,137 +55,89 @@ const useStyles = makeStyles(theme => ({ cursor: 'pointer' }, textField: { - height: 70 + height: theme.spacing(8) } })); -interface ErrorStates { - username: boolean | undefined; - email: boolean | undefined; - password: boolean | undefined; -} - const Registration: React.FC = () => { - const [errors, setErrors] = useState<ErrorStates>({ - username: undefined, - email: undefined, - password: undefined - }); - const [showPassword, setShowPassword] = useState<boolean>(false); - const classes = useStyles(); - const usernameRef = useRef<HTMLInputElement>(); - const emailRef = useRef<HTMLInputElement>(); - const passwordRef = useRef<HTMLInputElement>(); const { login } = useAuth(); const history = useHistory(); - - const isValid = useMemo(() => { - return !errors.username && !errors.email && !errors.password; - }, [errors]); - - const handleSubmit = (event: FormEvent<HTMLFormElement>) => { - event.preventDefault(); - const username = usernameRef.current?.value?.toLowerCase(); - const password = passwordRef.current?.value; - const email = emailRef.current?.value; - if (username && password && isValid) { - post('/users', { username, password, email }) - .then(() => login(username, password)) - .then(() => history.push(`/profile/${username}`)); - } - }; + const [showPassword, setShowPassword] = useState<boolean>(false); const handleLogin = () => { history.push('/login'); }; - const handleClickShowPassword = () => { - setShowPassword(prevState => !prevState); - }; - const handleMouseDownPassword = (event: React.MouseEvent<HTMLButtonElement>) => { - event.preventDefault(); - }; - const handleUsernameChange = (event: React.FocusEvent<HTMLInputElement>) => { - setErrors({ - ...errors, - username: event.currentTarget.value.length === 0 - }); - }; - const handleEmailChange = (event: React.FocusEvent<HTMLInputElement>) => { - setErrors({ - ...errors, - email: !/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,}$/i.test(event.currentTarget.value) - }); - }; - const handlePasswordChange = (event: React.FocusEvent<HTMLInputElement>) => { - setErrors({ - ...errors, - password: event.currentTarget.value.length < 6 - }); - }; - const handleToLowerCase = (e: React.ChangeEvent<HTMLInputElement>) => { - e.target.value = e.target.value.toString().toLowerCase(); - }; + const handleSubmit = ({ username, email, password }: Fields) => { + post('/users', { username, email, password }) + .then(() => login(username, password)) + .then(() => history.push(`/profile/${username}`)); + } - const handleFocus = (value: string) => () => { - setErrors({ - ...errors, - [value]: undefined - }); + const handleClickShowPassword = () => { + setShowPassword(prevState => !prevState); }; return ( <> <div className={classes.formHeader}>Sign Up</div> - <form className={classes.root} noValidate autoComplete="off" onSubmit={handleSubmit}> - <TextField - inputRef={usernameRef} - label="Username" - error={errors.username} - helperText={errors.username && 'This field is required'} - required - onBlur={handleUsernameChange} - onInput={handleToLowerCase} - onFocus={handleFocus('username')} - className={classes.textField} - /> - <TextField - inputRef={emailRef} - label="Email" - error={errors.email} - helperText={errors.email && 'Invalid email address'} - onBlur={handleEmailChange} - onFocus={handleFocus('email')} - className={classes.textField} - /> - <TextField - inputRef={passwordRef} - label="Password" - required - error={errors.password} - helperText={errors.password && 'Should be at least 6 characters'} - type={showPassword ? 'text' : 'password'} - onBlur={handlePasswordChange} - onFocus={handleFocus('password')} - InputProps={{ - endAdornment: ( - <InputAdornment position="end"> - <IconButton - size="small" - aria-label="toggle password visibility" - onClick={handleClickShowPassword} - onMouseDown={handleMouseDownPassword} - > - {showPassword ? <Visibility /> : <VisibilityOff />} - </IconButton> - </InputAdornment> - ) - }} - className={classes.textField} - /> - <Button variant="contained" type="submit">submit</Button> - </form> + <Formik + initialValues={{ username: '', email: '', password: '' }} + validationSchema={validationSchema} + onSubmit={handleSubmit} + > + {({ values, errors, touched, isSubmitting }) => ( + <Form className={classes.root}> + <Field + id="username" + name="username" + label="Username" + value={values.username.toLowerCase()} + error={touched.username && !!errors.username} + helperText={touched.username && errors.username} + required + className={classes.textField} + as={TextField} + /> + <Field + name="email" + label="Email" + value={values.email} + error={touched.email && !!errors.email} + helperText={touched.email && errors.email} + required + className={classes.textField} + as={TextField} + /> + <Field + name="password" + label="Password" + value={values.password} + error={touched.password && !!errors.password} + helperText={touched.password && errors.password} + required + type={showPassword ? 'text' : 'password'} + as={TextField} + InputProps={{ + endAdornment: ( + <InputAdornment position="end"> + <IconButton + size="small" + aria-label="toggle password visibility" + onClick={handleClickShowPassword} + > + {showPassword ? <Visibility /> : <VisibilityOff />} + </IconButton> + </InputAdornment> + ) + }} + className={classes.textField} + /> + <Button variant="contained" type="submit" disabled={isSubmitting}>submit</Button> + </Form> + )} + </Formik> <div className={classes.formTransfer}> <div>Already have an account?</div> <span |