convert base login component and ChooseAccountForm
This commit is contained in:
		
							parent
							
								
									44b3a37f65
								
							
						
					
					
						commit
						f5b39f2755
					
				
					 6 changed files with 363 additions and 330 deletions
				
			
		|  | @ -154,6 +154,12 @@ export const atoms = { | ||||||
|   align_end: { |   align_end: { | ||||||
|     alignItems: 'flex-end', |     alignItems: 'flex-end', | ||||||
|   }, |   }, | ||||||
|  |   align_baseline: { | ||||||
|  |     alignItems: 'baseline', | ||||||
|  |   }, | ||||||
|  |   align_stretch: { | ||||||
|  |     alignItems: 'stretch', | ||||||
|  |   }, | ||||||
|   self_auto: { |   self_auto: { | ||||||
|     alignSelf: 'auto', |     alignSelf: 'auto', | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
							
								
								
									
										183
									
								
								src/screens/Login/ChooseAccountForm.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										183
									
								
								src/screens/Login/ChooseAccountForm.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,183 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {ScrollView, TouchableOpacity, View} from 'react-native' | ||||||
|  | import {Trans, msg} from '@lingui/macro' | ||||||
|  | import {useLingui} from '@lingui/react' | ||||||
|  | import flattenReactChildren from 'react-keyed-flatten-children' | ||||||
|  | 
 | ||||||
|  | import {useAnalytics} from 'lib/analytics/analytics' | ||||||
|  | import {UserAvatar} from '../../view/com/util/UserAvatar' | ||||||
|  | import {colors} from 'lib/styles' | ||||||
|  | import {styles} from '../../view/com/auth/login/styles' | ||||||
|  | import {useSession, useSessionApi, SessionAccount} from '#/state/session' | ||||||
|  | import {useProfileQuery} from '#/state/queries/profile' | ||||||
|  | import {useLoggedOutViewControls} from '#/state/shell/logged-out' | ||||||
|  | import * as Toast from '#/view/com/util/Toast' | ||||||
|  | import {Button} from '#/components/Button' | ||||||
|  | import {atoms as a, useTheme} from '#/alf' | ||||||
|  | import {Text} from '#/components/Typography' | ||||||
|  | import {ChevronRight_Stroke2_Corner0_Rounded as Chevron} from '#/components/icons/Chevron' | ||||||
|  | import {Check_Stroke2_Corner0_Rounded as Check} from '#/components/icons/Check' | ||||||
|  | 
 | ||||||
|  | function Group({children}: {children: React.ReactNode}) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <View | ||||||
|  |       style={[ | ||||||
|  |         a.rounded_md, | ||||||
|  |         a.overflow_hidden, | ||||||
|  |         a.border, | ||||||
|  |         t.atoms.border_contrast_low, | ||||||
|  |       ]}> | ||||||
|  |       {flattenReactChildren(children).map((child, i) => { | ||||||
|  |         return React.isValidElement(child) ? ( | ||||||
|  |           <React.Fragment key={i}> | ||||||
|  |             {i > 0 ? ( | ||||||
|  |               <View style={[a.border_b, t.atoms.border_contrast_low]} /> | ||||||
|  |             ) : null} | ||||||
|  |             {React.cloneElement(child, { | ||||||
|  |               // @ts-ignore
 | ||||||
|  |               style: { | ||||||
|  |                 borderRadius: 0, | ||||||
|  |                 borderWidth: 0, | ||||||
|  |               }, | ||||||
|  |             })} | ||||||
|  |           </React.Fragment> | ||||||
|  |         ) : null | ||||||
|  |       })} | ||||||
|  |     </View> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function AccountItem({ | ||||||
|  |   account, | ||||||
|  |   onSelect, | ||||||
|  |   isCurrentAccount, | ||||||
|  | }: { | ||||||
|  |   account: SessionAccount | ||||||
|  |   onSelect: (account: SessionAccount) => void | ||||||
|  |   isCurrentAccount: boolean | ||||||
|  | }) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   const {_} = useLingui() | ||||||
|  |   const {data: profile} = useProfileQuery({did: account.did}) | ||||||
|  | 
 | ||||||
|  |   const onPress = React.useCallback(() => { | ||||||
|  |     onSelect(account) | ||||||
|  |   }, [account, onSelect]) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <TouchableOpacity | ||||||
|  |       testID={`chooseAccountBtn-${account.handle}`} | ||||||
|  |       key={account.did} | ||||||
|  |       style={[a.flex_1]} | ||||||
|  |       onPress={onPress} | ||||||
|  |       accessibilityRole="button" | ||||||
|  |       accessibilityLabel={_(msg`Sign in as ${account.handle}`)} | ||||||
|  |       accessibilityHint={_(msg`Double tap to sign in`)}> | ||||||
|  |       <View style={[a.flex_1, a.flex_row, a.align_center, {height: 48}]}> | ||||||
|  |         <View style={a.p_md}> | ||||||
|  |           <UserAvatar avatar={profile?.avatar} size={24} /> | ||||||
|  |         </View> | ||||||
|  |         <Text style={[a.align_baseline, a.flex_1, a.flex_row, a.py_sm]}> | ||||||
|  |           <Text style={[a.font_bold]}> | ||||||
|  |             {profile?.displayName || account.handle}{' '} | ||||||
|  |           </Text> | ||||||
|  |           <Text style={[t.atoms.text_contrast_medium]}>{account.handle}</Text> | ||||||
|  |         </Text> | ||||||
|  |         {isCurrentAccount ? ( | ||||||
|  |           <Check size="sm" style={[{color: colors.green3}, a.mr_md]} /> | ||||||
|  |         ) : ( | ||||||
|  |           <Chevron size="sm" style={[t.atoms.text, a.mr_md]} /> | ||||||
|  |         )} | ||||||
|  |       </View> | ||||||
|  |     </TouchableOpacity> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | export const ChooseAccountForm = ({ | ||||||
|  |   onSelectAccount, | ||||||
|  |   onPressBack, | ||||||
|  | }: { | ||||||
|  |   onSelectAccount: (account?: SessionAccount) => void | ||||||
|  |   onPressBack: () => void | ||||||
|  | }) => { | ||||||
|  |   const {track, screen} = useAnalytics() | ||||||
|  |   const {_} = useLingui() | ||||||
|  |   const t = useTheme() | ||||||
|  |   const {accounts, currentAccount} = useSession() | ||||||
|  |   const {initSession} = useSessionApi() | ||||||
|  |   const {setShowLoggedOut} = useLoggedOutViewControls() | ||||||
|  | 
 | ||||||
|  |   React.useEffect(() => { | ||||||
|  |     screen('Choose Account') | ||||||
|  |   }, [screen]) | ||||||
|  | 
 | ||||||
|  |   const onSelect = React.useCallback( | ||||||
|  |     async (account: SessionAccount) => { | ||||||
|  |       if (account.accessJwt) { | ||||||
|  |         if (account.did === currentAccount?.did) { | ||||||
|  |           setShowLoggedOut(false) | ||||||
|  |           Toast.show(_(msg`Already signed in as @${account.handle}`)) | ||||||
|  |         } else { | ||||||
|  |           await initSession(account) | ||||||
|  |           track('Sign In', {resumedSession: true}) | ||||||
|  |           setTimeout(() => { | ||||||
|  |             Toast.show(_(msg`Signed in as @${account.handle}`)) | ||||||
|  |           }, 100) | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         onSelectAccount(account) | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     [currentAccount, track, initSession, onSelectAccount, setShowLoggedOut, _], | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <ScrollView testID="chooseAccountForm" style={styles.maxHeight}> | ||||||
|  |       <Text style={[a.mt_md, a.mb_lg, a.font_bold]}> | ||||||
|  |         <Trans>Sign in as...</Trans> | ||||||
|  |       </Text> | ||||||
|  |       <Group> | ||||||
|  |         {accounts.map(account => ( | ||||||
|  |           <AccountItem | ||||||
|  |             key={account.did} | ||||||
|  |             account={account} | ||||||
|  |             onSelect={onSelect} | ||||||
|  |             isCurrentAccount={account.did === currentAccount?.did} | ||||||
|  |           /> | ||||||
|  |         ))} | ||||||
|  |         <TouchableOpacity | ||||||
|  |           testID="chooseNewAccountBtn" | ||||||
|  |           style={[a.flex_1]} | ||||||
|  |           onPress={() => onSelectAccount(undefined)} | ||||||
|  |           accessibilityRole="button" | ||||||
|  |           accessibilityLabel={_(msg`Login to account that is not listed`)} | ||||||
|  |           accessibilityHint=""> | ||||||
|  |           <View style={[a.flex_row, a.flex_row, a.align_center, {height: 48}]}> | ||||||
|  |             <Text | ||||||
|  |               style={[ | ||||||
|  |                 a.align_baseline, | ||||||
|  |                 a.flex_1, | ||||||
|  |                 a.flex_row, | ||||||
|  |                 a.py_sm, | ||||||
|  |                 {paddingLeft: 48}, | ||||||
|  |               ]}> | ||||||
|  |               <Trans>Other account</Trans> | ||||||
|  |             </Text> | ||||||
|  |             <Chevron size="sm" style={[t.atoms.text, a.mr_md]} /> | ||||||
|  |           </View> | ||||||
|  |         </TouchableOpacity> | ||||||
|  |       </Group> | ||||||
|  |       <View style={[a.flex_row, a.mt_lg]}> | ||||||
|  |         <Button | ||||||
|  |           label={_(msg`Back`)} | ||||||
|  |           variant="solid" | ||||||
|  |           color="secondary" | ||||||
|  |           size="small" | ||||||
|  |           onPress={onPressBack}> | ||||||
|  |           <Trans>Back</Trans> | ||||||
|  |         </Button> | ||||||
|  |         <View style={[a.flex_1]} /> | ||||||
|  |       </View> | ||||||
|  |     </ScrollView> | ||||||
|  |   ) | ||||||
|  | } | ||||||
							
								
								
									
										166
									
								
								src/screens/Login/index.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								src/screens/Login/index.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,166 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {useAnalytics} from '#/lib/analytics/analytics' | ||||||
|  | import {useLingui} from '@lingui/react' | ||||||
|  | import {LoggedOutLayout} from '#/view/com/util/layouts/LoggedOutLayout' | ||||||
|  | import {SessionAccount, useSession} from '#/state/session' | ||||||
|  | import {DEFAULT_SERVICE} from '#/lib/constants' | ||||||
|  | import {useLoggedOutView} from '#/state/shell/logged-out' | ||||||
|  | import {useServiceQuery} from '#/state/queries/service' | ||||||
|  | import {msg} from '@lingui/macro' | ||||||
|  | import {logger} from '#/logger' | ||||||
|  | import {atoms as a} from '#/alf' | ||||||
|  | import {KeyboardAvoidingView} from 'react-native' | ||||||
|  | import {ChooseAccountForm} from './ChooseAccountForm' | ||||||
|  | import {ForgotPasswordForm} from '#/view/com/auth/login/ForgotPasswordForm' | ||||||
|  | import {SetNewPasswordForm} from '#/view/com/auth/login/SetNewPasswordForm' | ||||||
|  | import {PasswordUpdatedForm} from '#/view/com/auth/login/PasswordUpdatedForm' | ||||||
|  | import {LoginForm} from '#/view/com/auth/login/LoginForm' | ||||||
|  | 
 | ||||||
|  | enum Forms { | ||||||
|  |   Login, | ||||||
|  |   ChooseAccount, | ||||||
|  |   ForgotPassword, | ||||||
|  |   SetNewPassword, | ||||||
|  |   PasswordUpdated, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const Login = ({onPressBack}: {onPressBack: () => void}) => { | ||||||
|  |   const {_} = useLingui() | ||||||
|  | 
 | ||||||
|  |   const {accounts} = useSession() | ||||||
|  |   const {track} = useAnalytics() | ||||||
|  |   const {requestedAccountSwitchTo} = useLoggedOutView() | ||||||
|  |   const requestedAccount = accounts.find( | ||||||
|  |     acc => acc.did === requestedAccountSwitchTo, | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const [error, setError] = React.useState<string>('') | ||||||
|  |   const [serviceUrl, setServiceUrl] = React.useState<string>( | ||||||
|  |     requestedAccount?.service || DEFAULT_SERVICE, | ||||||
|  |   ) | ||||||
|  |   const [initialHandle, setInitialHandle] = React.useState<string>( | ||||||
|  |     requestedAccount?.handle || '', | ||||||
|  |   ) | ||||||
|  |   const [currentForm, setCurrentForm] = React.useState<Forms>( | ||||||
|  |     requestedAccount | ||||||
|  |       ? Forms.Login | ||||||
|  |       : accounts.length | ||||||
|  |       ? Forms.ChooseAccount | ||||||
|  |       : Forms.Login, | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const { | ||||||
|  |     data: serviceDescription, | ||||||
|  |     error: serviceError, | ||||||
|  |     refetch: refetchService, | ||||||
|  |   } = useServiceQuery(serviceUrl) | ||||||
|  | 
 | ||||||
|  |   const onSelectAccount = (account?: SessionAccount) => { | ||||||
|  |     if (account?.service) { | ||||||
|  |       setServiceUrl(account.service) | ||||||
|  |     } | ||||||
|  |     setInitialHandle(account?.handle || '') | ||||||
|  |     setCurrentForm(Forms.Login) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const gotoForm = (form: Forms) => () => { | ||||||
|  |     setError('') | ||||||
|  |     setCurrentForm(form) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   React.useEffect(() => { | ||||||
|  |     if (serviceError) { | ||||||
|  |       setError( | ||||||
|  |         _( | ||||||
|  |           msg`Unable to contact your service. Please check your Internet connection.`, | ||||||
|  |         ), | ||||||
|  |       ) | ||||||
|  |       logger.warn(`Failed to fetch service description for ${serviceUrl}`, { | ||||||
|  |         error: String(serviceError), | ||||||
|  |       }) | ||||||
|  |     } else { | ||||||
|  |       setError('') | ||||||
|  |     } | ||||||
|  |   }, [serviceError, serviceUrl, _]) | ||||||
|  | 
 | ||||||
|  |   const onPressRetryConnect = () => refetchService() | ||||||
|  |   const onPressForgotPassword = () => { | ||||||
|  |     track('Signin:PressedForgotPassword') | ||||||
|  |     setCurrentForm(Forms.ForgotPassword) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   let content = null | ||||||
|  |   let title = '' | ||||||
|  |   let description = '' | ||||||
|  | 
 | ||||||
|  |   switch (currentForm) { | ||||||
|  |     case Forms.Login: | ||||||
|  |       title = _(msg`Sign in`) | ||||||
|  |       description = _(msg`Enter your username and password`) | ||||||
|  |       content = ( | ||||||
|  |         <LoginForm | ||||||
|  |           error={error} | ||||||
|  |           serviceUrl={serviceUrl} | ||||||
|  |           serviceDescription={serviceDescription} | ||||||
|  |           initialHandle={initialHandle} | ||||||
|  |           setError={setError} | ||||||
|  |           setServiceUrl={setServiceUrl} | ||||||
|  |           onPressBack={onPressBack} | ||||||
|  |           onPressForgotPassword={onPressForgotPassword} | ||||||
|  |           onPressRetryConnect={onPressRetryConnect} | ||||||
|  |         /> | ||||||
|  |       ) | ||||||
|  |       break | ||||||
|  |     case Forms.ChooseAccount: | ||||||
|  |       title = _(msg`Sign in`) | ||||||
|  |       description = _(msg`Select from an existing account`) | ||||||
|  |       content = ( | ||||||
|  |         <ChooseAccountForm | ||||||
|  |           onSelectAccount={onSelectAccount} | ||||||
|  |           onPressBack={onPressBack} | ||||||
|  |         /> | ||||||
|  |       ) | ||||||
|  |       break | ||||||
|  |     case Forms.ForgotPassword: | ||||||
|  |       title = _(msg`Forgot Password`) | ||||||
|  |       description = _(msg`Let's get your password reset!`) | ||||||
|  |       content = ( | ||||||
|  |         <ForgotPasswordForm | ||||||
|  |           error={error} | ||||||
|  |           serviceUrl={serviceUrl} | ||||||
|  |           serviceDescription={serviceDescription} | ||||||
|  |           setError={setError} | ||||||
|  |           setServiceUrl={setServiceUrl} | ||||||
|  |           onPressBack={gotoForm(Forms.Login)} | ||||||
|  |           onEmailSent={gotoForm(Forms.SetNewPassword)} | ||||||
|  |         /> | ||||||
|  |       ) | ||||||
|  |       break | ||||||
|  |     case Forms.SetNewPassword: | ||||||
|  |       title = _(msg`Forgot Password`) | ||||||
|  |       description = _(msg`Let's get your password reset!`) | ||||||
|  |       content = ( | ||||||
|  |         <SetNewPasswordForm | ||||||
|  |           error={error} | ||||||
|  |           serviceUrl={serviceUrl} | ||||||
|  |           setError={setError} | ||||||
|  |           onPressBack={gotoForm(Forms.ForgotPassword)} | ||||||
|  |           onPasswordSet={gotoForm(Forms.PasswordUpdated)} | ||||||
|  |         /> | ||||||
|  |       ) | ||||||
|  |       break | ||||||
|  |     case Forms.PasswordUpdated: | ||||||
|  |       title = _(msg`Password updated`) | ||||||
|  |       description = _(msg`You can now sign in with your new password.`) | ||||||
|  |       content = <PasswordUpdatedForm onPressNext={gotoForm(Forms.Login)} /> | ||||||
|  |       break | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <KeyboardAvoidingView testID="signIn" behavior="padding" style={a.flex_1}> | ||||||
|  |       <LoggedOutLayout leadin="" title={title} description={description}> | ||||||
|  |         {content} | ||||||
|  |       </LoggedOutLayout> | ||||||
|  |     </KeyboardAvoidingView> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | @ -5,16 +5,16 @@ import {useLingui} from '@lingui/react' | ||||||
| import {Trans, msg} from '@lingui/macro' | import {Trans, msg} from '@lingui/macro' | ||||||
| import {useNavigation} from '@react-navigation/native' | import {useNavigation} from '@react-navigation/native' | ||||||
| 
 | 
 | ||||||
| import {isIOS, isNative} from 'platform/detection' | import {isIOS, isNative} from '#/platform/detection' | ||||||
| import {Login} from 'view/com/auth/login/Login' | import {Login} from '#/screens/Login' | ||||||
| import {CreateAccount} from 'view/com/auth/create/CreateAccount' | import {CreateAccount} from '#/view/com/auth/create/CreateAccount' | ||||||
| import {ErrorBoundary} from 'view/com/util/ErrorBoundary' | import {ErrorBoundary} from '#/view/com/util/ErrorBoundary' | ||||||
| import {s} from 'lib/styles' | import {s} from '#/lib/styles' | ||||||
| import {usePalette} from 'lib/hooks/usePalette' | import {usePalette} from '#/lib/hooks/usePalette' | ||||||
| import {useAnalytics} from 'lib/analytics/analytics' | import {useAnalytics} from '#/lib/analytics/analytics' | ||||||
| import {SplashScreen} from './SplashScreen' | import {SplashScreen} from './SplashScreen' | ||||||
| import {useSetMinimalShellMode} from '#/state/shell/minimal-mode' | import {useSetMinimalShellMode} from '#/state/shell/minimal-mode' | ||||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | import {useWebMediaQueries} from '#/lib/hooks/useWebMediaQueries' | ||||||
| import { | import { | ||||||
|   useLoggedOutView, |   useLoggedOutView, | ||||||
|   useLoggedOutViewControls, |   useLoggedOutViewControls, | ||||||
|  |  | ||||||
|  | @ -1,158 +0,0 @@ | ||||||
| import React from 'react' |  | ||||||
| import {ScrollView, TouchableOpacity, View} from 'react-native' |  | ||||||
| import { |  | ||||||
|   FontAwesomeIcon, |  | ||||||
|   FontAwesomeIconStyle, |  | ||||||
| } from '@fortawesome/react-native-fontawesome' |  | ||||||
| import {useAnalytics} from 'lib/analytics/analytics' |  | ||||||
| import {Text} from '../../util/text/Text' |  | ||||||
| import {UserAvatar} from '../../util/UserAvatar' |  | ||||||
| import {s, colors} from 'lib/styles' |  | ||||||
| import {usePalette} from 'lib/hooks/usePalette' |  | ||||||
| import {Trans, msg} from '@lingui/macro' |  | ||||||
| import {useLingui} from '@lingui/react' |  | ||||||
| import {styles} from './styles' |  | ||||||
| import {useSession, useSessionApi, SessionAccount} from '#/state/session' |  | ||||||
| import {useProfileQuery} from '#/state/queries/profile' |  | ||||||
| import {useLoggedOutViewControls} from '#/state/shell/logged-out' |  | ||||||
| import * as Toast from '#/view/com/util/Toast' |  | ||||||
| 
 |  | ||||||
| function AccountItem({ |  | ||||||
|   account, |  | ||||||
|   onSelect, |  | ||||||
|   isCurrentAccount, |  | ||||||
| }: { |  | ||||||
|   account: SessionAccount |  | ||||||
|   onSelect: (account: SessionAccount) => void |  | ||||||
|   isCurrentAccount: boolean |  | ||||||
| }) { |  | ||||||
|   const pal = usePalette('default') |  | ||||||
|   const {_} = useLingui() |  | ||||||
|   const {data: profile} = useProfileQuery({did: account.did}) |  | ||||||
| 
 |  | ||||||
|   const onPress = React.useCallback(() => { |  | ||||||
|     onSelect(account) |  | ||||||
|   }, [account, onSelect]) |  | ||||||
| 
 |  | ||||||
|   return ( |  | ||||||
|     <TouchableOpacity |  | ||||||
|       testID={`chooseAccountBtn-${account.handle}`} |  | ||||||
|       key={account.did} |  | ||||||
|       style={[pal.view, pal.border, styles.account]} |  | ||||||
|       onPress={onPress} |  | ||||||
|       accessibilityRole="button" |  | ||||||
|       accessibilityLabel={_(msg`Sign in as ${account.handle}`)} |  | ||||||
|       accessibilityHint={_(msg`Double tap to sign in`)}> |  | ||||||
|       <View style={[pal.borderDark, styles.groupContent, styles.noTopBorder]}> |  | ||||||
|         <View style={s.p10}> |  | ||||||
|           <UserAvatar avatar={profile?.avatar} size={30} /> |  | ||||||
|         </View> |  | ||||||
|         <Text style={styles.accountText}> |  | ||||||
|           <Text type="lg-bold" style={pal.text}> |  | ||||||
|             {profile?.displayName || account.handle}{' '} |  | ||||||
|           </Text> |  | ||||||
|           <Text type="lg" style={[pal.textLight]}> |  | ||||||
|             {account.handle} |  | ||||||
|           </Text> |  | ||||||
|         </Text> |  | ||||||
|         {isCurrentAccount ? ( |  | ||||||
|           <FontAwesomeIcon |  | ||||||
|             icon="check" |  | ||||||
|             size={16} |  | ||||||
|             style={[{color: colors.green3} as FontAwesomeIconStyle, s.mr10]} |  | ||||||
|           /> |  | ||||||
|         ) : ( |  | ||||||
|           <FontAwesomeIcon |  | ||||||
|             icon="angle-right" |  | ||||||
|             size={16} |  | ||||||
|             style={[pal.text, s.mr10]} |  | ||||||
|           /> |  | ||||||
|         )} |  | ||||||
|       </View> |  | ||||||
|     </TouchableOpacity> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
| export const ChooseAccountForm = ({ |  | ||||||
|   onSelectAccount, |  | ||||||
|   onPressBack, |  | ||||||
| }: { |  | ||||||
|   onSelectAccount: (account?: SessionAccount) => void |  | ||||||
|   onPressBack: () => void |  | ||||||
| }) => { |  | ||||||
|   const {track, screen} = useAnalytics() |  | ||||||
|   const pal = usePalette('default') |  | ||||||
|   const {_} = useLingui() |  | ||||||
|   const {accounts, currentAccount} = useSession() |  | ||||||
|   const {initSession} = useSessionApi() |  | ||||||
|   const {setShowLoggedOut} = useLoggedOutViewControls() |  | ||||||
| 
 |  | ||||||
|   React.useEffect(() => { |  | ||||||
|     screen('Choose Account') |  | ||||||
|   }, [screen]) |  | ||||||
| 
 |  | ||||||
|   const onSelect = React.useCallback( |  | ||||||
|     async (account: SessionAccount) => { |  | ||||||
|       if (account.accessJwt) { |  | ||||||
|         if (account.did === currentAccount?.did) { |  | ||||||
|           setShowLoggedOut(false) |  | ||||||
|           Toast.show(_(msg`Already signed in as @${account.handle}`)) |  | ||||||
|         } else { |  | ||||||
|           await initSession(account) |  | ||||||
|           track('Sign In', {resumedSession: true}) |  | ||||||
|           setTimeout(() => { |  | ||||||
|             Toast.show(_(msg`Signed in as @${account.handle}`)) |  | ||||||
|           }, 100) |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         onSelectAccount(account) |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     [currentAccount, track, initSession, onSelectAccount, setShowLoggedOut, _], |  | ||||||
|   ) |  | ||||||
| 
 |  | ||||||
|   return ( |  | ||||||
|     <ScrollView testID="chooseAccountForm" style={styles.maxHeight}> |  | ||||||
|       <Text |  | ||||||
|         type="2xl-medium" |  | ||||||
|         style={[pal.text, styles.groupLabel, s.mt5, s.mb10]}> |  | ||||||
|         <Trans>Sign in as...</Trans> |  | ||||||
|       </Text> |  | ||||||
|       {accounts.map(account => ( |  | ||||||
|         <AccountItem |  | ||||||
|           key={account.did} |  | ||||||
|           account={account} |  | ||||||
|           onSelect={onSelect} |  | ||||||
|           isCurrentAccount={account.did === currentAccount?.did} |  | ||||||
|         /> |  | ||||||
|       ))} |  | ||||||
|       <TouchableOpacity |  | ||||||
|         testID="chooseNewAccountBtn" |  | ||||||
|         style={[pal.view, pal.border, styles.account, styles.accountLast]} |  | ||||||
|         onPress={() => onSelectAccount(undefined)} |  | ||||||
|         accessibilityRole="button" |  | ||||||
|         accessibilityLabel={_(msg`Login to account that is not listed`)} |  | ||||||
|         accessibilityHint=""> |  | ||||||
|         <View style={[pal.borderDark, styles.groupContent, styles.noTopBorder]}> |  | ||||||
|           <Text style={[styles.accountText, styles.accountTextOther]}> |  | ||||||
|             <Text type="lg" style={pal.text}> |  | ||||||
|               <Trans>Other account</Trans> |  | ||||||
|             </Text> |  | ||||||
|           </Text> |  | ||||||
|           <FontAwesomeIcon |  | ||||||
|             icon="angle-right" |  | ||||||
|             size={16} |  | ||||||
|             style={[pal.text, s.mr10]} |  | ||||||
|           /> |  | ||||||
|         </View> |  | ||||||
|       </TouchableOpacity> |  | ||||||
|       <View style={[s.flexRow, s.alignCenter, s.pl20, s.pr20]}> |  | ||||||
|         <TouchableOpacity onPress={onPressBack} accessibilityRole="button"> |  | ||||||
|           <Text type="xl" style={[pal.link, s.pl5]}> |  | ||||||
|             <Trans>Back</Trans> |  | ||||||
|           </Text> |  | ||||||
|         </TouchableOpacity> |  | ||||||
|         <View style={s.flex1} /> |  | ||||||
|       </View> |  | ||||||
|     </ScrollView> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
|  | @ -1,164 +0,0 @@ | ||||||
| import React, {useState, useEffect} from 'react' |  | ||||||
| import {KeyboardAvoidingView} from 'react-native' |  | ||||||
| import {useAnalytics} from 'lib/analytics/analytics' |  | ||||||
| import {LoggedOutLayout} from 'view/com/util/layouts/LoggedOutLayout' |  | ||||||
| import {DEFAULT_SERVICE} from '#/lib/constants' |  | ||||||
| import {usePalette} from 'lib/hooks/usePalette' |  | ||||||
| import {logger} from '#/logger' |  | ||||||
| import {ChooseAccountForm} from './ChooseAccountForm' |  | ||||||
| import {LoginForm} from './LoginForm' |  | ||||||
| import {ForgotPasswordForm} from './ForgotPasswordForm' |  | ||||||
| import {SetNewPasswordForm} from './SetNewPasswordForm' |  | ||||||
| import {PasswordUpdatedForm} from './PasswordUpdatedForm' |  | ||||||
| import {useLingui} from '@lingui/react' |  | ||||||
| import {msg} from '@lingui/macro' |  | ||||||
| import {useSession, SessionAccount} from '#/state/session' |  | ||||||
| import {useServiceQuery} from '#/state/queries/service' |  | ||||||
| import {useLoggedOutView} from '#/state/shell/logged-out' |  | ||||||
| 
 |  | ||||||
| enum Forms { |  | ||||||
|   Login, |  | ||||||
|   ChooseAccount, |  | ||||||
|   ForgotPassword, |  | ||||||
|   SetNewPassword, |  | ||||||
|   PasswordUpdated, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export const Login = ({onPressBack}: {onPressBack: () => void}) => { |  | ||||||
|   const {_} = useLingui() |  | ||||||
|   const pal = usePalette('default') |  | ||||||
| 
 |  | ||||||
|   const {accounts} = useSession() |  | ||||||
|   const {track} = useAnalytics() |  | ||||||
|   const {requestedAccountSwitchTo} = useLoggedOutView() |  | ||||||
|   const requestedAccount = accounts.find( |  | ||||||
|     a => a.did === requestedAccountSwitchTo, |  | ||||||
|   ) |  | ||||||
| 
 |  | ||||||
|   const [error, setError] = useState<string>('') |  | ||||||
|   const [serviceUrl, setServiceUrl] = useState<string>( |  | ||||||
|     requestedAccount?.service || DEFAULT_SERVICE, |  | ||||||
|   ) |  | ||||||
|   const [initialHandle, setInitialHandle] = useState<string>( |  | ||||||
|     requestedAccount?.handle || '', |  | ||||||
|   ) |  | ||||||
|   const [currentForm, setCurrentForm] = useState<Forms>( |  | ||||||
|     requestedAccount |  | ||||||
|       ? Forms.Login |  | ||||||
|       : accounts.length |  | ||||||
|       ? Forms.ChooseAccount |  | ||||||
|       : Forms.Login, |  | ||||||
|   ) |  | ||||||
| 
 |  | ||||||
|   const { |  | ||||||
|     data: serviceDescription, |  | ||||||
|     error: serviceError, |  | ||||||
|     refetch: refetchService, |  | ||||||
|   } = useServiceQuery(serviceUrl) |  | ||||||
| 
 |  | ||||||
|   const onSelectAccount = (account?: SessionAccount) => { |  | ||||||
|     if (account?.service) { |  | ||||||
|       setServiceUrl(account.service) |  | ||||||
|     } |  | ||||||
|     setInitialHandle(account?.handle || '') |  | ||||||
|     setCurrentForm(Forms.Login) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   const gotoForm = (form: Forms) => () => { |  | ||||||
|     setError('') |  | ||||||
|     setCurrentForm(form) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   useEffect(() => { |  | ||||||
|     if (serviceError) { |  | ||||||
|       setError( |  | ||||||
|         _( |  | ||||||
|           msg`Unable to contact your service. Please check your Internet connection.`, |  | ||||||
|         ), |  | ||||||
|       ) |  | ||||||
|       logger.warn(`Failed to fetch service description for ${serviceUrl}`, { |  | ||||||
|         error: String(serviceError), |  | ||||||
|       }) |  | ||||||
|     } else { |  | ||||||
|       setError('') |  | ||||||
|     } |  | ||||||
|   }, [serviceError, serviceUrl, _]) |  | ||||||
| 
 |  | ||||||
|   const onPressRetryConnect = () => refetchService() |  | ||||||
|   const onPressForgotPassword = () => { |  | ||||||
|     track('Signin:PressedForgotPassword') |  | ||||||
|     setCurrentForm(Forms.ForgotPassword) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   return ( |  | ||||||
|     <KeyboardAvoidingView testID="signIn" behavior="padding" style={pal.view}> |  | ||||||
|       {currentForm === Forms.Login ? ( |  | ||||||
|         <LoggedOutLayout |  | ||||||
|           leadin="" |  | ||||||
|           title={_(msg`Sign in`)} |  | ||||||
|           description={_(msg`Enter your username and password`)}> |  | ||||||
|           <LoginForm |  | ||||||
|             error={error} |  | ||||||
|             serviceUrl={serviceUrl} |  | ||||||
|             serviceDescription={serviceDescription} |  | ||||||
|             initialHandle={initialHandle} |  | ||||||
|             setError={setError} |  | ||||||
|             setServiceUrl={setServiceUrl} |  | ||||||
|             onPressBack={onPressBack} |  | ||||||
|             onPressForgotPassword={onPressForgotPassword} |  | ||||||
|             onPressRetryConnect={onPressRetryConnect} |  | ||||||
|           /> |  | ||||||
|         </LoggedOutLayout> |  | ||||||
|       ) : undefined} |  | ||||||
|       {currentForm === Forms.ChooseAccount ? ( |  | ||||||
|         <LoggedOutLayout |  | ||||||
|           leadin="" |  | ||||||
|           title={_(msg`Sign in as...`)} |  | ||||||
|           description={_(msg`Select from an existing account`)}> |  | ||||||
|           <ChooseAccountForm |  | ||||||
|             onSelectAccount={onSelectAccount} |  | ||||||
|             onPressBack={onPressBack} |  | ||||||
|           /> |  | ||||||
|         </LoggedOutLayout> |  | ||||||
|       ) : undefined} |  | ||||||
|       {currentForm === Forms.ForgotPassword ? ( |  | ||||||
|         <LoggedOutLayout |  | ||||||
|           leadin="" |  | ||||||
|           title={_(msg`Forgot Password`)} |  | ||||||
|           description={_(msg`Let's get your password reset!`)}> |  | ||||||
|           <ForgotPasswordForm |  | ||||||
|             error={error} |  | ||||||
|             serviceUrl={serviceUrl} |  | ||||||
|             serviceDescription={serviceDescription} |  | ||||||
|             setError={setError} |  | ||||||
|             setServiceUrl={setServiceUrl} |  | ||||||
|             onPressBack={gotoForm(Forms.Login)} |  | ||||||
|             onEmailSent={gotoForm(Forms.SetNewPassword)} |  | ||||||
|           /> |  | ||||||
|         </LoggedOutLayout> |  | ||||||
|       ) : undefined} |  | ||||||
|       {currentForm === Forms.SetNewPassword ? ( |  | ||||||
|         <LoggedOutLayout |  | ||||||
|           leadin="" |  | ||||||
|           title={_(msg`Forgot Password`)} |  | ||||||
|           description={_(msg`Let's get your password reset!`)}> |  | ||||||
|           <SetNewPasswordForm |  | ||||||
|             error={error} |  | ||||||
|             serviceUrl={serviceUrl} |  | ||||||
|             setError={setError} |  | ||||||
|             onPressBack={gotoForm(Forms.ForgotPassword)} |  | ||||||
|             onPasswordSet={gotoForm(Forms.PasswordUpdated)} |  | ||||||
|           /> |  | ||||||
|         </LoggedOutLayout> |  | ||||||
|       ) : undefined} |  | ||||||
|       {currentForm === Forms.PasswordUpdated ? ( |  | ||||||
|         <LoggedOutLayout |  | ||||||
|           leadin="" |  | ||||||
|           title={_(msg`Password updated`)} |  | ||||||
|           description={_(msg`You can now sign in with your new password.`)}> |  | ||||||
|           <PasswordUpdatedForm onPressNext={gotoForm(Forms.Login)} /> |  | ||||||
|         </LoggedOutLayout> |  | ||||||
|       ) : undefined} |  | ||||||
|     </KeyboardAvoidingView> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue