import {useState} from 'react' import {View} from 'react-native' import {XRPCError} from '@atproto/xrpc' import {msg, Trans} from '@lingui/macro' import {useLingui} from '@lingui/react' import {validate as validateEmail} from 'email-validator' import {useCleanError} from '#/lib/hooks/useCleanError' import { SupportCode, useCreateSupportLink, } from '#/lib/hooks/useCreateSupportLink' import {useGetTimeAgo} from '#/lib/hooks/useTimeAgo' import {useTLDs} from '#/lib/hooks/useTLDs' import {isEmailMaybeInvalid} from '#/lib/strings/email' import {type AppLanguage} from '#/locale/languages' import {useAgeAssuranceContext} from '#/state/ageAssurance' import {useInitAgeAssurance} from '#/state/ageAssurance/useInitAgeAssurance' import {logger} from '#/state/ageAssurance/util' import {useLanguagePrefs} from '#/state/preferences' import {useSession} from '#/state/session' import {atoms as a, useTheme, web} from '#/alf' import {Admonition} from '#/components/Admonition' import {AgeAssuranceBadge} from '#/components/ageAssurance/AgeAssuranceBadge' import {urls} from '#/components/ageAssurance/const' import {KWS_SUPPORTED_LANGS} from '#/components/ageAssurance/const' import {Button, ButtonIcon, ButtonText} from '#/components/Button' import * as Dialog from '#/components/Dialog' import {Divider} from '#/components/Divider' import * as TextField from '#/components/forms/TextField' import {ShieldCheck_Stroke2_Corner0_Rounded as Shield} from '#/components/icons/Shield' import {LanguageSelect} from '#/components/LanguageSelect' import {InlineLinkText} from '#/components/Link' import {Loader} from '#/components/Loader' import {Text} from '#/components/Typography' export {useDialogControl} from '#/components/Dialog/context' export function AgeAssuranceInitDialog({ control, }: { control: Dialog.DialogControlProps }) { const {_} = useLingui() return ( ) } function Inner() { const t = useTheme() const {_} = useLingui() const {currentAccount} = useSession() const langPrefs = useLanguagePrefs() const cleanError = useCleanError() const {close} = Dialog.useDialogContext() const {lastInitiatedAt} = useAgeAssuranceContext() const getTimeAgo = useGetTimeAgo() const tlds = useTLDs() const createSupportLink = useCreateSupportLink() const wasRecentlyInitiated = lastInitiatedAt && new Date(lastInitiatedAt).getTime() > Date.now() - 5 * 60 * 1000 // 5 minutes const [success, setSuccess] = useState(false) const [email, setEmail] = useState(currentAccount?.email || '') const [emailError, setEmailError] = useState('') const [languageError, setLanguageError] = useState(false) const [disabled, setDisabled] = useState(false) const [language, setLanguage] = useState( convertToKWSSupportedLanguage(langPrefs.appLanguage), ) const [error, setError] = useState(null) const {mutateAsync: init, isPending} = useInitAgeAssurance() const runEmailValidation = () => { if (validateEmail(email)) { setEmailError('') setDisabled(false) if (tlds && isEmailMaybeInvalid(email, tlds)) { setEmailError( _( msg`Please double-check that you have entered your email address correctly.`, ), ) return {status: 'maybe'} } return {status: 'valid'} } setEmailError(_(msg`Please enter a valid email address.`)) setDisabled(true) return {status: 'invalid'} } const onSubmit = async () => { setLanguageError(false) logger.metric('ageAssurance:initDialogSubmit', {}) try { const {status} = runEmailValidation() if (status === 'invalid') return if (!language) { setLanguageError(true) return } await init({ email, language, }) setSuccess(true) } catch (e) { let error: React.ReactNode = _( msg`Something went wrong, please try again`, ) if (e instanceof XRPCError) { if (e.error === 'InvalidEmail') { error = _( msg`Please enter a valid, non-temporary email address. You may need to access this email in the future.`, ) logger.metric('ageAssurance:initDialogError', {code: 'InvalidEmail'}) } else if (e.error === 'DidTooLong') { error = ( <> We're having issues initializing the age assurance process for your account. Please{' '} contact support {' '} for assistance. ) logger.metric('ageAssurance:initDialogError', {code: 'DidTooLong'}) } else { logger.metric('ageAssurance:initDialogError', {code: 'other'}) } } else { const {clean, raw} = cleanError(e) error = clean || raw || error logger.metric('ageAssurance:initDialogError', {code: 'other'}) } setError(error) } } return ( {success ? Success! : Verify your age} {success ? ( Please check your email inbox for further instructions. It may take a minute or two to arrive. ) : ( <> We have partnered with{' '} KWS {' '} to verify that you’re an adult. When you click "Begin" below, KWS will check if you have previously verified your age using this email address for other games/services powered by KWS technology. If not, KWS will email you instructions for verifying your age. When you’re done, you'll be brought back to continue using Bluesky. This should only take a few minutes. )} {success ? ( ) : ( <> {wasRecentlyInitiated && ( You initiated this flow already,{' '} {getTimeAgo(lastInitiatedAt, new Date(), {format: 'long'})}{' '} ago. It may take up to 5 minutes for emails to reach your inbox. Please consider waiting a few minutes before trying again. )} Your email setEmailError('')} onBlur={() => { runEmailValidation() }} returnKeyType="done" autoCapitalize="none" autoComplete="off" autoCorrect={false} onSubmitEditing={onSubmit} /> {emailError ? ( {emailError} ) : ( Use your account email address, or another real email address you control, in case KWS or Bluesky needs to contact you. )} Your preferred language { setLanguage(value) setLanguageError(false) }} items={KWS_SUPPORTED_LANGS} /> {languageError && ( Please select a language )} {error && {error}} By continuing, you agree to the{' '} KWS Terms of Use {' '} and acknowledge that KWS will store your verified status with your hashed email address in accordance with the{' '} KWS Privacy Policy . This means you won’t need to verify again the next time you use this email for other apps, games, and services powered by KWS technology. )} ) } // best-effort mapping of our languages to KWS supported languages function convertToKWSSupportedLanguage( appLanguage: string, ): string | undefined { // `${Enum}` is how you get a type of string union of the enum values (???) -sfn switch (appLanguage as `${AppLanguage}`) { // only en is supported case 'en-GB': return 'en' // pt-PT is pt (pt-BR is supported independently) case 'pt-PT': return 'pt' // only chinese (simplified) is supported, map all chinese variants case 'zh-Hans-CN': case 'zh-Hant-HK': case 'zh-Hant-TW': return 'zh-Hans' default: // try and map directly - if undefined, they will have to pick from the dropdown return KWS_SUPPORTED_LANGS.find(v => v.value === appLanguage)?.value } }