2022-03-21 11:09:01 -07:00
import PropTypes from 'prop-types' ;
import * as React from 'react' ;
import { defineMessages , useIntl } from 'react-intl' ;
import { useDispatch } from 'react-redux' ;
import snackbar from 'soapbox/actions/snackbar' ;
import { confirmEmailVerification } from 'soapbox/actions/verification' ;
import { Icon , Spinner , Stack , Text } from 'soapbox/components/ui' ;
const Statuses = {
IDLE : 'IDLE' ,
SUCCESS : 'SUCCESS' ,
GENERIC _FAIL : 'GENERIC_FAIL' ,
TOKEN _NOT _FOUND : 'TOKEN_NOT_FOUND' ,
TOKEN _EXPIRED : 'TOKEN_EXPIRED' ,
} ;
const messages = defineMessages ( {
emailConfirmedHeading : { id : 'email_passthru.confirmed.heading' , defaultMessage : 'Email Confirmed!' } ,
emailConfirmedBody : { id : 'email_passthru.confirmed.body' , defaultMessage : 'Close this tab and continue the registration process on the {bold} from which you sent this email confirmation.' } ,
genericFailHeading : { id : 'email_passthru.generic_fail.heading' , defaultMessage : 'Something Went Wrong' } ,
genericFailBody : { id : 'email_passthru.generic_fail.body' , defaultMessage : 'Please request a new email confirmation.' } ,
tokenNotFoundHeading : { id : 'email_passthru.token_not_found.heading' , defaultMessage : 'Invalid Token' } ,
tokenNotFoundBody : { id : 'email_passthru.token_not_found.body' , defaultMessage : 'Your email token was not found. Please request a new email confirmation from the {bold} from which you sent this email confirmation.' } ,
tokenExpiredHeading : { id : 'email_passthru.token_expired.heading' , defaultMessage : 'Token Expired' } ,
tokenExpiredBody : { id : 'email_passthru.token_expired.body' , defaultMessage : 'Your email token has expired. Please request a new email confirmation from the {bold} from which you sent this email confirmation.' } ,
} ) ;
const Success = ( ) => {
const intl = useIntl ( ) ;
return (
< Stack space = { 4 } alignItems = 'center' >
< Icon src = { require ( '@tabler/icons/icons/circle-check.svg' ) } className = 'text-primary-600 h-10 w-10' / >
< Text size = '3xl' weight = 'semibold' align = 'center' >
{ intl . formatMessage ( messages . emailConfirmedHeading ) }
< / T e x t >
< Text theme = 'muted' align = 'center' >
{ intl . formatMessage ( messages . emailConfirmedBody , { bold : < Text tag = 'span' weight = 'medium' > same device < / T e x t > } ) }
< / T e x t >
< / S t a c k >
) ;
} ;
const GenericFail = ( ) => {
const intl = useIntl ( ) ;
return (
< Stack space = { 4 } alignItems = 'center' >
< Icon src = { require ( '@tabler/icons/icons/circle-x.svg' ) } className = 'text-danger-600 h-10 w-10' / >
< Text size = '3xl' weight = 'semibold' align = 'center' >
{ intl . formatMessage ( messages . genericFailHeading ) }
< / T e x t >
< Text theme = 'muted' align = 'center' >
{ intl . formatMessage ( messages . genericFailBody ) }
< / T e x t >
< / S t a c k >
) ;
} ;
const TokenNotFound = ( ) => {
const intl = useIntl ( ) ;
return (
< Stack space = { 4 } alignItems = 'center' >
< Icon src = { require ( '@tabler/icons/icons/circle-x.svg' ) } className = 'text-danger-600 h-10 w-10' / >
< Text size = '3xl' weight = 'semibold' align = 'center' >
{ intl . formatMessage ( messages . tokenNotFoundHeading ) }
< / T e x t >
< Text theme = 'muted' align = 'center' >
{ intl . formatMessage ( messages . tokenNotFoundBody , { bold : < Text tag = 'span' weight = 'medium' > same device < / T e x t > } ) }
< / T e x t >
< / S t a c k >
) ;
} ;
const TokenExpired = ( ) => {
const intl = useIntl ( ) ;
return (
< Stack space = { 4 } alignItems = 'center' >
< Icon src = { require ( ' @ tabler / icons / icons / circle - x . svg ')} className=' text - danger - 600 h - 10 w - 10 ' / >
< Text size = '3xl' weight = 'semibold' align = 'center' >
{ intl . formatMessage ( messages . tokenExpiredHeading ) }
< / T e x t >
< Text theme = 'muted' align = 'center' >
{ intl . formatMessage ( messages . tokenExpiredBody , { bold : < Text tag = 'span' weight = 'medium' > same device < / T e x t > } ) }
< / T e x t >
< / S t a c k >
) ;
} ;
const EmailPassThru = ( { match } ) => {
const { token } = match . params ;
const dispatch = useDispatch ( ) ;
2022-03-22 05:42:26 -07:00
const intl = useIntl ( ) ;
2022-03-21 11:09:01 -07:00
const [ status , setStatus ] = React . useState ( Statuses . IDLE ) ;
React . useEffect ( ( ) => {
if ( token ) {
dispatch ( confirmEmailVerification ( token ) )
. then ( ( ) => {
setStatus ( Statuses . SUCCESS ) ;
dispatch ( snackbar . success ( intl . formatMessage ( { id : 'email_passthru.success' , defaultMessage : 'Your email has been verified!' } ) ) ) ;
} )
. catch ( ( error ) => {
const errorKey = error ? . response ? . data ? . error ;
let message = intl . formatMessage ( {
id : 'email_passthru.fail.generic' ,
defaultMessage : 'Unable to confirm your email' ,
} ) ;
if ( errorKey ) {
switch ( errorKey ) {
case 'token_expired' :
message = intl . formatMessage ( {
id : 'email_passthru.fail.expired' ,
defaultMessage : 'Your email token has expired.' ,
} ) ;
setStatus ( Statuses . TOKEN _EXPIRED ) ;
break ;
case 'token_not_found' :
message = intl . formatMessage ( {
id : 'email_passthru.fail.not_found' ,
defaultMessage : 'Your email token is invalid.' ,
} ) ;
message = 'Your token is invalid' ;
setStatus ( Statuses . TOKEN _NOT _FOUND ) ;
break ;
default :
setStatus ( Statuses . GENERIC _FAIL ) ;
break ;
}
}
dispatch ( snackbar . error ( message ) ) ;
} ) ;
}
} , [ token ] ) ;
switch ( status ) {
case Statuses . SUCCESS :
return < Success / > ;
case Statuses . TOKEN _EXPIRED :
return < TokenExpired / > ;
case Statuses . TOKEN _NOT _FOUND :
return < TokenNotFound / > ;
case Statuses . GENERIC _FAIL :
return < GenericFail / > ;
default :
return < Spinner / > ;
}
} ;
EmailPassThru . propTypes = {
match : PropTypes . object ,
} ;
2022-03-22 05:42:26 -07:00
export default EmailPassThru ;