import {useState} from 'react' import {View} from 'react-native' 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 {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 {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 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('') 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) try { const {status} = runEmailValidation() if (status === 'invalid') return if (!language) { setLanguageError(true) return } await init({ email, language, }) setSuccess(true) } catch (e) { const {clean, raw} = cleanError(e) if (clean) { setError(clean || _(msg`Something went wrong, please try again`)) } else { let message = _(msg`Something went wrong, please try again`) if (raw) { if (raw.startsWith('This email address is not supported')) { message = _( msg`Please enter a valid, non-temporary email address. You may need to access this email in the future.`, ) } } setError(message) } } } 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 use{' '} KWS {' '} to verify that you’re an adult. When you click "Begin" below, 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 } }