password flow improvements (#2730)
* add button to skip sending reset code * add validation to reset code * comments * update test id * consistency sneak in - everything capitalized * add change password button to settings * create a modal for password change * change password modal * remove unused styles * more improvements * improve layout * change done button color * add already have a code to modal * remove unused prop * icons, auto add dash * cleanup * better appearance on android * Remove log * Improve error messages and add specificity to function names --------- Co-authored-by: Paul Frazee <pfrazee@gmail.com>
This commit is contained in:
		
							parent
							
								
									b9e00afdb1
								
							
						
					
					
						commit
						a9ab13e5a9
					
				
					 8 changed files with 448 additions and 7 deletions
				
			
		
							
								
								
									
										19
									
								
								src/lib/strings/password.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								src/lib/strings/password.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,19 @@ | ||||||
|  | // Regex for base32 string for testing reset code
 | ||||||
|  | const RESET_CODE_REGEX = /^[A-Z2-7]{5}-[A-Z2-7]{5}$/ | ||||||
|  | 
 | ||||||
|  | export function checkAndFormatResetCode(code: string): string | false { | ||||||
|  |   // Trim the reset code
 | ||||||
|  |   let fixed = code.trim().toUpperCase() | ||||||
|  | 
 | ||||||
|  |   // Add a dash if needed
 | ||||||
|  |   if (fixed.length === 10) { | ||||||
|  |     fixed = `${fixed.slice(0, 5)}-${fixed.slice(5, 10)}` | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // Check that it is a valid format
 | ||||||
|  |   if (!RESET_CODE_REGEX.test(fixed)) { | ||||||
|  |     return false | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return fixed | ||||||
|  | } | ||||||
|  | @ -171,6 +171,10 @@ export interface ChangeEmailModal { | ||||||
|   name: 'change-email' |   name: 'change-email' | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export interface ChangePasswordModal { | ||||||
|  |   name: 'change-password' | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export interface SwitchAccountModal { | export interface SwitchAccountModal { | ||||||
|   name: 'switch-account' |   name: 'switch-account' | ||||||
| } | } | ||||||
|  | @ -202,6 +206,7 @@ export type Modal = | ||||||
|   | BirthDateSettingsModal |   | BirthDateSettingsModal | ||||||
|   | VerifyEmailModal |   | VerifyEmailModal | ||||||
|   | ChangeEmailModal |   | ChangeEmailModal | ||||||
|  |   | ChangePasswordModal | ||||||
|   | SwitchAccountModal |   | SwitchAccountModal | ||||||
| 
 | 
 | ||||||
|   // Curation
 |   // Curation
 | ||||||
|  |  | ||||||
|  | @ -195,6 +195,29 @@ export const ForgotPasswordForm = ({ | ||||||
|             </Text> |             </Text> | ||||||
|           ) : undefined} |           ) : undefined} | ||||||
|         </View> |         </View> | ||||||
|  |         <View | ||||||
|  |           style={[ | ||||||
|  |             s.flexRow, | ||||||
|  |             s.alignCenter, | ||||||
|  |             s.mt20, | ||||||
|  |             s.mb20, | ||||||
|  |             pal.border, | ||||||
|  |             s.borderBottom1, | ||||||
|  |             {alignSelf: 'center', width: '90%'}, | ||||||
|  |           ]} | ||||||
|  |         /> | ||||||
|  |         <View style={[s.flexRow, s.justifyCenter]}> | ||||||
|  |           <TouchableOpacity | ||||||
|  |             testID="skipSendEmailButton" | ||||||
|  |             onPress={onEmailSent} | ||||||
|  |             accessibilityRole="button" | ||||||
|  |             accessibilityLabel={_(msg`Go to next`)} | ||||||
|  |             accessibilityHint={_(msg`Navigates to the next screen`)}> | ||||||
|  |             <Text type="xl" style={[pal.link, s.pr5]}> | ||||||
|  |               <Trans>Already have a code?</Trans> | ||||||
|  |             </Text> | ||||||
|  |           </TouchableOpacity> | ||||||
|  |         </View> | ||||||
|       </View> |       </View> | ||||||
|     </> |     </> | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  | @ -14,6 +14,7 @@ import {isNetworkError} from 'lib/strings/errors' | ||||||
| import {usePalette} from 'lib/hooks/usePalette' | import {usePalette} from 'lib/hooks/usePalette' | ||||||
| import {useTheme} from 'lib/ThemeContext' | import {useTheme} from 'lib/ThemeContext' | ||||||
| import {cleanError} from 'lib/strings/errors' | import {cleanError} from 'lib/strings/errors' | ||||||
|  | import {checkAndFormatResetCode} from 'lib/strings/password' | ||||||
| import {logger} from '#/logger' | import {logger} from '#/logger' | ||||||
| import {styles} from './styles' | import {styles} from './styles' | ||||||
| import {Trans, msg} from '@lingui/macro' | import {Trans, msg} from '@lingui/macro' | ||||||
|  | @ -46,14 +47,26 @@ export const SetNewPasswordForm = ({ | ||||||
|   const [password, setPassword] = useState<string>('') |   const [password, setPassword] = useState<string>('') | ||||||
| 
 | 
 | ||||||
|   const onPressNext = async () => { |   const onPressNext = async () => { | ||||||
|  |     // Check that the code is correct. We do this again just incase the user enters the code after their pw and we
 | ||||||
|  |     // don't get to call onBlur first
 | ||||||
|  |     const formattedCode = checkAndFormatResetCode(resetCode) | ||||||
|  |     // TODO Better password strength check
 | ||||||
|  |     if (!formattedCode || !password) { | ||||||
|  |       setError( | ||||||
|  |         _( | ||||||
|  |           msg`You have entered an invalid code. It should look like XXXXX-XXXXX.`, | ||||||
|  |         ), | ||||||
|  |       ) | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     setError('') |     setError('') | ||||||
|     setIsProcessing(true) |     setIsProcessing(true) | ||||||
| 
 | 
 | ||||||
|     try { |     try { | ||||||
|       const agent = new BskyAgent({service: serviceUrl}) |       const agent = new BskyAgent({service: serviceUrl}) | ||||||
|       const token = resetCode.replace(/\s/g, '') |  | ||||||
|       await agent.com.atproto.server.resetPassword({ |       await agent.com.atproto.server.resetPassword({ | ||||||
|         token, |         token: formattedCode, | ||||||
|         password, |         password, | ||||||
|       }) |       }) | ||||||
|       onPasswordSet() |       onPasswordSet() | ||||||
|  | @ -71,6 +84,19 @@ export const SetNewPasswordForm = ({ | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   const onBlur = () => { | ||||||
|  |     const formattedCode = checkAndFormatResetCode(resetCode) | ||||||
|  |     if (!formattedCode) { | ||||||
|  |       setError( | ||||||
|  |         _( | ||||||
|  |           msg`You have entered an invalid code. It should look like XXXXX-XXXXX.`, | ||||||
|  |         ), | ||||||
|  |       ) | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |     setResetCode(formattedCode) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <View> |       <View> | ||||||
|  | @ -100,9 +126,11 @@ export const SetNewPasswordForm = ({ | ||||||
|               autoCapitalize="none" |               autoCapitalize="none" | ||||||
|               autoCorrect={false} |               autoCorrect={false} | ||||||
|               keyboardAppearance={theme.colorScheme} |               keyboardAppearance={theme.colorScheme} | ||||||
|               autoFocus |               autoComplete="off" | ||||||
|               value={resetCode} |               value={resetCode} | ||||||
|               onChangeText={setResetCode} |               onChangeText={setResetCode} | ||||||
|  |               onFocus={() => setError('')} | ||||||
|  |               onBlur={onBlur} | ||||||
|               editable={!isProcessing} |               editable={!isProcessing} | ||||||
|               accessible={true} |               accessible={true} | ||||||
|               accessibilityLabel={_(msg`Reset code`)} |               accessibilityLabel={_(msg`Reset code`)} | ||||||
|  | @ -123,6 +151,7 @@ export const SetNewPasswordForm = ({ | ||||||
|               placeholderTextColor={pal.colors.textLight} |               placeholderTextColor={pal.colors.textLight} | ||||||
|               autoCapitalize="none" |               autoCapitalize="none" | ||||||
|               autoCorrect={false} |               autoCorrect={false} | ||||||
|  |               autoComplete="new-password" | ||||||
|               keyboardAppearance={theme.colorScheme} |               keyboardAppearance={theme.colorScheme} | ||||||
|               secureTextEntry |               secureTextEntry | ||||||
|               value={password} |               value={password} | ||||||
|  | @ -160,6 +189,7 @@ export const SetNewPasswordForm = ({ | ||||||
|           ) : ( |           ) : ( | ||||||
|             <TouchableOpacity |             <TouchableOpacity | ||||||
|               testID="setNewPasswordButton" |               testID="setNewPasswordButton" | ||||||
|  |               // Check the code before running the callback
 | ||||||
|               onPress={onPressNext} |               onPress={onPressNext} | ||||||
|               accessibilityRole="button" |               accessibilityRole="button" | ||||||
|               accessibilityLabel={_(msg`Go to next`)} |               accessibilityLabel={_(msg`Go to next`)} | ||||||
|  |  | ||||||
							
								
								
									
										336
									
								
								src/view/com/modals/ChangePassword.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										336
									
								
								src/view/com/modals/ChangePassword.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,336 @@ | ||||||
|  | import React, {useState} from 'react' | ||||||
|  | import { | ||||||
|  |   ActivityIndicator, | ||||||
|  |   SafeAreaView, | ||||||
|  |   StyleSheet, | ||||||
|  |   TouchableOpacity, | ||||||
|  |   View, | ||||||
|  | } from 'react-native' | ||||||
|  | import {ScrollView} from './util' | ||||||
|  | import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||||
|  | import {TextInput} from './util' | ||||||
|  | import {Text} from '../util/text/Text' | ||||||
|  | import {Button} from '../util/forms/Button' | ||||||
|  | import {ErrorMessage} from '../util/error/ErrorMessage' | ||||||
|  | import {s, colors} from 'lib/styles' | ||||||
|  | import {usePalette} from 'lib/hooks/usePalette' | ||||||
|  | import {isAndroid, isWeb} from 'platform/detection' | ||||||
|  | import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||||
|  | import {cleanError, isNetworkError} from 'lib/strings/errors' | ||||||
|  | import {checkAndFormatResetCode} from 'lib/strings/password' | ||||||
|  | import {Trans, msg} from '@lingui/macro' | ||||||
|  | import {useLingui} from '@lingui/react' | ||||||
|  | import {useModalControls} from '#/state/modals' | ||||||
|  | import {useSession, getAgent} from '#/state/session' | ||||||
|  | import * as EmailValidator from 'email-validator' | ||||||
|  | import {logger} from '#/logger' | ||||||
|  | 
 | ||||||
|  | enum Stages { | ||||||
|  |   RequestCode, | ||||||
|  |   ChangePassword, | ||||||
|  |   Done, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const snapPoints = isAndroid ? ['90%'] : ['45%'] | ||||||
|  | 
 | ||||||
|  | export function Component() { | ||||||
|  |   const pal = usePalette('default') | ||||||
|  |   const {currentAccount} = useSession() | ||||||
|  |   const {_} = useLingui() | ||||||
|  |   const [stage, setStage] = useState<Stages>(Stages.RequestCode) | ||||||
|  |   const [isProcessing, setIsProcessing] = useState<boolean>(false) | ||||||
|  |   const [resetCode, setResetCode] = useState<string>('') | ||||||
|  |   const [newPassword, setNewPassword] = useState<string>('') | ||||||
|  |   const [error, setError] = useState<string>('') | ||||||
|  |   const {isMobile} = useWebMediaQueries() | ||||||
|  |   const {closeModal} = useModalControls() | ||||||
|  |   const agent = getAgent() | ||||||
|  | 
 | ||||||
|  |   const onRequestCode = async () => { | ||||||
|  |     if ( | ||||||
|  |       !currentAccount?.email || | ||||||
|  |       !EmailValidator.validate(currentAccount.email) | ||||||
|  |     ) { | ||||||
|  |       return setError(_(msg`Your email appears to be invalid.`)) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     setError('') | ||||||
|  |     setIsProcessing(true) | ||||||
|  |     try { | ||||||
|  |       await agent.com.atproto.server.requestPasswordReset({ | ||||||
|  |         email: currentAccount.email, | ||||||
|  |       }) | ||||||
|  |       setStage(Stages.ChangePassword) | ||||||
|  |     } catch (e: any) { | ||||||
|  |       const errMsg = e.toString() | ||||||
|  |       logger.warn('Failed to request password reset', {error: e}) | ||||||
|  |       if (isNetworkError(e)) { | ||||||
|  |         setError( | ||||||
|  |           _( | ||||||
|  |             msg`Unable to contact your service. Please check your Internet connection.`, | ||||||
|  |           ), | ||||||
|  |         ) | ||||||
|  |       } else { | ||||||
|  |         setError(cleanError(errMsg)) | ||||||
|  |       } | ||||||
|  |     } finally { | ||||||
|  |       setIsProcessing(false) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const onChangePassword = async () => { | ||||||
|  |     const formattedCode = checkAndFormatResetCode(resetCode) | ||||||
|  |     // TODO Better password strength check
 | ||||||
|  |     if (!formattedCode || !newPassword) { | ||||||
|  |       setError( | ||||||
|  |         _( | ||||||
|  |           msg`You have entered an invalid code. It should look like XXXXX-XXXXX.`, | ||||||
|  |         ), | ||||||
|  |       ) | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     setError('') | ||||||
|  |     setIsProcessing(true) | ||||||
|  |     try { | ||||||
|  |       await agent.com.atproto.server.resetPassword({ | ||||||
|  |         token: formattedCode, | ||||||
|  |         password: newPassword, | ||||||
|  |       }) | ||||||
|  |       setStage(Stages.Done) | ||||||
|  |     } catch (e: any) { | ||||||
|  |       const errMsg = e.toString() | ||||||
|  |       logger.warn('Failed to set new password', {error: e}) | ||||||
|  |       if (isNetworkError(e)) { | ||||||
|  |         setError( | ||||||
|  |           'Unable to contact your service. Please check your Internet connection.', | ||||||
|  |         ) | ||||||
|  |       } else { | ||||||
|  |         setError(cleanError(errMsg)) | ||||||
|  |       } | ||||||
|  |     } finally { | ||||||
|  |       setIsProcessing(false) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const onBlur = () => { | ||||||
|  |     const formattedCode = checkAndFormatResetCode(resetCode) | ||||||
|  |     if (!formattedCode) { | ||||||
|  |       setError( | ||||||
|  |         _( | ||||||
|  |           msg`You have entered an invalid code. It should look like XXXXX-XXXXX.`, | ||||||
|  |         ), | ||||||
|  |       ) | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |     setResetCode(formattedCode) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <SafeAreaView style={[pal.view, s.flex1]}> | ||||||
|  |       <ScrollView | ||||||
|  |         contentContainerStyle={[ | ||||||
|  |           styles.container, | ||||||
|  |           isMobile && styles.containerMobile, | ||||||
|  |         ]} | ||||||
|  |         keyboardShouldPersistTaps="handled"> | ||||||
|  |         <View> | ||||||
|  |           <View style={styles.titleSection}> | ||||||
|  |             <Text type="title-lg" style={[pal.text, styles.title]}> | ||||||
|  |               {stage !== Stages.Done ? 'Change Password' : 'Password Changed'} | ||||||
|  |             </Text> | ||||||
|  |           </View> | ||||||
|  | 
 | ||||||
|  |           <Text type="lg" style={[pal.textLight, {marginBottom: 10}]}> | ||||||
|  |             {stage === Stages.RequestCode ? ( | ||||||
|  |               <Trans> | ||||||
|  |                 If you want to change your password, we will send you a code to | ||||||
|  |                 verify that this is your account. | ||||||
|  |               </Trans> | ||||||
|  |             ) : stage === Stages.ChangePassword ? ( | ||||||
|  |               <Trans> | ||||||
|  |                 Enter the code you received to change your password. | ||||||
|  |               </Trans> | ||||||
|  |             ) : ( | ||||||
|  |               <Trans>Your password has been changed successfully!</Trans> | ||||||
|  |             )} | ||||||
|  |           </Text> | ||||||
|  | 
 | ||||||
|  |           {stage === Stages.RequestCode && ( | ||||||
|  |             <View style={[s.flexRow, s.justifyCenter, s.mt10]}> | ||||||
|  |               <TouchableOpacity | ||||||
|  |                 testID="skipSendEmailButton" | ||||||
|  |                 onPress={() => setStage(Stages.ChangePassword)} | ||||||
|  |                 accessibilityRole="button" | ||||||
|  |                 accessibilityLabel={_(msg`Go to next`)} | ||||||
|  |                 accessibilityHint={_(msg`Navigates to the next screen`)}> | ||||||
|  |                 <Text type="xl" style={[pal.link, s.pr5]}> | ||||||
|  |                   <Trans>Already have a code?</Trans> | ||||||
|  |                 </Text> | ||||||
|  |               </TouchableOpacity> | ||||||
|  |             </View> | ||||||
|  |           )} | ||||||
|  |           {stage === Stages.ChangePassword && ( | ||||||
|  |             <View style={[pal.border, styles.group]}> | ||||||
|  |               <View style={[styles.groupContent]}> | ||||||
|  |                 <FontAwesomeIcon | ||||||
|  |                   icon="ticket" | ||||||
|  |                   style={[pal.textLight, styles.groupContentIcon]} | ||||||
|  |                 /> | ||||||
|  |                 <TextInput | ||||||
|  |                   testID="codeInput" | ||||||
|  |                   style={[pal.text, styles.textInput]} | ||||||
|  |                   placeholder="Reset code" | ||||||
|  |                   placeholderTextColor={pal.colors.textLight} | ||||||
|  |                   value={resetCode} | ||||||
|  |                   onChangeText={setResetCode} | ||||||
|  |                   onFocus={() => setError('')} | ||||||
|  |                   onBlur={onBlur} | ||||||
|  |                   accessible={true} | ||||||
|  |                   accessibilityLabel={_(msg`Reset Code`)} | ||||||
|  |                   accessibilityHint="" | ||||||
|  |                   autoCapitalize="none" | ||||||
|  |                   autoCorrect={false} | ||||||
|  |                   autoComplete="off" | ||||||
|  |                 /> | ||||||
|  |               </View> | ||||||
|  |               <View | ||||||
|  |                 style={[ | ||||||
|  |                   pal.borderDark, | ||||||
|  |                   styles.groupContent, | ||||||
|  |                   styles.groupBottom, | ||||||
|  |                 ]}> | ||||||
|  |                 <FontAwesomeIcon | ||||||
|  |                   icon="lock" | ||||||
|  |                   style={[pal.textLight, styles.groupContentIcon]} | ||||||
|  |                 /> | ||||||
|  |                 <TextInput | ||||||
|  |                   testID="codeInput" | ||||||
|  |                   style={[pal.text, styles.textInput]} | ||||||
|  |                   placeholder="New password" | ||||||
|  |                   placeholderTextColor={pal.colors.textLight} | ||||||
|  |                   onChangeText={setNewPassword} | ||||||
|  |                   secureTextEntry | ||||||
|  |                   accessible={true} | ||||||
|  |                   accessibilityLabel={_(msg`New Password`)} | ||||||
|  |                   accessibilityHint="" | ||||||
|  |                   autoCapitalize="none" | ||||||
|  |                   autoComplete="new-password" | ||||||
|  |                 /> | ||||||
|  |               </View> | ||||||
|  |             </View> | ||||||
|  |           )} | ||||||
|  |           {error ? ( | ||||||
|  |             <ErrorMessage message={error} style={styles.error} /> | ||||||
|  |           ) : undefined} | ||||||
|  |         </View> | ||||||
|  |         <View style={[styles.btnContainer]}> | ||||||
|  |           {isProcessing ? ( | ||||||
|  |             <View style={styles.btn}> | ||||||
|  |               <ActivityIndicator color="#fff" /> | ||||||
|  |             </View> | ||||||
|  |           ) : ( | ||||||
|  |             <View style={{gap: 6}}> | ||||||
|  |               {stage === Stages.RequestCode && ( | ||||||
|  |                 <Button | ||||||
|  |                   testID="requestChangeBtn" | ||||||
|  |                   type="primary" | ||||||
|  |                   onPress={onRequestCode} | ||||||
|  |                   accessibilityLabel={_(msg`Request Code`)} | ||||||
|  |                   accessibilityHint="" | ||||||
|  |                   label={_(msg`Request Code`)} | ||||||
|  |                   labelContainerStyle={{justifyContent: 'center', padding: 4}} | ||||||
|  |                   labelStyle={[s.f18]} | ||||||
|  |                 /> | ||||||
|  |               )} | ||||||
|  |               {stage === Stages.ChangePassword && ( | ||||||
|  |                 <Button | ||||||
|  |                   testID="confirmBtn" | ||||||
|  |                   type="primary" | ||||||
|  |                   onPress={onChangePassword} | ||||||
|  |                   accessibilityLabel={_(msg`Next`)} | ||||||
|  |                   accessibilityHint="" | ||||||
|  |                   label={_(msg`Next`)} | ||||||
|  |                   labelContainerStyle={{justifyContent: 'center', padding: 4}} | ||||||
|  |                   labelStyle={[s.f18]} | ||||||
|  |                 /> | ||||||
|  |               )} | ||||||
|  |               <Button | ||||||
|  |                 testID="cancelBtn" | ||||||
|  |                 type={stage !== Stages.Done ? 'default' : 'primary'} | ||||||
|  |                 onPress={() => { | ||||||
|  |                   closeModal() | ||||||
|  |                 }} | ||||||
|  |                 accessibilityLabel={ | ||||||
|  |                   stage !== Stages.Done ? _(msg`Cancel`) : _(msg`Close`) | ||||||
|  |                 } | ||||||
|  |                 accessibilityHint="" | ||||||
|  |                 label={stage !== Stages.Done ? _(msg`Cancel`) : _(msg`Close`)} | ||||||
|  |                 labelContainerStyle={{justifyContent: 'center', padding: 4}} | ||||||
|  |                 labelStyle={[s.f18]} | ||||||
|  |               /> | ||||||
|  |             </View> | ||||||
|  |           )} | ||||||
|  |         </View> | ||||||
|  |       </ScrollView> | ||||||
|  |     </SafeAreaView> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const styles = StyleSheet.create({ | ||||||
|  |   container: { | ||||||
|  |     justifyContent: 'space-between', | ||||||
|  |   }, | ||||||
|  |   containerMobile: { | ||||||
|  |     paddingHorizontal: 18, | ||||||
|  |     paddingBottom: 35, | ||||||
|  |   }, | ||||||
|  |   titleSection: { | ||||||
|  |     paddingTop: isWeb ? 0 : 4, | ||||||
|  |     paddingBottom: isWeb ? 14 : 10, | ||||||
|  |   }, | ||||||
|  |   title: { | ||||||
|  |     textAlign: 'center', | ||||||
|  |     fontWeight: '600', | ||||||
|  |     marginBottom: 5, | ||||||
|  |   }, | ||||||
|  |   error: { | ||||||
|  |     borderRadius: 6, | ||||||
|  |   }, | ||||||
|  |   textInput: { | ||||||
|  |     width: '100%', | ||||||
|  |     paddingHorizontal: 14, | ||||||
|  |     paddingVertical: 10, | ||||||
|  |     fontSize: 16, | ||||||
|  |   }, | ||||||
|  |   btn: { | ||||||
|  |     flexDirection: 'row', | ||||||
|  |     alignItems: 'center', | ||||||
|  |     justifyContent: 'center', | ||||||
|  |     borderRadius: 32, | ||||||
|  |     padding: 14, | ||||||
|  |     backgroundColor: colors.blue3, | ||||||
|  |   }, | ||||||
|  |   btnContainer: { | ||||||
|  |     paddingTop: 20, | ||||||
|  |   }, | ||||||
|  |   group: { | ||||||
|  |     borderWidth: 1, | ||||||
|  |     borderRadius: 10, | ||||||
|  |     marginVertical: 20, | ||||||
|  |   }, | ||||||
|  |   groupLabel: { | ||||||
|  |     paddingHorizontal: 20, | ||||||
|  |     paddingBottom: 5, | ||||||
|  |   }, | ||||||
|  |   groupContent: { | ||||||
|  |     flexDirection: 'row', | ||||||
|  |     alignItems: 'center', | ||||||
|  |   }, | ||||||
|  |   groupBottom: { | ||||||
|  |     borderTopWidth: 1, | ||||||
|  |   }, | ||||||
|  |   groupContentIcon: { | ||||||
|  |     marginLeft: 10, | ||||||
|  |   }, | ||||||
|  | }) | ||||||
|  | @ -36,6 +36,7 @@ import * as ModerationDetailsModal from './ModerationDetails' | ||||||
| import * as BirthDateSettingsModal from './BirthDateSettings' | import * as BirthDateSettingsModal from './BirthDateSettings' | ||||||
| import * as VerifyEmailModal from './VerifyEmail' | import * as VerifyEmailModal from './VerifyEmail' | ||||||
| import * as ChangeEmailModal from './ChangeEmail' | import * as ChangeEmailModal from './ChangeEmail' | ||||||
|  | import * as ChangePasswordModal from './ChangePassword' | ||||||
| import * as SwitchAccountModal from './SwitchAccount' | import * as SwitchAccountModal from './SwitchAccount' | ||||||
| import * as LinkWarningModal from './LinkWarning' | import * as LinkWarningModal from './LinkWarning' | ||||||
| import * as EmbedConsentModal from './EmbedConsent' | import * as EmbedConsentModal from './EmbedConsent' | ||||||
|  | @ -172,6 +173,9 @@ export function ModalsContainer() { | ||||||
|   } else if (activeModal?.name === 'change-email') { |   } else if (activeModal?.name === 'change-email') { | ||||||
|     snapPoints = ChangeEmailModal.snapPoints |     snapPoints = ChangeEmailModal.snapPoints | ||||||
|     element = <ChangeEmailModal.Component /> |     element = <ChangeEmailModal.Component /> | ||||||
|  |   } else if (activeModal?.name === 'change-password') { | ||||||
|  |     snapPoints = ChangePasswordModal.snapPoints | ||||||
|  |     element = <ChangePasswordModal.Component /> | ||||||
|   } else if (activeModal?.name === 'switch-account') { |   } else if (activeModal?.name === 'switch-account') { | ||||||
|     snapPoints = SwitchAccountModal.snapPoints |     snapPoints = SwitchAccountModal.snapPoints | ||||||
|     element = <SwitchAccountModal.Component /> |     element = <SwitchAccountModal.Component /> | ||||||
|  |  | ||||||
|  | @ -34,6 +34,7 @@ import * as ModerationDetailsModal from './ModerationDetails' | ||||||
| import * as BirthDateSettingsModal from './BirthDateSettings' | import * as BirthDateSettingsModal from './BirthDateSettings' | ||||||
| import * as VerifyEmailModal from './VerifyEmail' | import * as VerifyEmailModal from './VerifyEmail' | ||||||
| import * as ChangeEmailModal from './ChangeEmail' | import * as ChangeEmailModal from './ChangeEmail' | ||||||
|  | import * as ChangePasswordModal from './ChangePassword' | ||||||
| import * as LinkWarningModal from './LinkWarning' | import * as LinkWarningModal from './LinkWarning' | ||||||
| import * as EmbedConsentModal from './EmbedConsent' | import * as EmbedConsentModal from './EmbedConsent' | ||||||
| 
 | 
 | ||||||
|  | @ -134,6 +135,8 @@ function Modal({modal}: {modal: ModalIface}) { | ||||||
|     element = <VerifyEmailModal.Component {...modal} /> |     element = <VerifyEmailModal.Component {...modal} /> | ||||||
|   } else if (modal.name === 'change-email') { |   } else if (modal.name === 'change-email') { | ||||||
|     element = <ChangeEmailModal.Component /> |     element = <ChangeEmailModal.Component /> | ||||||
|  |   } else if (modal.name === 'change-password') { | ||||||
|  |     element = <ChangePasswordModal.Component /> | ||||||
|   } else if (modal.name === 'link-warning') { |   } else if (modal.name === 'link-warning') { | ||||||
|     element = <LinkWarningModal.Component {...modal} /> |     element = <LinkWarningModal.Component {...modal} /> | ||||||
|   } else if (modal.name === 'embed-consent') { |   } else if (modal.name === 'embed-consent') { | ||||||
|  |  | ||||||
|  | @ -647,7 +647,7 @@ export function SettingsScreen({}: Props) { | ||||||
|             /> |             /> | ||||||
|           </View> |           </View> | ||||||
|           <Text type="lg" style={pal.text}> |           <Text type="lg" style={pal.text}> | ||||||
|             <Trans>App passwords</Trans> |             <Trans>App Passwords</Trans> | ||||||
|           </Text> |           </Text> | ||||||
|         </TouchableOpacity> |         </TouchableOpacity> | ||||||
|         <TouchableOpacity |         <TouchableOpacity | ||||||
|  | @ -668,7 +668,7 @@ export function SettingsScreen({}: Props) { | ||||||
|             /> |             /> | ||||||
|           </View> |           </View> | ||||||
|           <Text type="lg" style={pal.text} numberOfLines={1}> |           <Text type="lg" style={pal.text} numberOfLines={1}> | ||||||
|             <Trans>Change handle</Trans> |             <Trans>Change Handle</Trans> | ||||||
|           </Text> |           </Text> | ||||||
|         </TouchableOpacity> |         </TouchableOpacity> | ||||||
|         {isNative && ( |         {isNative && ( | ||||||
|  | @ -684,8 +684,29 @@ export function SettingsScreen({}: Props) { | ||||||
|         )} |         )} | ||||||
|         <View style={styles.spacer20} /> |         <View style={styles.spacer20} /> | ||||||
|         <Text type="xl-bold" style={[pal.text, styles.heading]}> |         <Text type="xl-bold" style={[pal.text, styles.heading]}> | ||||||
|           <Trans>Danger Zone</Trans> |           <Trans>Account</Trans> | ||||||
|         </Text> |         </Text> | ||||||
|  |         <TouchableOpacity | ||||||
|  |           testID="changePasswordBtn" | ||||||
|  |           style={[ | ||||||
|  |             styles.linkCard, | ||||||
|  |             pal.view, | ||||||
|  |             isSwitchingAccounts && styles.dimmed, | ||||||
|  |           ]} | ||||||
|  |           onPress={() => openModal({name: 'change-password'})} | ||||||
|  |           accessibilityRole="button" | ||||||
|  |           accessibilityLabel={_(msg`Change password`)} | ||||||
|  |           accessibilityHint={_(msg`Change your Bluesky password`)}> | ||||||
|  |           <View style={[styles.iconContainer, pal.btn]}> | ||||||
|  |             <FontAwesomeIcon | ||||||
|  |               icon="lock" | ||||||
|  |               style={pal.text as FontAwesomeIconStyle} | ||||||
|  |             /> | ||||||
|  |           </View> | ||||||
|  |           <Text type="lg" style={pal.text} numberOfLines={1}> | ||||||
|  |             <Trans>Change Password</Trans> | ||||||
|  |           </Text> | ||||||
|  |         </TouchableOpacity> | ||||||
|         <TouchableOpacity |         <TouchableOpacity | ||||||
|           style={[pal.view, styles.linkCard]} |           style={[pal.view, styles.linkCard]} | ||||||
|           onPress={onPressDeleteAccount} |           onPress={onPressDeleteAccount} | ||||||
|  | @ -703,7 +724,7 @@ export function SettingsScreen({}: Props) { | ||||||
|             /> |             /> | ||||||
|           </View> |           </View> | ||||||
|           <Text type="lg" style={dangerText}> |           <Text type="lg" style={dangerText}> | ||||||
|             <Trans>Delete my account…</Trans> |             <Trans>Delete My Account…</Trans> | ||||||
|           </Text> |           </Text> | ||||||
|         </TouchableOpacity> |         </TouchableOpacity> | ||||||
|         <View style={styles.spacer20} /> |         <View style={styles.spacer20} /> | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue