Refactor app passwords to use react-query (#1932)
This commit is contained in:
		
							parent
							
								
									310a7eaca7
								
							
						
					
					
						commit
						9f7a162a96
					
				
					 6 changed files with 177 additions and 115 deletions
				
			
		|  | @ -3,7 +3,6 @@ import {StyleSheet, TextInput, View, TouchableOpacity} from 'react-native' | |||
| import {Text} from '../util/text/Text' | ||||
| import {Button} from '../util/forms/Button' | ||||
| import {s} from 'lib/styles' | ||||
| import {useStores} from 'state/index' | ||||
| import {usePalette} from 'lib/hooks/usePalette' | ||||
| import {isNative} from 'platform/detection' | ||||
| import { | ||||
|  | @ -16,6 +15,10 @@ import {logger} from '#/logger' | |||
| import {Trans, msg} from '@lingui/macro' | ||||
| import {useLingui} from '@lingui/react' | ||||
| import {useModalControls} from '#/state/modals' | ||||
| import { | ||||
|   useAppPasswordsQuery, | ||||
|   useAppPasswordCreateMutation, | ||||
| } from '#/state/queries/app-passwords' | ||||
| 
 | ||||
| export const snapPoints = ['70%'] | ||||
| 
 | ||||
|  | @ -56,9 +59,10 @@ const shadesOfBlue: string[] = [ | |||
| 
 | ||||
| export function Component({}: {}) { | ||||
|   const pal = usePalette('default') | ||||
|   const store = useStores() | ||||
|   const {_} = useLingui() | ||||
|   const {closeModal} = useModalControls() | ||||
|   const {data: passwords} = useAppPasswordsQuery() | ||||
|   const createMutation = useAppPasswordCreateMutation() | ||||
|   const [name, setName] = useState( | ||||
|     shadesOfBlue[Math.floor(Math.random() * shadesOfBlue.length)], | ||||
|   ) | ||||
|  | @ -82,25 +86,34 @@ export function Component({}: {}) { | |||
|     if (!name || !name.trim()) { | ||||
|       Toast.show( | ||||
|         'Please enter a name for your app password. All spaces is not allowed.', | ||||
|         'times', | ||||
|       ) | ||||
|       return | ||||
|     } | ||||
|     // if name is too short (under 4 chars), we don't allow it
 | ||||
|     if (name.length < 4) { | ||||
|       Toast.show('App Password names must be at least 4 characters long.') | ||||
|       Toast.show( | ||||
|         'App Password names must be at least 4 characters long.', | ||||
|         'times', | ||||
|       ) | ||||
|       return | ||||
|     } | ||||
| 
 | ||||
|     if (passwords?.find(p => p.name === name)) { | ||||
|       Toast.show('This name is already in use', 'times') | ||||
|       return | ||||
|     } | ||||
| 
 | ||||
|     try { | ||||
|       const newPassword = await store.me.createAppPassword(name) | ||||
|       const newPassword = await createMutation.mutateAsync({name}) | ||||
|       if (newPassword) { | ||||
|         setAppPassword(newPassword.password) | ||||
|       } else { | ||||
|         Toast.show('Failed to create app password.') | ||||
|         Toast.show('Failed to create app password.', 'times') | ||||
|         // TODO: better error handling (?)
 | ||||
|       } | ||||
|     } catch (e) { | ||||
|       Toast.show('Failed to create app password.') | ||||
|       Toast.show('Failed to create app password.', 'times') | ||||
|       logger.error('Failed to create app password', {error: e}) | ||||
|     } | ||||
|   } | ||||
|  |  | |||
|  | @ -1,6 +1,7 @@ | |||
| import RootSiblings from 'react-native-root-siblings' | ||||
| import React from 'react' | ||||
| import {Animated, StyleSheet, View} from 'react-native' | ||||
| import {Props as FontAwesomeProps} from '@fortawesome/react-native-fontawesome' | ||||
| import {Text} from './text/Text' | ||||
| import {colors} from 'lib/styles' | ||||
| import {useTheme} from 'lib/ThemeContext' | ||||
|  | @ -9,7 +10,10 @@ import {useAnimatedValue} from 'lib/hooks/useAnimatedValue' | |||
| 
 | ||||
| const TIMEOUT = 4e3 | ||||
| 
 | ||||
| export function show(message: string) { | ||||
| export function show( | ||||
|   message: string, | ||||
|   _icon: FontAwesomeProps['icon'] = 'check', | ||||
| ) { | ||||
|   const item = new RootSiblings(<Toast message={message} />) | ||||
|   setTimeout(() => { | ||||
|     item.destroy() | ||||
|  |  | |||
|  | @ -7,12 +7,14 @@ import {StyleSheet, Text, View} from 'react-native' | |||
| import { | ||||
|   FontAwesomeIcon, | ||||
|   FontAwesomeIconStyle, | ||||
|   Props as FontAwesomeProps, | ||||
| } from '@fortawesome/react-native-fontawesome' | ||||
| 
 | ||||
| const DURATION = 3500 | ||||
| 
 | ||||
| interface ActiveToast { | ||||
|   text: string | ||||
|   icon: FontAwesomeProps['icon'] | ||||
| } | ||||
| type GlobalSetActiveToast = (_activeToast: ActiveToast | undefined) => void | ||||
| 
 | ||||
|  | @ -36,7 +38,7 @@ export const ToastContainer: React.FC<ToastContainerProps> = ({}) => { | |||
|       {activeToast && ( | ||||
|         <View style={styles.container}> | ||||
|           <FontAwesomeIcon | ||||
|             icon="check" | ||||
|             icon={activeToast.icon} | ||||
|             size={24} | ||||
|             style={styles.icon as FontAwesomeIconStyle} | ||||
|           /> | ||||
|  | @ -49,11 +51,12 @@ export const ToastContainer: React.FC<ToastContainerProps> = ({}) => { | |||
| 
 | ||||
| // methods
 | ||||
| // =
 | ||||
| export function show(text: string) { | ||||
| 
 | ||||
| export function show(text: string, icon: FontAwesomeProps['icon'] = 'check') { | ||||
|   if (toastTimeout) { | ||||
|     clearTimeout(toastTimeout) | ||||
|   } | ||||
|   globalSetActiveToast?.({text}) | ||||
|   globalSetActiveToast?.({text, icon}) | ||||
|   toastTimeout = setTimeout(() => { | ||||
|     globalSetActiveToast?.(undefined) | ||||
|   }, DURATION) | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue