import {useState} from 'react'
import {View} from 'react-native'
import Animated, {
FadeIn,
FadeOut,
LayoutAnimationConfig,
LinearTransition,
} from 'react-native-reanimated'
import {msg, Plural, Trans} from '@lingui/macro'
import {useLingui} from '@lingui/react'
import {useGate} from '#/lib/statsig/statsig'
import {
createFullHandle,
MAX_SERVICE_HANDLE_LENGTH,
validateServiceHandle,
} from '#/lib/strings/handles'
import {logger} from '#/logger'
import {
checkHandleAvailability,
useHandleAvailabilityQuery,
} from '#/state/queries/handle-availability'
import {ScreenTransition} from '#/screens/Login/ScreenTransition'
import {useSignupContext} from '#/screens/Signup/state'
import {atoms as a, native, useTheme} from '#/alf'
import * as TextField from '#/components/forms/TextField'
import {useThrottledValue} from '#/components/hooks/useThrottledValue'
import {At_Stroke2_Corner0_Rounded as AtIcon} from '#/components/icons/At'
import {Check_Stroke2_Corner0_Rounded as CheckIcon} from '#/components/icons/Check'
import {Text} from '#/components/Typography'
import {IS_INTERNAL} from '#/env'
import {BackNextButtons} from '../BackNextButtons'
import {HandleSuggestions} from './HandleSuggestions'
export function StepHandle() {
const {_} = useLingui()
const t = useTheme()
const gate = useGate()
const {state, dispatch} = useSignupContext()
const [draftValue, setDraftValue] = useState(state.handle)
const isNextLoading = useThrottledValue(state.isLoading, 500)
const validCheck = validateServiceHandle(draftValue, state.userDomain)
const {
debouncedUsername: debouncedDraftValue,
enabled: queryEnabled,
query: {data: isHandleAvailable, isPending},
} = useHandleAvailabilityQuery({
username: draftValue,
serviceDid: state.serviceDescription?.did ?? 'UNKNOWN',
serviceDomain: state.userDomain,
birthDate: state.dateOfBirth.toISOString(),
email: state.email,
enabled: validCheck.overall,
})
const onNextPress = async () => {
const handle = draftValue.trim()
dispatch({
type: 'setHandle',
value: handle,
})
if (!validCheck.overall) {
return
}
dispatch({type: 'setIsLoading', value: true})
try {
const {available: handleAvailable} = await checkHandleAvailability(
createFullHandle(handle, state.userDomain),
state.serviceDescription?.did ?? 'UNKNOWN',
{typeahead: false},
)
if (!handleAvailable) {
dispatch({
type: 'setError',
value: _(msg`That username is already taken`),
field: 'handle',
})
return
}
} catch (error) {
logger.error('Failed to check handle availability on next press', {
safeMessage: error,
})
// do nothing on error, let them pass
} finally {
dispatch({type: 'setIsLoading', value: false})
}
logger.metric(
'signup:nextPressed',
{
activeStep: state.activeStep,
phoneVerificationRequired:
state.serviceDescription?.phoneVerificationRequired,
},
{statsig: true},
)
// phoneVerificationRequired is actually whether a captcha is required
if (!state.serviceDescription?.phoneVerificationRequired) {
dispatch({
type: 'submit',
task: {verificationCode: undefined, mutableProcessed: false},
})
return
}
dispatch({type: 'next'})
}
const onBackPress = () => {
const handle = draftValue.trim()
dispatch({
type: 'setHandle',
value: handle,
})
dispatch({type: 'prev'})
logger.metric(
'signup:backPressed',
{activeStep: state.activeStep},
{statsig: true},
)
}
const hasDebounceSettled = draftValue === debouncedDraftValue
const isHandleTaken =
!isPending &&
queryEnabled &&
isHandleAvailable &&
!isHandleAvailable.available
const isNotReady = isPending || !hasDebounceSettled
const isNextDisabled =
!validCheck.overall || !!state.error || isNotReady ? true : isHandleTaken
const textFieldInvalid =
isHandleTaken ||
!validCheck.frontLengthNotTooLong ||
!validCheck.handleChars ||
!validCheck.hyphenStartOrEnd ||
!validCheck.totalLength
return (
{
if (state.error) {
dispatch({type: 'setError', value: ''})
}
setDraftValue(val.toLocaleLowerCase())
}}
label={state.userDomain}
value={draftValue}
keyboardType="ascii-capable" // fix for iOS replacing -- with —
autoCapitalize="none"
autoCorrect={false}
autoFocus
autoComplete="off"
/>
{draftValue.length > 0 && (
{draftValue}
)}
{isHandleAvailable?.available && (
)}
{state.error && (
{state.error}
)}
{isHandleTaken && validCheck.overall && (
<>
{createFullHandle(draftValue, state.userDomain)} is not
available
{isHandleAvailable.suggestions &&
isHandleAvailable.suggestions.length > 0 &&
(gate('handle_suggestions') || IS_INTERNAL) && (
{
setDraftValue(
suggestion.handle.slice(
0,
state.userDomain.length * -1,
),
)
logger.metric('signup:handleSuggestionSelected', {
method: suggestion.method,
})
}}
/>
)}
>
)}
{(!validCheck.handleChars || !validCheck.hyphenStartOrEnd) && (
{!validCheck.hyphenStartOrEnd ? (
Username cannot begin or end with a hyphen
) : (
Username must only contain letters (a-z), numbers, and
hyphens
)}
)}
{(!validCheck.frontLengthNotTooLong ||
!validCheck.totalLength) && (
Username cannot be longer than{' '}
)}
)
}
function Requirement({children}: {children: React.ReactNode}) {
return (
{children}
)
}
function RequirementText({children}: {children: React.ReactNode}) {
const t = useTheme()
return (
{children}
)
}