Proyectos de Subversion LeadersLinked - Antes de SPA

Rev

Rev 6510 | Autoría | Comparar con el anterior | Ultima modificación | Ver Log |

import React, { useRef, useState, useEffect } from 'react'
import { axios } from '../../../../utils'
import { Link } from 'react-router-dom'
import { useForm } from 'react-hook-form'
import { useDispatch, useSelector } from 'react-redux'
import { addNotification } from '../../../redux/notification/notification.actions'
import styled from 'styled-components'
import Recaptcha from 'react-recaptcha'
import CryptoJSAesJson from '../../../utils/crypto-js/cryptojs-aes-format'

import Spinner from '../../UI/Spinner'
import SwitchInput from '../../UI/SwitchInput'
import FormErrorFeedback from '../../UI/FormErrorFeedback'

const StyledCheck = styled.div`
  display: flex;
  flex-direction: column;
  justify-content: center;
  align-items: center;
  img {
    width: 100px;
    margin-bottom: 1rem;
  }
  p {
    text-align: center;
  }
`
const StyledSpinnerContainer = styled.div`
  position: absolute;
  left: 0;
  top: 0;
  width: 100%;
  height: 100%;
  background: rgba(255, 255, 255, 0.4);
  display: flex;
  justify-content: center;
  align-items: center;
  z-index: 300;
`
const Signup = () => {
  const { site_key, aes } = useSelector(({ auth }) => auth)
  const { register, handleSubmit, setError, errors, watch } = useForm({
    mode: 'onBlur',
  })
  const [termsChecked, setTermsChecked] = useState(false)
  const [registered, setRegistered] = useState(false)
  const [isLoading, setIsLoading] = useState(false)
  const [isAdult, setIsAdult] = useState(false)
  const [isVerified, setIsVerified] = useState(false)

  const reCaptchaInstance = useRef(null)
  const reCaptchaToken = useRef('')

  const dispatch = useDispatch()

  const signupVerifyCallbackHandler = (response) => {
    if (!response) return
    reCaptchaToken.current = response
    setIsVerified(true)
  }

  const signupExpiredCallbackHandler = () => {
    reCaptchaToken.current = ''
    setIsVerified(false)
  }

  const onSubmitHandler = async (data) => {
    setIsLoading(true)
    const formData = new FormData()

    Object.entries(data).map(([key, value]) => {
      if (key === 'email' || key === 'password' || key === 'confirmation')
        return formData.append(key, CryptoJSAesJson.encrypt(value, aes))
      return formData.append(key, value)
    })
    formData.append('captcha', reCaptchaToken.current)
    formData.append('is_adult', isAdult ? 'y' : 'n')

    await axios
      .post('/signup', formData)
      .then(({ data }) => {
        if (!data.success) {
          if (typeof data.data !== 'string') {
            Object.entries(data.data).map(([key, value]) => {
              setError(key, {
                type: 'manual',
                message: Array.isArray(value) ? value[0] : value,
              })
            })
          }

          dispatch(addNotification({ style: 'danger', msg: data.data }))
          reCaptchaInstance.current.reset()
          signupVerifyCallbackHandler()
          return
        }

        reCaptchaInstance.current.reset()
        signupExpiredCallbackHandler()
        setRegistered(true)
      })
      .catch((err) => {
        dispatch(
          addNotification({
            style: 'danger',
            msg: 'Disculpe, ha ocurrido un error',
          })
        )
        console.log(`Error: ${err}`)
        throw new Error(err)
      })
      .finally(() => setIsLoading(false))
  }

  const handleOnRecaptchaLoad = () => {
    reCaptchaToken.current = ''
  }

  useEffect(() => {
    reCaptchaInstance.current?.reset()
  }, [])

  if (registered) {
    return (
      <StyledCheck>
        <img src="/images/check.png" alt="check" />
        <p>
          Se ha registrado correctamente. Por favor, active la cuenta desde su
          correo
        </p>
        <Link to="/signin">
          <button id="btn-submit" className="sign_in_sec_button">
            Entrar
          </button>
        </Link>
      </StyledCheck>
    )
  }

  return (
    <>
      <h3>Registrarse</h3>
      <form onSubmit={handleSubmit(onSubmitHandler)}>
        <div className="inputContainer">
          <div className="sn-field">
            <input
              type="email"
              name="email"
              ref={register({
                required: 'Este campo es requerido',
                pattern: {
                  value: /^[\w-.]+@([\w-]+\.)+[\w-]{2,4}$/i,
                  message:
                    'Debe ser una dirección de correo electrónico valida',
                },
              })}
              maxLength="64"
              placeholder="Correo electrónico"
            />
            <i className="la la-envelope"></i>
          </div>
          {errors.email && (
            <FormErrorFeedback>{errors.email.message}</FormErrorFeedback>
          )}
        </div>

        <div className="inputContainer">
          <div className="sn-field">
            <input
              type="text"
              name="first_name"
              ref={register({
                required: 'Este campo es requerido',
                maxLength: {
                  value: 64,
                  message: 'Limite de carateres superior al permitido',
                },
              })}
              placeholder="Nombre"
            />
            <i className="la la-user"></i>
          </div>
          {errors.first_name && (
            <FormErrorFeedback>{errors.first_name.message}</FormErrorFeedback>
          )}
        </div>

        <div className="inputContainer">
          <div className="sn-field">
            <input
              type="text"
              name="last_name"
              ref={register({
                required: 'Este campo es requerido',
                maxLength: {
                  value: 64,
                  message: 'Limite de carateres superior al permitido',
                },
              })}
              placeholder="Apellido"
            />
            <i className="la la-user"></i>
          </div>
          {errors.last_name && (
            <FormErrorFeedback>{errors.last_name.message}</FormErrorFeedback>
          )}
        </div>

        <div className="inputContainer">
          <div className="sn-field">
            <input
              type="password"
              name="password"
              ref={register({
                required: 'Este campo es requerido',
                pattern: {
                  value:
                    /^(?=.*?[A-Z])(?=.*?[a-z])(?=.*?[0-9])(?=.*?[#?!@$^x%x*-]).{6,16}$/i,
                  message:
                    'Debe contener entre 6 y 16 caracteres, incluida una letra mayúscula, un número y un carácter especial #?!@$^%*-',
                },
              })}
              title="La clave debe contener entre 6 y 16 caracteres, incluida una letra mayúscula, un número y un carácter especial #?!@$^%*-"
              placeholder="Clave"
            />
            <i className="la la-lock"></i>
          </div>
          {errors.password && (
            <FormErrorFeedback>{errors.password.message}</FormErrorFeedback>
          )}
        </div>

        <div className="inputContainer">
          <div className="sn-field">
            <input
              type="password"
              name="confirmation"
              ref={register({
                required: 'Este campo es requerido',
                validate: (v) =>
                  v === watch('password') ||
                  'Disculpe, las claves tienen que coincidir',
              })}
              placeholder="Confirme su clave"
            />
            <i className="la la-lock" />
          </div>
          {errors.confirmation && (
            <FormErrorFeedback>{errors.confirmation.message}</FormErrorFeedback>
          )}
        </div>

        <div className="d-flex flex-column" style={{ gap: '.5rem' }}>
          <label>Eres mayor de 18</label>
          <SwitchInput setValue={(value) => setIsAdult(value)} />
        </div>

        <div className="inputContainer">
          <div className="checky-sec st2">
            <div className="sn-field fgt-sec">
              <input
                type="checkbox"
                name="terms_and_conditions"
                id="terms_and_conditions"
                checked={termsChecked}
                ref={register({
                  required: 'Este campo es requerido',
                })}
                value="1"
                readOnly
              />
              <label
                htmlFor="terms_and_conditions"
                onClick={() => setTermsChecked(!termsChecked)}
              >
                <span></span>
              </label>
              <small onClick={() => setTermsChecked(!termsChecked)}>
                Si, acepto los{' '}
                <a href="/terms-and-conditions">Términos y Condiciones.</a>
              </small>
            </div>
          </div>
          {errors.terms_and_conditions && (
            <FormErrorFeedback>
              {errors.terms_and_conditions.message}
            </FormErrorFeedback>
          )}
        </div>

        <div className="sn-field">
          <Recaptcha
            sitekey={site_key}
            verifyCallback={signupVerifyCallbackHandler}
            verifyCallbackName="signupVerifyCallbackHandler"
            expiredCallback={signupExpiredCallbackHandler}
            expiredCallbackName="signupExpiredCallbackHandler"
            ref={reCaptchaInstance}
            render="explicit"
            onloadCallback={handleOnRecaptchaLoad}
            hl="es"
          />
        </div>

        <button
          type="submit"
          value="submit"
          id="btn-submit"
          disabled={!isVerified}
        >
          Registrarse
        </button>
      </form>
      {isLoading && (
        <StyledSpinnerContainer>
          <Spinner />
        </StyledSpinnerContainer>
      )}
    </>
  )
}

export default Signup