Merge branch 'bluesky-social:main' into patch-3
This commit is contained in:
		
						commit
						f3db23a3b1
					
				
					 24 changed files with 1134 additions and 123 deletions
				
			
		|  | @ -11,6 +11,17 @@ const DARK_SPLASH_CONFIG = { | ||||||
|   resizeMode: 'cover', |   resizeMode: 'cover', | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const SPLASH_CONFIG_ANDROID = { | ||||||
|  |   backgroundColor: '#0c7cff', | ||||||
|  |   image: './assets/splash.png', | ||||||
|  |   resizeMode: 'cover', | ||||||
|  | } | ||||||
|  | const DARK_SPLASH_CONFIG_ANDROID = { | ||||||
|  |   backgroundColor: '#0f141b', | ||||||
|  |   image: './assets/splash-dark.png', | ||||||
|  |   resizeMode: 'cover', | ||||||
|  | } | ||||||
|  | 
 | ||||||
| module.exports = function (config) { | module.exports = function (config) { | ||||||
|   /** |   /** | ||||||
|    * App version number. Should be incremented as part of a release cycle. |    * App version number. Should be incremented as part of a release cycle. | ||||||
|  | @ -70,8 +81,8 @@ module.exports = function (config) { | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|       androidStatusBar: { |       androidStatusBar: { | ||||||
|         barStyle: 'dark-content', |         barStyle: 'light-content', | ||||||
|         backgroundColor: '#ffffff', |         backgroundColor: '#00000000', | ||||||
|       }, |       }, | ||||||
|       android: { |       android: { | ||||||
|         icon: './assets/icon.png', |         icon: './assets/icon.png', | ||||||
|  | @ -101,8 +112,8 @@ module.exports = function (config) { | ||||||
|           }, |           }, | ||||||
|         ], |         ], | ||||||
|         splash: { |         splash: { | ||||||
|           ...SPLASH_CONFIG, |           ...SPLASH_CONFIG_ANDROID, | ||||||
|           dark: DARK_SPLASH_CONFIG, |           dark: DARK_SPLASH_CONFIG_ANDROID, | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|       web: { |       web: { | ||||||
|  | @ -121,12 +132,14 @@ module.exports = function (config) { | ||||||
|           { |           { | ||||||
|             ios: { |             ios: { | ||||||
|               deploymentTarget: '13.4', |               deploymentTarget: '13.4', | ||||||
|  |               newArchEnabled: false, | ||||||
|             }, |             }, | ||||||
|             android: { |             android: { | ||||||
|               compileSdkVersion: 34, |               compileSdkVersion: 34, | ||||||
|               targetSdkVersion: 34, |               targetSdkVersion: 34, | ||||||
|               buildToolsVersion: '34.0.0', |               buildToolsVersion: '34.0.0', | ||||||
|               kotlinVersion: '1.8.0', |               kotlinVersion: '1.8.0', | ||||||
|  |               newArchEnabled: false, | ||||||
|             }, |             }, | ||||||
|           }, |           }, | ||||||
|         ], |         ], | ||||||
|  | @ -144,6 +157,7 @@ module.exports = function (config) { | ||||||
|           }, |           }, | ||||||
|         ], |         ], | ||||||
|         './plugins/withAndroidManifestPlugin.js', |         './plugins/withAndroidManifestPlugin.js', | ||||||
|  |         './plugins/withAndroidStylesWindowBackgroundPlugin.js', | ||||||
|         './plugins/shareExtension/withShareExtensions.js', |         './plugins/shareExtension/withShareExtensions.js', | ||||||
|       ].filter(Boolean), |       ].filter(Boolean), | ||||||
|       extra: { |       extra: { | ||||||
|  |  | ||||||
|  | @ -213,6 +213,7 @@ | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /* NativeDropdown component */ |     /* NativeDropdown component */ | ||||||
|  |     .radix-dropdown-item:focus, | ||||||
|     .nativeDropdown-item:focus { |     .nativeDropdown-item:focus { | ||||||
|       outline: none; |       outline: none; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -58,6 +58,7 @@ | ||||||
|     "@lingui/react": "^4.5.0", |     "@lingui/react": "^4.5.0", | ||||||
|     "@mattermost/react-native-paste-input": "^0.6.4", |     "@mattermost/react-native-paste-input": "^0.6.4", | ||||||
|     "@miblanchard/react-native-slider": "^2.3.1", |     "@miblanchard/react-native-slider": "^2.3.1", | ||||||
|  |     "@radix-ui/react-dropdown-menu": "^2.0.6", | ||||||
|     "@react-native-async-storage/async-storage": "1.21.0", |     "@react-native-async-storage/async-storage": "1.21.0", | ||||||
|     "@react-native-camera-roll/camera-roll": "^5.2.2", |     "@react-native-camera-roll/camera-roll": "^5.2.2", | ||||||
|     "@react-native-clipboard/clipboard": "^1.10.0", |     "@react-native-clipboard/clipboard": "^1.10.0", | ||||||
|  | @ -148,6 +149,7 @@ | ||||||
|     "react-avatar-editor": "^13.0.0", |     "react-avatar-editor": "^13.0.0", | ||||||
|     "react-circular-progressbar": "^2.1.0", |     "react-circular-progressbar": "^2.1.0", | ||||||
|     "react-dom": "^18.2.0", |     "react-dom": "^18.2.0", | ||||||
|  |     "react-keyed-flatten-children": "^3.0.0", | ||||||
|     "react-native": "0.73.2", |     "react-native": "0.73.2", | ||||||
|     "react-native-appstate-hook": "^1.0.6", |     "react-native-appstate-hook": "^1.0.6", | ||||||
|     "react-native-drawer-layout": "^4.0.0-alpha.3", |     "react-native-drawer-layout": "^4.0.0-alpha.3", | ||||||
|  | @ -177,6 +179,8 @@ | ||||||
|     "react-responsive": "^9.0.2", |     "react-responsive": "^9.0.2", | ||||||
|     "rn-fetch-blob": "^0.12.0", |     "rn-fetch-blob": "^0.12.0", | ||||||
|     "sentry-expo": "~7.0.1", |     "sentry-expo": "~7.0.1", | ||||||
|  |     "statsig-react": "^1.36.0", | ||||||
|  |     "statsig-react-native-expo": "^4.6.1", | ||||||
|     "tippy.js": "^6.3.7", |     "tippy.js": "^6.3.7", | ||||||
|     "tlds": "^1.234.0", |     "tlds": "^1.234.0", | ||||||
|     "use-deep-compare": "^1.1.0", |     "use-deep-compare": "^1.1.0", | ||||||
|  |  | ||||||
							
								
								
									
										20
									
								
								plugins/withAndroidStylesWindowBackgroundPlugin.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								plugins/withAndroidStylesWindowBackgroundPlugin.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,20 @@ | ||||||
|  | const {withAndroidStyles, AndroidConfig} = require('@expo/config-plugins') | ||||||
|  | 
 | ||||||
|  | module.exports = function withAndroidStylesWindowBackgroundPlugin(appConfig) { | ||||||
|  |   return withAndroidStyles(appConfig, function (decoratedAppConfig) { | ||||||
|  |     try { | ||||||
|  |       decoratedAppConfig.modResults = AndroidConfig.Styles.assignStylesValue( | ||||||
|  |         decoratedAppConfig.modResults, | ||||||
|  |         { | ||||||
|  |           add: true, | ||||||
|  |           parent: AndroidConfig.Styles.getAppThemeLightNoActionBarGroup(), | ||||||
|  |           name: 'android:windowBackground', | ||||||
|  |           value: '@drawable/splashscreen', | ||||||
|  |         }, | ||||||
|  |       ) | ||||||
|  |     } catch (e) { | ||||||
|  |       console.error(`withAndroidStylesWindowBackgroundPlugin failed`, e) | ||||||
|  |     } | ||||||
|  |     return decoratedAppConfig | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | @ -43,9 +43,12 @@ import {Provider as UnreadNotifsProvider} from 'state/queries/notifications/unre | ||||||
| import * as persisted from '#/state/persisted' | import * as persisted from '#/state/persisted' | ||||||
| import {Splash} from '#/Splash' | import {Splash} from '#/Splash' | ||||||
| import {Provider as PortalProvider} from '#/components/Portal' | import {Provider as PortalProvider} from '#/components/Portal' | ||||||
|  | import {Provider as StatsigProvider} from '#/lib/statsig/statsig' | ||||||
| import {msg} from '@lingui/macro' | import {msg} from '@lingui/macro' | ||||||
| import {useLingui} from '@lingui/react' | import {useLingui} from '@lingui/react' | ||||||
| import {useIntentHandler} from 'lib/hooks/useIntentHandler' | import {useIntentHandler} from 'lib/hooks/useIntentHandler' | ||||||
|  | import {StatusBar} from 'expo-status-bar' | ||||||
|  | import {isAndroid} from 'platform/detection' | ||||||
| 
 | 
 | ||||||
| SplashScreen.preventAutoHideAsync() | SplashScreen.preventAutoHideAsync() | ||||||
| 
 | 
 | ||||||
|  | @ -69,11 +72,13 @@ function InnerApp() { | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <SafeAreaProvider initialMetrics={initialWindowMetrics}> |     <SafeAreaProvider initialMetrics={initialWindowMetrics}> | ||||||
|  |       {isAndroid && <StatusBar />} | ||||||
|       <Alf theme={theme}> |       <Alf theme={theme}> | ||||||
|         <Splash isReady={!isInitialLoad}> |         <Splash isReady={!isInitialLoad}> | ||||||
|           <React.Fragment |           <React.Fragment | ||||||
|             // Resets the entire tree below when it changes:
 |             // Resets the entire tree below when it changes:
 | ||||||
|             key={currentAccount?.did}> |             key={currentAccount?.did}> | ||||||
|  |             <StatsigProvider> | ||||||
|               <LoggedOutViewProvider> |               <LoggedOutViewProvider> | ||||||
|                 <SelectedFeedProvider> |                 <SelectedFeedProvider> | ||||||
|                   <UnreadNotifsProvider> |                   <UnreadNotifsProvider> | ||||||
|  | @ -89,6 +94,7 @@ function InnerApp() { | ||||||
|                   </UnreadNotifsProvider> |                   </UnreadNotifsProvider> | ||||||
|                 </SelectedFeedProvider> |                 </SelectedFeedProvider> | ||||||
|               </LoggedOutViewProvider> |               </LoggedOutViewProvider> | ||||||
|  |             </StatsigProvider> | ||||||
|           </React.Fragment> |           </React.Fragment> | ||||||
|         </Splash> |         </Splash> | ||||||
|       </Alf> |       </Alf> | ||||||
|  |  | ||||||
|  | @ -32,6 +32,7 @@ import { | ||||||
| import {Provider as UnreadNotifsProvider} from 'state/queries/notifications/unread' | import {Provider as UnreadNotifsProvider} from 'state/queries/notifications/unread' | ||||||
| import * as persisted from '#/state/persisted' | import * as persisted from '#/state/persisted' | ||||||
| import {Provider as PortalProvider} from '#/components/Portal' | import {Provider as PortalProvider} from '#/components/Portal' | ||||||
|  | import {Provider as StatsigProvider} from '#/lib/statsig/statsig' | ||||||
| import {useIntentHandler} from 'lib/hooks/useIntentHandler' | import {useIntentHandler} from 'lib/hooks/useIntentHandler' | ||||||
| 
 | 
 | ||||||
| function InnerApp() { | function InnerApp() { | ||||||
|  | @ -54,6 +55,7 @@ function InnerApp() { | ||||||
|       <React.Fragment |       <React.Fragment | ||||||
|         // Resets the entire tree below when it changes:
 |         // Resets the entire tree below when it changes:
 | ||||||
|         key={currentAccount?.did}> |         key={currentAccount?.did}> | ||||||
|  |         <StatsigProvider> | ||||||
|           <LoggedOutViewProvider> |           <LoggedOutViewProvider> | ||||||
|             <SelectedFeedProvider> |             <SelectedFeedProvider> | ||||||
|               <UnreadNotifsProvider> |               <UnreadNotifsProvider> | ||||||
|  | @ -69,6 +71,7 @@ function InnerApp() { | ||||||
|               </UnreadNotifsProvider> |               </UnreadNotifsProvider> | ||||||
|             </SelectedFeedProvider> |             </SelectedFeedProvider> | ||||||
|           </LoggedOutViewProvider> |           </LoggedOutViewProvider> | ||||||
|  |         </StatsigProvider> | ||||||
|       </React.Fragment> |       </React.Fragment> | ||||||
|     </Alf> |     </Alf> | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  | @ -21,7 +21,8 @@ export function useDialogControl(): DialogOuterProps['control'] { | ||||||
|     open: () => {}, |     open: () => {}, | ||||||
|     close: () => {}, |     close: () => {}, | ||||||
|   }) |   }) | ||||||
|   const {activeDialogs} = useDialogStateContext() |   const {activeDialogs, openDialogs} = useDialogStateContext() | ||||||
|  |   const isOpen = openDialogs.includes(id) | ||||||
| 
 | 
 | ||||||
|   React.useEffect(() => { |   React.useEffect(() => { | ||||||
|     activeDialogs.current.set(id, control) |     activeDialogs.current.set(id, control) | ||||||
|  | @ -31,14 +32,18 @@ export function useDialogControl(): DialogOuterProps['control'] { | ||||||
|     } |     } | ||||||
|   }, [id, activeDialogs]) |   }, [id, activeDialogs]) | ||||||
| 
 | 
 | ||||||
|   return { |   return React.useMemo<DialogOuterProps['control']>( | ||||||
|  |     () => ({ | ||||||
|       id, |       id, | ||||||
|       ref: control, |       ref: control, | ||||||
|  |       isOpen, | ||||||
|       open: () => { |       open: () => { | ||||||
|         control.current.open() |         control.current.open() | ||||||
|       }, |       }, | ||||||
|       close: cb => { |       close: cb => { | ||||||
|         control.current.close(cb) |         control.current.close(cb) | ||||||
|       }, |       }, | ||||||
|   } |     }), | ||||||
|  |     [id, control, isOpen], | ||||||
|  |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ import {useTheme, atoms as a, flatten} from '#/alf' | ||||||
| import {Portal} from '#/components/Portal' | import {Portal} from '#/components/Portal' | ||||||
| import {createInput} from '#/components/forms/TextField' | import {createInput} from '#/components/forms/TextField' | ||||||
| import {logger} from '#/logger' | import {logger} from '#/logger' | ||||||
| import {useDialogStateContext} from '#/state/dialogs' | import {useDialogStateControlContext} from '#/state/dialogs' | ||||||
| 
 | 
 | ||||||
| import { | import { | ||||||
|   DialogOuterProps, |   DialogOuterProps, | ||||||
|  | @ -82,7 +82,7 @@ export function Outer({ | ||||||
|   const hasSnapPoints = !!sheetOptions.snapPoints |   const hasSnapPoints = !!sheetOptions.snapPoints | ||||||
|   const insets = useSafeAreaInsets() |   const insets = useSafeAreaInsets() | ||||||
|   const closeCallback = React.useRef<() => void>() |   const closeCallback = React.useRef<() => void>() | ||||||
|   const {openDialogs} = useDialogStateContext() |   const {setDialogIsOpen} = useDialogStateControlContext() | ||||||
| 
 | 
 | ||||||
|   /* |   /* | ||||||
|    * Used to manage open/closed, but index is otherwise handled internally by `BottomSheet` |    * Used to manage open/closed, but index is otherwise handled internally by `BottomSheet` | ||||||
|  | @ -96,11 +96,11 @@ export function Outer({ | ||||||
| 
 | 
 | ||||||
|   const open = React.useCallback<DialogControlProps['open']>( |   const open = React.useCallback<DialogControlProps['open']>( | ||||||
|     ({index} = {}) => { |     ({index} = {}) => { | ||||||
|       openDialogs.current.add(control.id) |       setDialogIsOpen(control.id, true) | ||||||
|       // can be set to any index of `snapPoints`, but `0` is the first i.e. "open"
 |       // can be set to any index of `snapPoints`, but `0` is the first i.e. "open"
 | ||||||
|       setOpenIndex(index || 0) |       setOpenIndex(index || 0) | ||||||
|     }, |     }, | ||||||
|     [setOpenIndex, openDialogs, control.id], |     [setOpenIndex, setDialogIsOpen, control.id], | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   const close = React.useCallback<DialogControlProps['close']>(cb => { |   const close = React.useCallback<DialogControlProps['close']>(cb => { | ||||||
|  | @ -119,9 +119,7 @@ export function Outer({ | ||||||
|     [open, close], |     [open, close], | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   const onChange = React.useCallback( |   const onCloseInner = React.useCallback(() => { | ||||||
|     (index: number) => { |  | ||||||
|       if (index === -1) { |  | ||||||
|     Keyboard.dismiss() |     Keyboard.dismiss() | ||||||
|     try { |     try { | ||||||
|       closeCallback.current?.() |       closeCallback.current?.() | ||||||
|  | @ -132,20 +130,22 @@ export function Outer({ | ||||||
|     } finally { |     } finally { | ||||||
|       closeCallback.current = undefined |       closeCallback.current = undefined | ||||||
|     } |     } | ||||||
| 
 |     setDialogIsOpen(control.id, false) | ||||||
|         openDialogs.current.delete(control.id) |  | ||||||
|     onClose?.() |     onClose?.() | ||||||
|     setOpenIndex(-1) |     setOpenIndex(-1) | ||||||
|       } |   }, [control.id, onClose, setDialogIsOpen]) | ||||||
|     }, |  | ||||||
|     [onClose, setOpenIndex, openDialogs, control.id], |  | ||||||
|   ) |  | ||||||
| 
 | 
 | ||||||
|   const context = React.useMemo(() => ({close}), [close]) |   const context = React.useMemo(() => ({close}), [close]) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     isOpen && ( |     isOpen && ( | ||||||
|       <Portal> |       <Portal> | ||||||
|  |         <View | ||||||
|  |           // iOS
 | ||||||
|  |           accessibilityViewIsModal | ||||||
|  |           // Android
 | ||||||
|  |           importantForAccessibility="yes" | ||||||
|  |           style={[a.absolute, a.inset_0]}> | ||||||
|           <BottomSheet |           <BottomSheet | ||||||
|             enableDynamicSizing={!hasSnapPoints} |             enableDynamicSizing={!hasSnapPoints} | ||||||
|             enablePanDownToClose |             enablePanDownToClose | ||||||
|  | @ -161,7 +161,7 @@ export function Outer({ | ||||||
|             backdropComponent={Backdrop} |             backdropComponent={Backdrop} | ||||||
|             handleIndicatorStyle={{backgroundColor: t.palette.primary_500}} |             handleIndicatorStyle={{backgroundColor: t.palette.primary_500}} | ||||||
|             handleStyle={{display: 'none'}} |             handleStyle={{display: 'none'}} | ||||||
|           onChange={onChange}> |             onClose={onCloseInner}> | ||||||
|             <Context.Provider value={context}> |             <Context.Provider value={context}> | ||||||
|               <View |               <View | ||||||
|                 style={[ |                 style={[ | ||||||
|  | @ -178,6 +178,7 @@ export function Outer({ | ||||||
|               {children} |               {children} | ||||||
|             </Context.Provider> |             </Context.Provider> | ||||||
|           </BottomSheet> |           </BottomSheet> | ||||||
|  |         </View> | ||||||
|       </Portal> |       </Portal> | ||||||
|     ) |     ) | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ import {DialogOuterProps, DialogInnerProps} from '#/components/Dialog/types' | ||||||
| import {Context} from '#/components/Dialog/context' | import {Context} from '#/components/Dialog/context' | ||||||
| import {Button, ButtonIcon} from '#/components/Button' | import {Button, ButtonIcon} from '#/components/Button' | ||||||
| import {TimesLarge_Stroke2_Corner0_Rounded as X} from '#/components/icons/Times' | import {TimesLarge_Stroke2_Corner0_Rounded as X} from '#/components/icons/Times' | ||||||
| import {useDialogStateContext} from '#/state/dialogs' | import {useDialogStateControlContext} from '#/state/dialogs' | ||||||
| 
 | 
 | ||||||
| export {useDialogControl, useDialogContext} from '#/components/Dialog/context' | export {useDialogControl, useDialogContext} from '#/components/Dialog/context' | ||||||
| export * from '#/components/Dialog/types' | export * from '#/components/Dialog/types' | ||||||
|  | @ -30,21 +30,21 @@ export function Outer({ | ||||||
|   const {gtMobile} = useBreakpoints() |   const {gtMobile} = useBreakpoints() | ||||||
|   const [isOpen, setIsOpen] = React.useState(false) |   const [isOpen, setIsOpen] = React.useState(false) | ||||||
|   const [isVisible, setIsVisible] = React.useState(true) |   const [isVisible, setIsVisible] = React.useState(true) | ||||||
|   const {openDialogs} = useDialogStateContext() |   const {setDialogIsOpen} = useDialogStateControlContext() | ||||||
| 
 | 
 | ||||||
|   const open = React.useCallback(() => { |   const open = React.useCallback(() => { | ||||||
|     setIsOpen(true) |     setIsOpen(true) | ||||||
|     openDialogs.current.add(control.id) |     setDialogIsOpen(control.id, true) | ||||||
|   }, [setIsOpen, openDialogs, control.id]) |   }, [setIsOpen, setDialogIsOpen, control.id]) | ||||||
| 
 | 
 | ||||||
|   const close = React.useCallback(async () => { |   const close = React.useCallback(async () => { | ||||||
|     setIsVisible(false) |     setIsVisible(false) | ||||||
|     await new Promise(resolve => setTimeout(resolve, 150)) |     await new Promise(resolve => setTimeout(resolve, 150)) | ||||||
|     setIsOpen(false) |     setIsOpen(false) | ||||||
|     setIsVisible(true) |     setIsVisible(true) | ||||||
|     openDialogs.current.delete(control.id) |     setDialogIsOpen(control.id, false) | ||||||
|     onClose?.() |     onClose?.() | ||||||
|   }, [onClose, setIsOpen, openDialogs, control.id]) |   }, [onClose, setIsOpen, setDialogIsOpen, control.id]) | ||||||
| 
 | 
 | ||||||
|   useImperativeHandle( |   useImperativeHandle( | ||||||
|     control.ref, |     control.ref, | ||||||
|  |  | ||||||
|  | @ -22,6 +22,7 @@ export type DialogControlRefProps = { | ||||||
| export type DialogControlProps = DialogControlRefProps & { | export type DialogControlProps = DialogControlRefProps & { | ||||||
|   id: string |   id: string | ||||||
|   ref: React.RefObject<DialogControlRefProps> |   ref: React.RefObject<DialogControlRefProps> | ||||||
|  |   isOpen: boolean | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export type DialogContextProps = { | export type DialogContextProps = { | ||||||
|  |  | ||||||
							
								
								
									
										8
									
								
								src/components/Menu/context.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								src/components/Menu/context.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | ||||||
|  | import React from 'react' | ||||||
|  | 
 | ||||||
|  | import type {ContextType} from '#/components/Menu/types' | ||||||
|  | 
 | ||||||
|  | export const Context = React.createContext<ContextType>({ | ||||||
|  |   // @ts-ignore
 | ||||||
|  |   control: null, | ||||||
|  | }) | ||||||
							
								
								
									
										190
									
								
								src/components/Menu/index.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										190
									
								
								src/components/Menu/index.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,190 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {View, Pressable} from 'react-native' | ||||||
|  | import flattenReactChildren from 'react-keyed-flatten-children' | ||||||
|  | 
 | ||||||
|  | import {atoms as a, useTheme} from '#/alf' | ||||||
|  | import * as Dialog from '#/components/Dialog' | ||||||
|  | import {useInteractionState} from '#/components/hooks/useInteractionState' | ||||||
|  | import {Text} from '#/components/Typography' | ||||||
|  | 
 | ||||||
|  | import {Context} from '#/components/Menu/context' | ||||||
|  | import { | ||||||
|  |   ContextType, | ||||||
|  |   TriggerProps, | ||||||
|  |   ItemProps, | ||||||
|  |   GroupProps, | ||||||
|  |   ItemTextProps, | ||||||
|  |   ItemIconProps, | ||||||
|  | } from '#/components/Menu/types' | ||||||
|  | 
 | ||||||
|  | export {useDialogControl as useMenuControl} from '#/components/Dialog' | ||||||
|  | 
 | ||||||
|  | export function useMemoControlContext() { | ||||||
|  |   return React.useContext(Context) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Root({ | ||||||
|  |   children, | ||||||
|  |   control, | ||||||
|  | }: React.PropsWithChildren<{ | ||||||
|  |   control?: Dialog.DialogOuterProps['control'] | ||||||
|  | }>) { | ||||||
|  |   const defaultControl = Dialog.useDialogControl() | ||||||
|  |   const context = React.useMemo<ContextType>( | ||||||
|  |     () => ({ | ||||||
|  |       control: control || defaultControl, | ||||||
|  |     }), | ||||||
|  |     [control, defaultControl], | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   return <Context.Provider value={context}>{children}</Context.Provider> | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Trigger({children, label}: TriggerProps) { | ||||||
|  |   const {control} = React.useContext(Context) | ||||||
|  |   const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState() | ||||||
|  |   const { | ||||||
|  |     state: pressed, | ||||||
|  |     onIn: onPressIn, | ||||||
|  |     onOut: onPressOut, | ||||||
|  |   } = useInteractionState() | ||||||
|  | 
 | ||||||
|  |   return children({ | ||||||
|  |     isNative: true, | ||||||
|  |     control, | ||||||
|  |     state: { | ||||||
|  |       hovered: false, | ||||||
|  |       focused, | ||||||
|  |       pressed, | ||||||
|  |     }, | ||||||
|  |     props: { | ||||||
|  |       onPress: control.open, | ||||||
|  |       onFocus, | ||||||
|  |       onBlur, | ||||||
|  |       onPressIn, | ||||||
|  |       onPressOut, | ||||||
|  |       accessibilityLabel: label, | ||||||
|  |     }, | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Outer({children}: React.PropsWithChildren<{}>) { | ||||||
|  |   const context = React.useContext(Context) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Dialog.Outer control={context.control}> | ||||||
|  |       <Dialog.Handle /> | ||||||
|  | 
 | ||||||
|  |       {/* Re-wrap with context since Dialogs are portal-ed to root */} | ||||||
|  |       <Context.Provider value={context}> | ||||||
|  |         <Dialog.ScrollableInner label="Menu TODO"> | ||||||
|  |           <View style={[a.gap_lg]}>{children}</View> | ||||||
|  |           <View style={{height: a.gap_lg.gap}} /> | ||||||
|  |         </Dialog.ScrollableInner> | ||||||
|  |       </Context.Provider> | ||||||
|  |     </Dialog.Outer> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Item({children, label, style, onPress, ...rest}: ItemProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   const {control} = React.useContext(Context) | ||||||
|  |   const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState() | ||||||
|  |   const { | ||||||
|  |     state: pressed, | ||||||
|  |     onIn: onPressIn, | ||||||
|  |     onOut: onPressOut, | ||||||
|  |   } = useInteractionState() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Pressable | ||||||
|  |       {...rest} | ||||||
|  |       accessibilityHint="" | ||||||
|  |       accessibilityLabel={label} | ||||||
|  |       onPress={e => { | ||||||
|  |         onPress(e) | ||||||
|  | 
 | ||||||
|  |         if (!e.defaultPrevented) { | ||||||
|  |           control?.close() | ||||||
|  |         } | ||||||
|  |       }} | ||||||
|  |       onFocus={onFocus} | ||||||
|  |       onBlur={onBlur} | ||||||
|  |       onPressIn={onPressIn} | ||||||
|  |       onPressOut={onPressOut} | ||||||
|  |       style={[ | ||||||
|  |         a.flex_row, | ||||||
|  |         a.align_center, | ||||||
|  |         a.gap_sm, | ||||||
|  |         a.px_md, | ||||||
|  |         a.rounded_md, | ||||||
|  |         a.border, | ||||||
|  |         t.atoms.bg_contrast_25, | ||||||
|  |         t.atoms.border_contrast_low, | ||||||
|  |         {minHeight: 44, paddingVertical: 10}, | ||||||
|  |         style, | ||||||
|  |         (focused || pressed) && [t.atoms.bg_contrast_50], | ||||||
|  |       ]}> | ||||||
|  |       {children} | ||||||
|  |     </Pressable> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function ItemText({children, style}: ItemTextProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <Text | ||||||
|  |       numberOfLines={1} | ||||||
|  |       ellipsizeMode="middle" | ||||||
|  |       style={[ | ||||||
|  |         a.flex_1, | ||||||
|  |         a.text_md, | ||||||
|  |         a.font_bold, | ||||||
|  |         t.atoms.text_contrast_medium, | ||||||
|  |         {paddingTop: 3}, | ||||||
|  |         style, | ||||||
|  |       ]}> | ||||||
|  |       {children} | ||||||
|  |     </Text> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function ItemIcon({icon: Comp}: ItemIconProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return <Comp size="lg" fill={t.atoms.text_contrast_medium.color} /> | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Group({children, style}: GroupProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <View | ||||||
|  |       style={[ | ||||||
|  |         a.rounded_md, | ||||||
|  |         a.overflow_hidden, | ||||||
|  |         a.border, | ||||||
|  |         t.atoms.border_contrast_low, | ||||||
|  |         style, | ||||||
|  |       ]}> | ||||||
|  |       {flattenReactChildren(children).map((child, i) => { | ||||||
|  |         return React.isValidElement(child) && child.type === Item ? ( | ||||||
|  |           <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> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Divider() { | ||||||
|  |   return null | ||||||
|  | } | ||||||
							
								
								
									
										247
									
								
								src/components/Menu/index.web.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										247
									
								
								src/components/Menu/index.web.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,247 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {View, Pressable} from 'react-native' | ||||||
|  | import * as DropdownMenu from '@radix-ui/react-dropdown-menu' | ||||||
|  | 
 | ||||||
|  | import * as Dialog from '#/components/Dialog' | ||||||
|  | import {useInteractionState} from '#/components/hooks/useInteractionState' | ||||||
|  | import {atoms as a, useTheme, flatten, web} from '#/alf' | ||||||
|  | import {Text} from '#/components/Typography' | ||||||
|  | 
 | ||||||
|  | import { | ||||||
|  |   ContextType, | ||||||
|  |   TriggerProps, | ||||||
|  |   ItemProps, | ||||||
|  |   GroupProps, | ||||||
|  |   ItemTextProps, | ||||||
|  |   ItemIconProps, | ||||||
|  | } from '#/components/Menu/types' | ||||||
|  | import {Context} from '#/components/Menu/context' | ||||||
|  | 
 | ||||||
|  | export function useMenuControl(): Dialog.DialogControlProps { | ||||||
|  |   const id = React.useId() | ||||||
|  |   const [isOpen, setIsOpen] = React.useState(false) | ||||||
|  | 
 | ||||||
|  |   return React.useMemo( | ||||||
|  |     () => ({ | ||||||
|  |       id, | ||||||
|  |       ref: {current: null}, | ||||||
|  |       isOpen, | ||||||
|  |       open() { | ||||||
|  |         setIsOpen(true) | ||||||
|  |       }, | ||||||
|  |       close() { | ||||||
|  |         setIsOpen(false) | ||||||
|  |       }, | ||||||
|  |     }), | ||||||
|  |     [id, isOpen, setIsOpen], | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useMemoControlContext() { | ||||||
|  |   return React.useContext(Context) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Root({ | ||||||
|  |   children, | ||||||
|  |   control, | ||||||
|  | }: React.PropsWithChildren<{ | ||||||
|  |   control?: Dialog.DialogOuterProps['control'] | ||||||
|  | }>) { | ||||||
|  |   const defaultControl = useMenuControl() | ||||||
|  |   const context = React.useMemo<ContextType>( | ||||||
|  |     () => ({ | ||||||
|  |       control: control || defaultControl, | ||||||
|  |     }), | ||||||
|  |     [control, defaultControl], | ||||||
|  |   ) | ||||||
|  |   const onOpenChange = React.useCallback( | ||||||
|  |     (open: boolean) => { | ||||||
|  |       if (context.control.isOpen && !open) { | ||||||
|  |         context.control.close() | ||||||
|  |       } else if (!context.control.isOpen && open) { | ||||||
|  |         context.control.open() | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     [context.control], | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Context.Provider value={context}> | ||||||
|  |       <DropdownMenu.Root | ||||||
|  |         open={context.control.isOpen} | ||||||
|  |         onOpenChange={onOpenChange}> | ||||||
|  |         {children} | ||||||
|  |       </DropdownMenu.Root> | ||||||
|  |     </Context.Provider> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Trigger({children, label, style}: TriggerProps) { | ||||||
|  |   const {control} = React.useContext(Context) | ||||||
|  |   const { | ||||||
|  |     state: hovered, | ||||||
|  |     onIn: onMouseEnter, | ||||||
|  |     onOut: onMouseLeave, | ||||||
|  |   } = useInteractionState() | ||||||
|  |   const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <DropdownMenu.Trigger asChild> | ||||||
|  |       <Pressable | ||||||
|  |         accessibilityHint="" | ||||||
|  |         accessibilityLabel={label} | ||||||
|  |         onFocus={onFocus} | ||||||
|  |         onBlur={onBlur} | ||||||
|  |         style={flatten([style, web({outline: 0})])} | ||||||
|  |         onPointerDown={() => { | ||||||
|  |           control.open() | ||||||
|  |         }} | ||||||
|  |         {...web({ | ||||||
|  |           onMouseEnter, | ||||||
|  |           onMouseLeave, | ||||||
|  |         })}> | ||||||
|  |         {children({ | ||||||
|  |           isNative: false, | ||||||
|  |           control, | ||||||
|  |           state: { | ||||||
|  |             hovered, | ||||||
|  |             focused, | ||||||
|  |             pressed: false, | ||||||
|  |           }, | ||||||
|  |           props: {}, | ||||||
|  |         })} | ||||||
|  |       </Pressable> | ||||||
|  |     </DropdownMenu.Trigger> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Outer({children}: React.PropsWithChildren<{}>) { | ||||||
|  |   const t = useTheme() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <DropdownMenu.Portal> | ||||||
|  |       <DropdownMenu.Content sideOffset={5} loop aria-label="Test"> | ||||||
|  |         <View | ||||||
|  |           style={[ | ||||||
|  |             a.rounded_sm, | ||||||
|  |             a.p_xs, | ||||||
|  |             t.name === 'light' ? t.atoms.bg : t.atoms.bg_contrast_25, | ||||||
|  |             t.atoms.shadow_md, | ||||||
|  |           ]}> | ||||||
|  |           {children} | ||||||
|  |         </View> | ||||||
|  | 
 | ||||||
|  |         <DropdownMenu.Arrow | ||||||
|  |           className="DropdownMenuArrow" | ||||||
|  |           fill={ | ||||||
|  |             (t.name === 'light' ? t.atoms.bg : t.atoms.bg_contrast_25) | ||||||
|  |               .backgroundColor | ||||||
|  |           } | ||||||
|  |         /> | ||||||
|  |       </DropdownMenu.Content> | ||||||
|  |     </DropdownMenu.Portal> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Item({children, label, onPress, ...rest}: ItemProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   const {control} = React.useContext(Context) | ||||||
|  |   const { | ||||||
|  |     state: hovered, | ||||||
|  |     onIn: onMouseEnter, | ||||||
|  |     onOut: onMouseLeave, | ||||||
|  |   } = useInteractionState() | ||||||
|  |   const {state: focused, onIn: onFocus, onOut: onBlur} = useInteractionState() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <DropdownMenu.Item asChild> | ||||||
|  |       <Pressable | ||||||
|  |         {...rest} | ||||||
|  |         className="radix-dropdown-item" | ||||||
|  |         accessibilityHint="" | ||||||
|  |         accessibilityLabel={label} | ||||||
|  |         onPress={e => { | ||||||
|  |           onPress(e) | ||||||
|  | 
 | ||||||
|  |           /** | ||||||
|  |            * Ported forward from Radix | ||||||
|  |            * @see https://www.radix-ui.com/primitives/docs/components/dropdown-menu#item
 | ||||||
|  |            */ | ||||||
|  |           if (!e.defaultPrevented) { | ||||||
|  |             control.close() | ||||||
|  |           } | ||||||
|  |         }} | ||||||
|  |         onFocus={onFocus} | ||||||
|  |         onBlur={onBlur} | ||||||
|  |         // need `flatten` here for Radix compat
 | ||||||
|  |         style={flatten([ | ||||||
|  |           a.flex_row, | ||||||
|  |           a.align_center, | ||||||
|  |           a.gap_sm, | ||||||
|  |           a.py_sm, | ||||||
|  |           a.rounded_xs, | ||||||
|  |           {minHeight: 32, paddingHorizontal: 10}, | ||||||
|  |           web({outline: 0}), | ||||||
|  |           (hovered || focused) && [ | ||||||
|  |             web({outline: '0 !important'}), | ||||||
|  |             t.name === 'light' | ||||||
|  |               ? t.atoms.bg_contrast_25 | ||||||
|  |               : t.atoms.bg_contrast_50, | ||||||
|  |           ], | ||||||
|  |         ])} | ||||||
|  |         {...web({ | ||||||
|  |           onMouseEnter, | ||||||
|  |           onMouseLeave, | ||||||
|  |         })}> | ||||||
|  |         {children} | ||||||
|  |       </Pressable> | ||||||
|  |     </DropdownMenu.Item> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function ItemText({children, style}: ItemTextProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <Text style={[a.flex_1, a.font_bold, t.atoms.text_contrast_high, style]}> | ||||||
|  |       {children} | ||||||
|  |     </Text> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function ItemIcon({icon: Comp, position = 'left'}: ItemIconProps) { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <Comp | ||||||
|  |       size="md" | ||||||
|  |       fill={t.atoms.text_contrast_medium.color} | ||||||
|  |       style={[ | ||||||
|  |         position === 'left' && { | ||||||
|  |           marginLeft: -2, | ||||||
|  |         }, | ||||||
|  |         position === 'right' && { | ||||||
|  |           marginRight: -2, | ||||||
|  |           marginLeft: 12, | ||||||
|  |         }, | ||||||
|  |       ]} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Group({children}: GroupProps) { | ||||||
|  |   return children | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Divider() { | ||||||
|  |   const t = useTheme() | ||||||
|  |   return ( | ||||||
|  |     <DropdownMenu.Separator | ||||||
|  |       style={flatten([ | ||||||
|  |         a.my_xs, | ||||||
|  |         t.atoms.bg_contrast_100, | ||||||
|  |         { | ||||||
|  |           height: 1, | ||||||
|  |         }, | ||||||
|  |       ])} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | } | ||||||
							
								
								
									
										72
									
								
								src/components/Menu/types.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								src/components/Menu/types.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,72 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {GestureResponderEvent, PressableProps} from 'react-native' | ||||||
|  | 
 | ||||||
|  | import {Props as SVGIconProps} from '#/components/icons/common' | ||||||
|  | import * as Dialog from '#/components/Dialog' | ||||||
|  | import {TextStyleProp, ViewStyleProp} from '#/alf' | ||||||
|  | 
 | ||||||
|  | export type ContextType = { | ||||||
|  |   control: Dialog.DialogOuterProps['control'] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export type TriggerProps = ViewStyleProp & { | ||||||
|  |   children(props: TriggerChildProps): React.ReactNode | ||||||
|  |   label: string | ||||||
|  | } | ||||||
|  | export type TriggerChildProps = | ||||||
|  |   | { | ||||||
|  |       isNative: true | ||||||
|  |       control: Dialog.DialogOuterProps['control'] | ||||||
|  |       state: { | ||||||
|  |         /** | ||||||
|  |          * Web only, `false` on native | ||||||
|  |          */ | ||||||
|  |         hovered: false | ||||||
|  |         focused: boolean | ||||||
|  |         pressed: boolean | ||||||
|  |       } | ||||||
|  |       /** | ||||||
|  |        * We don't necessarily know what these will be spread on to, so we | ||||||
|  |        * should add props one-by-one. | ||||||
|  |        * | ||||||
|  |        * On web, these properties are applied to a parent `Pressable`, so this | ||||||
|  |        * object is empty. | ||||||
|  |        */ | ||||||
|  |       props: { | ||||||
|  |         onPress: () => void | ||||||
|  |         onFocus: () => void | ||||||
|  |         onBlur: () => void | ||||||
|  |         onPressIn: () => void | ||||||
|  |         onPressOut: () => void | ||||||
|  |         accessibilityLabel: string | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   | { | ||||||
|  |       isNative: false | ||||||
|  |       control: Dialog.DialogOuterProps['control'] | ||||||
|  |       state: { | ||||||
|  |         hovered: boolean | ||||||
|  |         focused: boolean | ||||||
|  |         /** | ||||||
|  |          * Native only, `false` on web | ||||||
|  |          */ | ||||||
|  |         pressed: false | ||||||
|  |       } | ||||||
|  |       props: {} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | export type ItemProps = React.PropsWithChildren< | ||||||
|  |   Omit<PressableProps, 'style'> & | ||||||
|  |     ViewStyleProp & { | ||||||
|  |       label: string | ||||||
|  |       onPress: (e: GestureResponderEvent) => void | ||||||
|  |     } | ||||||
|  | > | ||||||
|  | 
 | ||||||
|  | export type ItemTextProps = React.PropsWithChildren<TextStyleProp & {}> | ||||||
|  | export type ItemIconProps = React.PropsWithChildren<{ | ||||||
|  |   icon: React.ComponentType<SVGIconProps> | ||||||
|  |   position?: 'left' | 'right' | ||||||
|  | }> | ||||||
|  | 
 | ||||||
|  | export type GroupProps = React.PropsWithChildren<ViewStyleProp & {}> | ||||||
							
								
								
									
										11
									
								
								src/lib/statsig/statsig.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								src/lib/statsig/statsig.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | ||||||
|  | import React from 'react' | ||||||
|  | 
 | ||||||
|  | export function useGate(_gateName: string) { | ||||||
|  |   // Not enabled for native yet.
 | ||||||
|  |   return false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Provider({children}: {children: React.ReactNode}) { | ||||||
|  |   // Not enabled for native yet.
 | ||||||
|  |   return children | ||||||
|  | } | ||||||
							
								
								
									
										51
									
								
								src/lib/statsig/statsig.web.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								src/lib/statsig/statsig.web.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,51 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {StatsigProvider, useGate as useStatsigGate} from 'statsig-react' | ||||||
|  | import {useSession} from '../../state/session' | ||||||
|  | import {sha256} from 'js-sha256' | ||||||
|  | 
 | ||||||
|  | const statsigOptions = { | ||||||
|  |   environment: { | ||||||
|  |     tier: process.env.NODE_ENV === 'development' ? 'development' : 'production', | ||||||
|  |   }, | ||||||
|  |   // Don't block on waiting for network. The fetched config will kick in on next load.
 | ||||||
|  |   // This ensures the UI is always consistent and doesn't update mid-session.
 | ||||||
|  |   // Note this makes cold load (no local storage) and private mode return `false` for all gates.
 | ||||||
|  |   initTimeoutMs: 1, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useGate(gateName: string) { | ||||||
|  |   const {isLoading, value} = useStatsigGate(gateName) | ||||||
|  |   if (isLoading) { | ||||||
|  |     // This should not happen because of waitForInitialization={true}.
 | ||||||
|  |     console.error('Did not expected isLoading to ever be true.') | ||||||
|  |   } | ||||||
|  |   return value | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function toStatsigUser(did: string | undefined) { | ||||||
|  |   let userID: string | undefined | ||||||
|  |   if (did) { | ||||||
|  |     userID = sha256(did) | ||||||
|  |   } | ||||||
|  |   return {userID} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function Provider({children}: {children: React.ReactNode}) { | ||||||
|  |   const {currentAccount} = useSession() | ||||||
|  |   const currentStatsigUser = React.useMemo( | ||||||
|  |     () => toStatsigUser(currentAccount?.did), | ||||||
|  |     [currentAccount?.did], | ||||||
|  |   ) | ||||||
|  |   return ( | ||||||
|  |     <StatsigProvider | ||||||
|  |       sdkKey="client-SXJakO39w9vIhl3D44u8UupyzFl4oZ2qPIkjwcvuPsV" | ||||||
|  |       mountKey={currentStatsigUser.userID} | ||||||
|  |       user={currentStatsigUser} | ||||||
|  |       // This isn't really blocking due to short initTimeoutMs above.
 | ||||||
|  |       // However, it ensures `isLoading` is always `false`.
 | ||||||
|  |       waitForInitialization={true} | ||||||
|  |       options={statsigOptions}> | ||||||
|  |       {children} | ||||||
|  |     </StatsigProvider> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | @ -13,20 +13,20 @@ const DialogContext = React.createContext<{ | ||||||
|    * The currently open dialogs, referenced by their IDs, generated from |    * The currently open dialogs, referenced by their IDs, generated from | ||||||
|    * `useId`. |    * `useId`. | ||||||
|    */ |    */ | ||||||
|   openDialogs: React.MutableRefObject<Set<string>> |   openDialogs: string[] | ||||||
| }>({ | }>({ | ||||||
|   activeDialogs: { |   activeDialogs: { | ||||||
|     current: new Map(), |     current: new Map(), | ||||||
|   }, |   }, | ||||||
|   openDialogs: { |   openDialogs: [], | ||||||
|     current: new Set(), |  | ||||||
|   }, |  | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| const DialogControlContext = React.createContext<{ | const DialogControlContext = React.createContext<{ | ||||||
|   closeAllDialogs(): boolean |   closeAllDialogs(): boolean | ||||||
|  |   setDialogIsOpen(id: string, isOpen: boolean): void | ||||||
| }>({ | }>({ | ||||||
|   closeAllDialogs: () => false, |   closeAllDialogs: () => false, | ||||||
|  |   setDialogIsOpen: () => {}, | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| export function useDialogStateContext() { | export function useDialogStateContext() { | ||||||
|  | @ -41,15 +41,31 @@ export function Provider({children}: React.PropsWithChildren<{}>) { | ||||||
|   const activeDialogs = React.useRef< |   const activeDialogs = React.useRef< | ||||||
|     Map<string, React.MutableRefObject<DialogControlRefProps>> |     Map<string, React.MutableRefObject<DialogControlRefProps>> | ||||||
|   >(new Map()) |   >(new Map()) | ||||||
|   const openDialogs = React.useRef<Set<string>>(new Set()) |   const [openDialogs, setOpenDialogs] = React.useState<string[]>([]) | ||||||
| 
 | 
 | ||||||
|   const closeAllDialogs = React.useCallback(() => { |   const closeAllDialogs = React.useCallback(() => { | ||||||
|     activeDialogs.current.forEach(dialog => dialog.current.close()) |     activeDialogs.current.forEach(dialog => dialog.current.close()) | ||||||
|     return openDialogs.current.size > 0 |     return openDialogs.length > 0 | ||||||
|   }, []) |   }, [openDialogs]) | ||||||
| 
 | 
 | ||||||
|   const context = React.useMemo(() => ({activeDialogs, openDialogs}), []) |   const setDialogIsOpen = React.useCallback( | ||||||
|   const controls = React.useMemo(() => ({closeAllDialogs}), [closeAllDialogs]) |     (id: string, isOpen: boolean) => { | ||||||
|  |       setOpenDialogs(prev => { | ||||||
|  |         const filtered = prev.filter(dialogId => dialogId !== id) as string[] | ||||||
|  |         return isOpen ? [...filtered, id] : filtered | ||||||
|  |       }) | ||||||
|  |     }, | ||||||
|  |     [setOpenDialogs], | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const context = React.useMemo( | ||||||
|  |     () => ({activeDialogs, openDialogs}), | ||||||
|  |     [openDialogs], | ||||||
|  |   ) | ||||||
|  |   const controls = React.useMemo( | ||||||
|  |     () => ({closeAllDialogs, setDialogIsOpen}), | ||||||
|  |     [closeAllDialogs, setDialogIsOpen], | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <DialogContext.Provider value={context}> |     <DialogContext.Provider value={context}> | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| import React from 'react' | import React from 'react' | ||||||
| import {StyleSheet, View} from 'react-native' | import {StyleSheet, View} from 'react-native' | ||||||
|  | import Animated from 'react-native-reanimated' | ||||||
| import {usePalette} from 'lib/hooks/usePalette' | import {usePalette} from 'lib/hooks/usePalette' | ||||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||||
| import {HomeHeaderLayoutMobile} from './HomeHeaderLayoutMobile' | import {HomeHeaderLayoutMobile} from './HomeHeaderLayoutMobile' | ||||||
|  | @ -12,6 +13,8 @@ import { | ||||||
| import {useLingui} from '@lingui/react' | import {useLingui} from '@lingui/react' | ||||||
| import {msg} from '@lingui/macro' | import {msg} from '@lingui/macro' | ||||||
| import {CogIcon} from '#/lib/icons' | import {CogIcon} from '#/lib/icons' | ||||||
|  | import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode' | ||||||
|  | import {useShellLayout} from '#/state/shell/shell-layout' | ||||||
| 
 | 
 | ||||||
| export function HomeHeaderLayout(props: { | export function HomeHeaderLayout(props: { | ||||||
|   children: React.ReactNode |   children: React.ReactNode | ||||||
|  | @ -33,6 +36,8 @@ function HomeHeaderLayoutDesktopAndTablet({ | ||||||
|   tabBarAnchor: JSX.Element | null | undefined |   tabBarAnchor: JSX.Element | null | undefined | ||||||
| }) { | }) { | ||||||
|   const pal = usePalette('default') |   const pal = usePalette('default') | ||||||
|  |   const {headerMinimalShellTransform} = useMinimalShellMode() | ||||||
|  |   const {headerHeight} = useShellLayout() | ||||||
|   const {_} = useLingui() |   const {_} = useLingui() | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|  | @ -60,9 +65,19 @@ function HomeHeaderLayoutDesktopAndTablet({ | ||||||
|         </Link> |         </Link> | ||||||
|       </View> |       </View> | ||||||
|       {tabBarAnchor} |       {tabBarAnchor} | ||||||
|       <View style={[pal.view, pal.border, styles.bar, styles.tabBar]}> |       <Animated.View | ||||||
|  |         onLayout={e => { | ||||||
|  |           headerHeight.value = e.nativeEvent.layout.height | ||||||
|  |         }} | ||||||
|  |         style={[ | ||||||
|  |           pal.view, | ||||||
|  |           pal.border, | ||||||
|  |           styles.bar, | ||||||
|  |           styles.tabBar, | ||||||
|  |           headerMinimalShellTransform, | ||||||
|  |         ]}> | ||||||
|         {children} |         {children} | ||||||
|       </View> |       </Animated.View> | ||||||
|     </> |     </> | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -228,6 +228,7 @@ let FeedItem = ({ | ||||||
|               text={sanitizeDisplayName( |               text={sanitizeDisplayName( | ||||||
|                 authors[0].displayName || authors[0].handle, |                 authors[0].displayName || authors[0].handle, | ||||||
|               )} |               )} | ||||||
|  |               disableMismatchWarning | ||||||
|             /> |             /> | ||||||
|             {authors.length > 1 ? ( |             {authors.length > 1 ? ( | ||||||
|               <> |               <> | ||||||
|  |  | ||||||
							
								
								
									
										79
									
								
								src/view/screens/Storybook/Menus.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								src/view/screens/Storybook/Menus.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,79 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {View} from 'react-native' | ||||||
|  | 
 | ||||||
|  | import {atoms as a, useTheme} from '#/alf' | ||||||
|  | import {Text} from '#/components/Typography' | ||||||
|  | import * as Menu from '#/components/Menu' | ||||||
|  | import {MagnifyingGlass2_Stroke2_Corner0_Rounded as Search} from '#/components/icons/MagnifyingGlass2' | ||||||
|  | // import {useDialogStateControlContext} from '#/state/dialogs'
 | ||||||
|  | 
 | ||||||
|  | export function Menus() { | ||||||
|  |   const t = useTheme() | ||||||
|  |   const menuControl = Menu.useMenuControl() | ||||||
|  |   // const {closeAllDialogs} = useDialogStateControlContext()
 | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <View style={[a.gap_md]}> | ||||||
|  |       <View style={[a.flex_row, a.align_start]}> | ||||||
|  |         <Menu.Root control={menuControl}> | ||||||
|  |           <Menu.Trigger label="Open basic menu" style={[a.flex_1]}> | ||||||
|  |             {({state, props}) => { | ||||||
|  |               return ( | ||||||
|  |                 <Text | ||||||
|  |                   {...props} | ||||||
|  |                   style={[ | ||||||
|  |                     a.py_sm, | ||||||
|  |                     a.px_md, | ||||||
|  |                     a.rounded_sm, | ||||||
|  |                     t.atoms.bg_contrast_50, | ||||||
|  |                     (state.hovered || state.focused || state.pressed) && [ | ||||||
|  |                       t.atoms.bg_contrast_200, | ||||||
|  |                     ], | ||||||
|  |                   ]}> | ||||||
|  |                   Open | ||||||
|  |                 </Text> | ||||||
|  |               ) | ||||||
|  |             }} | ||||||
|  |           </Menu.Trigger> | ||||||
|  | 
 | ||||||
|  |           <Menu.Outer> | ||||||
|  |             <Menu.Group> | ||||||
|  |               <Menu.Item label="Click me" onPress={() => {}}> | ||||||
|  |                 <Menu.ItemIcon icon={Search} /> | ||||||
|  |                 <Menu.ItemText>Click me</Menu.ItemText> | ||||||
|  |               </Menu.Item> | ||||||
|  | 
 | ||||||
|  |               <Menu.Item | ||||||
|  |                 label="Another item" | ||||||
|  |                 onPress={() => menuControl.close()}> | ||||||
|  |                 <Menu.ItemText>Another item</Menu.ItemText> | ||||||
|  |               </Menu.Item> | ||||||
|  |             </Menu.Group> | ||||||
|  | 
 | ||||||
|  |             <Menu.Divider /> | ||||||
|  | 
 | ||||||
|  |             <Menu.Group> | ||||||
|  |               <Menu.Item label="Click me" onPress={() => {}}> | ||||||
|  |                 <Menu.ItemIcon icon={Search} /> | ||||||
|  |                 <Menu.ItemText>Click me</Menu.ItemText> | ||||||
|  |               </Menu.Item> | ||||||
|  | 
 | ||||||
|  |               <Menu.Item | ||||||
|  |                 label="Another item" | ||||||
|  |                 onPress={() => menuControl.close()}> | ||||||
|  |                 <Menu.ItemText>Another item</Menu.ItemText> | ||||||
|  |               </Menu.Item> | ||||||
|  |             </Menu.Group> | ||||||
|  | 
 | ||||||
|  |             <Menu.Divider /> | ||||||
|  | 
 | ||||||
|  |             <Menu.Item label="Click me" onPress={() => {}}> | ||||||
|  |               <Menu.ItemIcon icon={Search} /> | ||||||
|  |               <Menu.ItemText>Click me</Menu.ItemText> | ||||||
|  |             </Menu.Item> | ||||||
|  |           </Menu.Outer> | ||||||
|  |         </Menu.Root> | ||||||
|  |       </View> | ||||||
|  |     </View> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | @ -16,6 +16,7 @@ import {Dialogs} from './Dialogs' | ||||||
| import {Breakpoints} from './Breakpoints' | import {Breakpoints} from './Breakpoints' | ||||||
| import {Shadows} from './Shadows' | import {Shadows} from './Shadows' | ||||||
| import {Icons} from './Icons' | import {Icons} from './Icons' | ||||||
|  | import {Menus} from './Menus' | ||||||
| 
 | 
 | ||||||
| export function Storybook() { | export function Storybook() { | ||||||
|   const t = useTheme() |   const t = useTheme() | ||||||
|  | @ -84,6 +85,7 @@ export function Storybook() { | ||||||
|           <Links /> |           <Links /> | ||||||
|           <Forms /> |           <Forms /> | ||||||
|           <Dialogs /> |           <Dialogs /> | ||||||
|  |           <Menus /> | ||||||
|           <Breakpoints /> |           <Breakpoints /> | ||||||
|         </View> |         </View> | ||||||
|       </CenteredView> |       </CenteredView> | ||||||
|  |  | ||||||
|  | @ -30,6 +30,7 @@ import {useCloseAnyActiveElement} from '#/state/util' | ||||||
| import * as notifications from 'lib/notifications/notifications' | import * as notifications from 'lib/notifications/notifications' | ||||||
| import {Outlet as PortalOutlet} from '#/components/Portal' | import {Outlet as PortalOutlet} from '#/components/Portal' | ||||||
| import {MutedWordsDialog} from '#/components/dialogs/MutedWords' | import {MutedWordsDialog} from '#/components/dialogs/MutedWords' | ||||||
|  | import {useDialogStateContext} from '#/state/dialogs' | ||||||
| 
 | 
 | ||||||
| function ShellInner() { | function ShellInner() { | ||||||
|   const isDrawerOpen = useIsDrawerOpen() |   const isDrawerOpen = useIsDrawerOpen() | ||||||
|  | @ -55,6 +56,7 @@ function ShellInner() { | ||||||
|   const closeAnyActiveElement = useCloseAnyActiveElement() |   const closeAnyActiveElement = useCloseAnyActiveElement() | ||||||
|   // start undefined
 |   // start undefined
 | ||||||
|   const currentAccountDid = React.useRef<string | undefined>(undefined) |   const currentAccountDid = React.useRef<string | undefined>(undefined) | ||||||
|  |   const {openDialogs} = useDialogStateContext() | ||||||
| 
 | 
 | ||||||
|   React.useEffect(() => { |   React.useEffect(() => { | ||||||
|     let listener = {remove() {}} |     let listener = {remove() {}} | ||||||
|  | @ -78,9 +80,21 @@ function ShellInner() { | ||||||
|     } |     } | ||||||
|   }, [currentAccount]) |   }, [currentAccount]) | ||||||
| 
 | 
 | ||||||
|  |   /** | ||||||
|  |    * The counterpart to `accessibilityViewIsModal` for Android. This property | ||||||
|  |    * applies to the parent of all non-modal views, and prevents TalkBack from | ||||||
|  |    * navigating within content beneath an open dialog. | ||||||
|  |    * | ||||||
|  |    * @see https://reactnative.dev/docs/accessibility#importantforaccessibility-android
 | ||||||
|  |    */ | ||||||
|  |   const importantForAccessibility = | ||||||
|  |     openDialogs.length > 0 ? 'no-hide-descendants' : undefined | ||||||
|  | 
 | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <View style={containerPadding}> |       <View | ||||||
|  |         style={containerPadding} | ||||||
|  |         importantForAccessibility={importantForAccessibility}> | ||||||
|         <ErrorBoundary> |         <ErrorBoundary> | ||||||
|           <Drawer |           <Drawer | ||||||
|             renderDrawerContent={renderDrawerContent} |             renderDrawerContent={renderDrawerContent} | ||||||
|  |  | ||||||
|  | @ -217,6 +217,7 @@ | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       /* NativeDropdown component */ |       /* NativeDropdown component */ | ||||||
|  |       .radix-dropdown-item:focus, | ||||||
|       .nativeDropdown-item:focus { |       .nativeDropdown-item:focus { | ||||||
|         outline: none; |         outline: none; | ||||||
|       } |       } | ||||||
|  |  | ||||||
							
								
								
									
										255
									
								
								yarn.lock
									
										
									
									
									
								
							
							
						
						
									
										255
									
								
								yarn.lock
									
										
									
									
									
								
							|  | @ -3115,6 +3115,27 @@ | ||||||
|     xcode "^3.0.1" |     xcode "^3.0.1" | ||||||
|     xml2js "0.6.0" |     xml2js "0.6.0" | ||||||
| 
 | 
 | ||||||
|  | "@expo/config-plugins@~5.0.3": | ||||||
|  |   version "5.0.4" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@expo/config-plugins/-/config-plugins-5.0.4.tgz#216fea6558fe66615af1370de55193f4181cb23e" | ||||||
|  |   integrity sha512-vzUcVpqOMs3h+hyRdhGwk+eGIOhXa5xYdd92yO17RMNHav3v/+ekMbs7XA2c3lepMO8Yd4/5hqmRw9ZTL6jGzg== | ||||||
|  |   dependencies: | ||||||
|  |     "@expo/config-types" "^47.0.0" | ||||||
|  |     "@expo/json-file" "8.2.36" | ||||||
|  |     "@expo/plist" "0.0.18" | ||||||
|  |     "@expo/sdk-runtime-versions" "^1.0.0" | ||||||
|  |     "@react-native/normalize-color" "^2.0.0" | ||||||
|  |     chalk "^4.1.2" | ||||||
|  |     debug "^4.3.1" | ||||||
|  |     find-up "~5.0.0" | ||||||
|  |     getenv "^1.0.0" | ||||||
|  |     glob "7.1.6" | ||||||
|  |     resolve-from "^5.0.0" | ||||||
|  |     semver "^7.3.5" | ||||||
|  |     slash "^3.0.0" | ||||||
|  |     xcode "^3.0.1" | ||||||
|  |     xml2js "0.4.23" | ||||||
|  | 
 | ||||||
| "@expo/config-plugins@~7.8.2": | "@expo/config-plugins@~7.8.2": | ||||||
|   version "7.8.2" |   version "7.8.2" | ||||||
|   resolved "https://registry.yarnpkg.com/@expo/config-plugins/-/config-plugins-7.8.2.tgz#c00ce93c4d6c2cb9e345ed9cd56ceeea05ab8ddb" |   resolved "https://registry.yarnpkg.com/@expo/config-plugins/-/config-plugins-7.8.2.tgz#c00ce93c4d6c2cb9e345ed9cd56ceeea05ab8ddb" | ||||||
|  | @ -3138,6 +3159,11 @@ | ||||||
|     xcode "^3.0.1" |     xcode "^3.0.1" | ||||||
|     xml2js "0.6.0" |     xml2js "0.6.0" | ||||||
| 
 | 
 | ||||||
|  | "@expo/config-types@^47.0.0": | ||||||
|  |   version "47.0.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@expo/config-types/-/config-types-47.0.0.tgz#99eeabe0bba7a776e0f252b78beb0c574692c38d" | ||||||
|  |   integrity sha512-r0pWfuhkv7KIcXMUiNACJmJKKwlTBGMw9VZHNdppS8/0Nve8HZMTkNRFQzTHW1uH3pBj8jEXpyw/2vSWDHex9g== | ||||||
|  | 
 | ||||||
| "@expo/config-types@^50.0.0", "@expo/config-types@^50.0.0-alpha.1": | "@expo/config-types@^50.0.0", "@expo/config-types@^50.0.0-alpha.1": | ||||||
|   version "50.0.0" |   version "50.0.0" | ||||||
|   resolved "https://registry.yarnpkg.com/@expo/config-types/-/config-types-50.0.0.tgz#b534d3ec997ec60f8af24f6ad56244c8afc71a0b" |   resolved "https://registry.yarnpkg.com/@expo/config-types/-/config-types-50.0.0.tgz#b534d3ec997ec60f8af24f6ad56244c8afc71a0b" | ||||||
|  | @ -3160,6 +3186,23 @@ | ||||||
|     slugify "^1.3.4" |     slugify "^1.3.4" | ||||||
|     sucrase "^3.20.0" |     sucrase "^3.20.0" | ||||||
| 
 | 
 | ||||||
|  | "@expo/config@~7.0.0": | ||||||
|  |   version "7.0.3" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@expo/config/-/config-7.0.3.tgz#c9c634e76186de25e296485e51418f1e52966e6e" | ||||||
|  |   integrity sha512-joVtB5o+NF40Tmsdp65UzryRtbnCuMbXkVO4wJnNJO4aaK0EYLdHCYSewORVqNcDfGN0LphQr8VTG2npbd9CJA== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/code-frame" "~7.10.4" | ||||||
|  |     "@expo/config-plugins" "~5.0.3" | ||||||
|  |     "@expo/config-types" "^47.0.0" | ||||||
|  |     "@expo/json-file" "8.2.36" | ||||||
|  |     getenv "^1.0.0" | ||||||
|  |     glob "7.1.6" | ||||||
|  |     require-from-string "^2.0.2" | ||||||
|  |     resolve-from "^5.0.0" | ||||||
|  |     semver "7.3.2" | ||||||
|  |     slugify "^1.3.4" | ||||||
|  |     sucrase "^3.20.0" | ||||||
|  | 
 | ||||||
| "@expo/config@~8.5.0": | "@expo/config@~8.5.0": | ||||||
|   version "8.5.0" |   version "8.5.0" | ||||||
|   resolved "https://registry.yarnpkg.com/@expo/config/-/config-8.5.0.tgz#c618e016c3272335e33fec02fb7fd67e4dcc3342" |   resolved "https://registry.yarnpkg.com/@expo/config/-/config-8.5.0.tgz#c618e016c3272335e33fec02fb7fd67e4dcc3342" | ||||||
|  | @ -3259,6 +3302,15 @@ | ||||||
|     semver "7.3.2" |     semver "7.3.2" | ||||||
|     tempy "0.3.0" |     tempy "0.3.0" | ||||||
| 
 | 
 | ||||||
|  | "@expo/json-file@8.2.36": | ||||||
|  |   version "8.2.36" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@expo/json-file/-/json-file-8.2.36.tgz#62a505cb7f30a34d097386476794680a3f7385ff" | ||||||
|  |   integrity sha512-tOZfTiIFA5KmMpdW9KF7bc6CFiGjb0xnbieJhTGlHrLL+ps2G0OkqmuZ3pFEXBOMnJYUVpnSy++52LFxvpa5ZQ== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/code-frame" "~7.10.4" | ||||||
|  |     json5 "^1.0.1" | ||||||
|  |     write-file-atomic "^2.3.0" | ||||||
|  | 
 | ||||||
| "@expo/json-file@^8.2.37": | "@expo/json-file@^8.2.37": | ||||||
|   version "8.2.37" |   version "8.2.37" | ||||||
|   resolved "https://registry.yarnpkg.com/@expo/json-file/-/json-file-8.2.37.tgz#9c02d3b42134907c69cc0a027b18671b69344049" |   resolved "https://registry.yarnpkg.com/@expo/json-file/-/json-file-8.2.37.tgz#9c02d3b42134907c69cc0a027b18671b69344049" | ||||||
|  | @ -3328,6 +3380,15 @@ | ||||||
|     split "^1.0.1" |     split "^1.0.1" | ||||||
|     sudo-prompt "9.1.1" |     sudo-prompt "9.1.1" | ||||||
| 
 | 
 | ||||||
|  | "@expo/plist@0.0.18": | ||||||
|  |   version "0.0.18" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@expo/plist/-/plist-0.0.18.tgz#9abcde78df703a88f6d9fa1a557ee2f045d178b0" | ||||||
|  |   integrity sha512-+48gRqUiz65R21CZ/IXa7RNBXgAI/uPSdvJqoN9x1hfL44DNbUoWHgHiEXTx7XelcATpDwNTz6sHLfy0iNqf+w== | ||||||
|  |   dependencies: | ||||||
|  |     "@xmldom/xmldom" "~0.7.0" | ||||||
|  |     base64-js "^1.2.3" | ||||||
|  |     xmlbuilder "^14.0.0" | ||||||
|  | 
 | ||||||
| "@expo/plist@^0.1.0": | "@expo/plist@^0.1.0": | ||||||
|   version "0.1.0" |   version "0.1.0" | ||||||
|   resolved "https://registry.yarnpkg.com/@expo/plist/-/plist-0.1.0.tgz#eabc95f951d14e10c87fd0443ee01d567371f058" |   resolved "https://registry.yarnpkg.com/@expo/plist/-/plist-0.1.0.tgz#eabc95f951d14e10c87fd0443ee01d567371f058" | ||||||
|  | @ -4467,6 +4528,18 @@ | ||||||
|     "@radix-ui/react-use-callback-ref" "1.0.1" |     "@radix-ui/react-use-callback-ref" "1.0.1" | ||||||
|     "@radix-ui/react-use-escape-keydown" "1.0.3" |     "@radix-ui/react-use-escape-keydown" "1.0.3" | ||||||
| 
 | 
 | ||||||
|  | "@radix-ui/react-dismissable-layer@1.0.5": | ||||||
|  |   version "1.0.5" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@radix-ui/react-dismissable-layer/-/react-dismissable-layer-1.0.5.tgz#3f98425b82b9068dfbab5db5fff3df6ebf48b9d4" | ||||||
|  |   integrity sha512-aJeDjQhywg9LBu2t/At58hCvr7pEm0o2Ke1x33B+MhjNmmZ17sy4KImo0KPLgsnc/zN7GPdce8Cnn0SWvwZO7g== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.13.10" | ||||||
|  |     "@radix-ui/primitive" "1.0.1" | ||||||
|  |     "@radix-ui/react-compose-refs" "1.0.1" | ||||||
|  |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|  |     "@radix-ui/react-use-callback-ref" "1.0.1" | ||||||
|  |     "@radix-ui/react-use-escape-keydown" "1.0.3" | ||||||
|  | 
 | ||||||
| "@radix-ui/react-dropdown-menu@^2.0.1": | "@radix-ui/react-dropdown-menu@^2.0.1": | ||||||
|   version "2.0.5" |   version "2.0.5" | ||||||
|   resolved "https://registry.yarnpkg.com/@radix-ui/react-dropdown-menu/-/react-dropdown-menu-2.0.5.tgz#19bf4de8ffa348b4eb6a86842f14eff93d741170" |   resolved "https://registry.yarnpkg.com/@radix-ui/react-dropdown-menu/-/react-dropdown-menu-2.0.5.tgz#19bf4de8ffa348b4eb6a86842f14eff93d741170" | ||||||
|  | @ -4481,6 +4554,20 @@ | ||||||
|     "@radix-ui/react-primitive" "1.0.3" |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|     "@radix-ui/react-use-controllable-state" "1.0.1" |     "@radix-ui/react-use-controllable-state" "1.0.1" | ||||||
| 
 | 
 | ||||||
|  | "@radix-ui/react-dropdown-menu@^2.0.6": | ||||||
|  |   version "2.0.6" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@radix-ui/react-dropdown-menu/-/react-dropdown-menu-2.0.6.tgz#cdf13c956c5e263afe4e5f3587b3071a25755b63" | ||||||
|  |   integrity sha512-i6TuFOoWmLWq+M/eCLGd/bQ2HfAX1RJgvrBQ6AQLmzfvsLdefxbWu8G9zczcPFfcSPehz9GcpF6K9QYreFV8hA== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.13.10" | ||||||
|  |     "@radix-ui/primitive" "1.0.1" | ||||||
|  |     "@radix-ui/react-compose-refs" "1.0.1" | ||||||
|  |     "@radix-ui/react-context" "1.0.1" | ||||||
|  |     "@radix-ui/react-id" "1.0.1" | ||||||
|  |     "@radix-ui/react-menu" "2.0.6" | ||||||
|  |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|  |     "@radix-ui/react-use-controllable-state" "1.0.1" | ||||||
|  | 
 | ||||||
| "@radix-ui/react-focus-guards@1.0.1": | "@radix-ui/react-focus-guards@1.0.1": | ||||||
|   version "1.0.1" |   version "1.0.1" | ||||||
|   resolved "https://registry.yarnpkg.com/@radix-ui/react-focus-guards/-/react-focus-guards-1.0.1.tgz#1ea7e32092216b946397866199d892f71f7f98ad" |   resolved "https://registry.yarnpkg.com/@radix-ui/react-focus-guards/-/react-focus-guards-1.0.1.tgz#1ea7e32092216b946397866199d892f71f7f98ad" | ||||||
|  | @ -4498,6 +4585,16 @@ | ||||||
|     "@radix-ui/react-primitive" "1.0.3" |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|     "@radix-ui/react-use-callback-ref" "1.0.1" |     "@radix-ui/react-use-callback-ref" "1.0.1" | ||||||
| 
 | 
 | ||||||
|  | "@radix-ui/react-focus-scope@1.0.4": | ||||||
|  |   version "1.0.4" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@radix-ui/react-focus-scope/-/react-focus-scope-1.0.4.tgz#2ac45fce8c5bb33eb18419cdc1905ef4f1906525" | ||||||
|  |   integrity sha512-sL04Mgvf+FmyvZeYfNu1EPAaaxD+aw7cYeIB9L9Fvq8+urhltTRaEo5ysKOpHuKPclsZcSUMKlN05x4u+CINpA== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.13.10" | ||||||
|  |     "@radix-ui/react-compose-refs" "1.0.1" | ||||||
|  |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|  |     "@radix-ui/react-use-callback-ref" "1.0.1" | ||||||
|  | 
 | ||||||
| "@radix-ui/react-id@1.0.1": | "@radix-ui/react-id@1.0.1": | ||||||
|   version "1.0.1" |   version "1.0.1" | ||||||
|   resolved "https://registry.yarnpkg.com/@radix-ui/react-id/-/react-id-1.0.1.tgz#73cdc181f650e4df24f0b6a5b7aa426b912c88c0" |   resolved "https://registry.yarnpkg.com/@radix-ui/react-id/-/react-id-1.0.1.tgz#73cdc181f650e4df24f0b6a5b7aa426b912c88c0" | ||||||
|  | @ -4531,6 +4628,31 @@ | ||||||
|     aria-hidden "^1.1.1" |     aria-hidden "^1.1.1" | ||||||
|     react-remove-scroll "2.5.5" |     react-remove-scroll "2.5.5" | ||||||
| 
 | 
 | ||||||
|  | "@radix-ui/react-menu@2.0.6": | ||||||
|  |   version "2.0.6" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@radix-ui/react-menu/-/react-menu-2.0.6.tgz#2c9e093c1a5d5daa87304b2a2f884e32288ae79e" | ||||||
|  |   integrity sha512-BVkFLS+bUC8HcImkRKPSiVumA1VPOOEC5WBMiT+QAVsPzW1FJzI9KnqgGxVDPBcql5xXrHkD3JOVoXWEXD8SYA== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.13.10" | ||||||
|  |     "@radix-ui/primitive" "1.0.1" | ||||||
|  |     "@radix-ui/react-collection" "1.0.3" | ||||||
|  |     "@radix-ui/react-compose-refs" "1.0.1" | ||||||
|  |     "@radix-ui/react-context" "1.0.1" | ||||||
|  |     "@radix-ui/react-direction" "1.0.1" | ||||||
|  |     "@radix-ui/react-dismissable-layer" "1.0.5" | ||||||
|  |     "@radix-ui/react-focus-guards" "1.0.1" | ||||||
|  |     "@radix-ui/react-focus-scope" "1.0.4" | ||||||
|  |     "@radix-ui/react-id" "1.0.1" | ||||||
|  |     "@radix-ui/react-popper" "1.1.3" | ||||||
|  |     "@radix-ui/react-portal" "1.0.4" | ||||||
|  |     "@radix-ui/react-presence" "1.0.1" | ||||||
|  |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|  |     "@radix-ui/react-roving-focus" "1.0.4" | ||||||
|  |     "@radix-ui/react-slot" "1.0.2" | ||||||
|  |     "@radix-ui/react-use-callback-ref" "1.0.1" | ||||||
|  |     aria-hidden "^1.1.1" | ||||||
|  |     react-remove-scroll "2.5.5" | ||||||
|  | 
 | ||||||
| "@radix-ui/react-popper@1.1.2": | "@radix-ui/react-popper@1.1.2": | ||||||
|   version "1.1.2" |   version "1.1.2" | ||||||
|   resolved "https://registry.yarnpkg.com/@radix-ui/react-popper/-/react-popper-1.1.2.tgz#4c0b96fcd188dc1f334e02dba2d538973ad842e9" |   resolved "https://registry.yarnpkg.com/@radix-ui/react-popper/-/react-popper-1.1.2.tgz#4c0b96fcd188dc1f334e02dba2d538973ad842e9" | ||||||
|  | @ -4548,6 +4670,23 @@ | ||||||
|     "@radix-ui/react-use-size" "1.0.1" |     "@radix-ui/react-use-size" "1.0.1" | ||||||
|     "@radix-ui/rect" "1.0.1" |     "@radix-ui/rect" "1.0.1" | ||||||
| 
 | 
 | ||||||
|  | "@radix-ui/react-popper@1.1.3": | ||||||
|  |   version "1.1.3" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@radix-ui/react-popper/-/react-popper-1.1.3.tgz#24c03f527e7ac348fabf18c89795d85d21b00b42" | ||||||
|  |   integrity sha512-cKpopj/5RHZWjrbF2846jBNacjQVwkP068DfmgrNJXpvVWrOvlAmE9xSiy5OqeE+Gi8D9fP+oDhUnPqNMY8/5w== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.13.10" | ||||||
|  |     "@floating-ui/react-dom" "^2.0.0" | ||||||
|  |     "@radix-ui/react-arrow" "1.0.3" | ||||||
|  |     "@radix-ui/react-compose-refs" "1.0.1" | ||||||
|  |     "@radix-ui/react-context" "1.0.1" | ||||||
|  |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|  |     "@radix-ui/react-use-callback-ref" "1.0.1" | ||||||
|  |     "@radix-ui/react-use-layout-effect" "1.0.1" | ||||||
|  |     "@radix-ui/react-use-rect" "1.0.1" | ||||||
|  |     "@radix-ui/react-use-size" "1.0.1" | ||||||
|  |     "@radix-ui/rect" "1.0.1" | ||||||
|  | 
 | ||||||
| "@radix-ui/react-portal@1.0.3": | "@radix-ui/react-portal@1.0.3": | ||||||
|   version "1.0.3" |   version "1.0.3" | ||||||
|   resolved "https://registry.yarnpkg.com/@radix-ui/react-portal/-/react-portal-1.0.3.tgz#ffb961244c8ed1b46f039e6c215a6c4d9989bda1" |   resolved "https://registry.yarnpkg.com/@radix-ui/react-portal/-/react-portal-1.0.3.tgz#ffb961244c8ed1b46f039e6c215a6c4d9989bda1" | ||||||
|  | @ -4556,6 +4695,14 @@ | ||||||
|     "@babel/runtime" "^7.13.10" |     "@babel/runtime" "^7.13.10" | ||||||
|     "@radix-ui/react-primitive" "1.0.3" |     "@radix-ui/react-primitive" "1.0.3" | ||||||
| 
 | 
 | ||||||
|  | "@radix-ui/react-portal@1.0.4": | ||||||
|  |   version "1.0.4" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@radix-ui/react-portal/-/react-portal-1.0.4.tgz#df4bfd353db3b1e84e639e9c63a5f2565fb00e15" | ||||||
|  |   integrity sha512-Qki+C/EuGUVCQTOTD5vzJzJuMUlewbzuKyUy+/iHM2uwGiru9gZeBJtHAPKAEkB5KWGi9mP/CHKcY0wt1aW45Q== | ||||||
|  |   dependencies: | ||||||
|  |     "@babel/runtime" "^7.13.10" | ||||||
|  |     "@radix-ui/react-primitive" "1.0.3" | ||||||
|  | 
 | ||||||
| "@radix-ui/react-presence@1.0.1": | "@radix-ui/react-presence@1.0.1": | ||||||
|   version "1.0.1" |   version "1.0.1" | ||||||
|   resolved "https://registry.yarnpkg.com/@radix-ui/react-presence/-/react-presence-1.0.1.tgz#491990ba913b8e2a5db1b06b203cb24b5cdef9ba" |   resolved "https://registry.yarnpkg.com/@radix-ui/react-presence/-/react-presence-1.0.1.tgz#491990ba913b8e2a5db1b06b203cb24b5cdef9ba" | ||||||
|  | @ -4664,6 +4811,13 @@ | ||||||
|   dependencies: |   dependencies: | ||||||
|     merge-options "^3.0.4" |     merge-options "^3.0.4" | ||||||
| 
 | 
 | ||||||
|  | "@react-native-async-storage/async-storage@^1.15.2": | ||||||
|  |   version "1.22.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/@react-native-async-storage/async-storage/-/async-storage-1.22.0.tgz#202a9afd15a5b829c39b709d0ca3942612441efc" | ||||||
|  |   integrity sha512-b5KD010iiZnot86RbAaHpLuHwmPW2qA3SSN/OSZhd1kBoINEQEVBuv+uFtcaTxAhX27bT0wd13GOb2IOSDUXSA== | ||||||
|  |   dependencies: | ||||||
|  |     merge-options "^3.0.4" | ||||||
|  | 
 | ||||||
| "@react-native-camera-roll/camera-roll@^5.2.2": | "@react-native-camera-roll/camera-roll@^5.2.2": | ||||||
|   version "5.7.2" |   version "5.7.2" | ||||||
|   resolved "https://registry.yarnpkg.com/@react-native-camera-roll/camera-roll/-/camera-roll-5.7.2.tgz#db11525ae26c8a61630c424aebd323a7c784a921" |   resolved "https://registry.yarnpkg.com/@react-native-camera-roll/camera-roll/-/camera-roll-5.7.2.tgz#db11525ae26c8a61630c424aebd323a7c784a921" | ||||||
|  | @ -8038,7 +8192,7 @@ | ||||||
|   resolved "https://registry.yarnpkg.com/@xmldom/xmldom/-/xmldom-0.8.10.tgz#a1337ca426aa61cef9fe15b5b28e340a72f6fa99" |   resolved "https://registry.yarnpkg.com/@xmldom/xmldom/-/xmldom-0.8.10.tgz#a1337ca426aa61cef9fe15b5b28e340a72f6fa99" | ||||||
|   integrity sha512-2WALfTl4xo2SkGCYRt6rDTFfk9R1czmBvUQy12gK2KuRKIpWEhcbbzy8EZXtz/jkRqHX8bFEc6FC1HjX4TUWYw== |   integrity sha512-2WALfTl4xo2SkGCYRt6rDTFfk9R1czmBvUQy12gK2KuRKIpWEhcbbzy8EZXtz/jkRqHX8bFEc6FC1HjX4TUWYw== | ||||||
| 
 | 
 | ||||||
| "@xmldom/xmldom@~0.7.7": | "@xmldom/xmldom@~0.7.0", "@xmldom/xmldom@~0.7.7": | ||||||
|   version "0.7.13" |   version "0.7.13" | ||||||
|   resolved "https://registry.yarnpkg.com/@xmldom/xmldom/-/xmldom-0.7.13.tgz#ff34942667a4e19a9f4a0996a76814daac364cf3" |   resolved "https://registry.yarnpkg.com/@xmldom/xmldom/-/xmldom-0.7.13.tgz#ff34942667a4e19a9f4a0996a76814daac364cf3" | ||||||
|   integrity sha512-lm2GW5PkosIzccsaZIz7tp8cPADSIlIHWDFTR1N0SzfinhhYgeIQjFMz4rYzanCScr3DqQLeomUDArp6MWKm+g== |   integrity sha512-lm2GW5PkosIzccsaZIz7tp8cPADSIlIHWDFTR1N0SzfinhhYgeIQjFMz4rYzanCScr3DqQLeomUDArp6MWKm+g== | ||||||
|  | @ -11651,6 +11805,14 @@ expo-camera@~14.0.1: | ||||||
|   dependencies: |   dependencies: | ||||||
|     invariant "^2.2.4" |     invariant "^2.2.4" | ||||||
| 
 | 
 | ||||||
|  | expo-constants@^13.0.2: | ||||||
|  |   version "13.2.4" | ||||||
|  |   resolved "https://registry.yarnpkg.com/expo-constants/-/expo-constants-13.2.4.tgz#eab4a553f074b2c60ad7a158d3b82e3484a94606" | ||||||
|  |   integrity sha512-Zobau8EuTk2GgafwkfGnWM6CmSLB7X8qnQXVuXe0nd3v92hfQUmRWGhJwH88uxXj3LrfqctM6PaJ8taG1vxfBw== | ||||||
|  |   dependencies: | ||||||
|  |     "@expo/config" "~7.0.0" | ||||||
|  |     uuid "^3.3.2" | ||||||
|  | 
 | ||||||
| expo-constants@~15.4.0: | expo-constants@~15.4.0: | ||||||
|   version "15.4.1" |   version "15.4.1" | ||||||
|   resolved "https://registry.yarnpkg.com/expo-constants/-/expo-constants-15.4.1.tgz#f76f347cf687b6630e1e3b9a385a4e42771671a4" |   resolved "https://registry.yarnpkg.com/expo-constants/-/expo-constants-15.4.1.tgz#f76f347cf687b6630e1e3b9a385a4e42771671a4" | ||||||
|  | @ -11700,6 +11862,13 @@ expo-dev-menu@4.5.3: | ||||||
|     expo-dev-menu-interface "1.7.2" |     expo-dev-menu-interface "1.7.2" | ||||||
|     semver "^7.5.3" |     semver "^7.5.3" | ||||||
| 
 | 
 | ||||||
|  | expo-device@~4.1.1: | ||||||
|  |   version "4.1.1" | ||||||
|  |   resolved "https://registry.yarnpkg.com/expo-device/-/expo-device-4.1.1.tgz#5de94144113ffb7fa0f37fa3d70e452113954c10" | ||||||
|  |   integrity sha512-It0SGtKcvzQSf+Co6zdPdB63zZvG2/rDolB1lqswMNKj03Y7KVU41s5tcQCqNczj7tmeN3CJy7A8YhYGKdb7gA== | ||||||
|  |   dependencies: | ||||||
|  |     ua-parser-js "^0.7.19" | ||||||
|  | 
 | ||||||
| expo-device@~5.9.2: | expo-device@~5.9.2: | ||||||
|   version "5.9.2" |   version "5.9.2" | ||||||
|   resolved "https://registry.yarnpkg.com/expo-device/-/expo-device-5.9.2.tgz#697e96f52d213a141b6f265f1e274e9d5e98c92c" |   resolved "https://registry.yarnpkg.com/expo-device/-/expo-device-5.9.2.tgz#697e96f52d213a141b6f265f1e274e9d5e98c92c" | ||||||
|  | @ -14892,6 +15061,16 @@ js-queue@2.0.2: | ||||||
|   dependencies: |   dependencies: | ||||||
|     easy-stack "^1.0.1" |     easy-stack "^1.0.1" | ||||||
| 
 | 
 | ||||||
|  | js-sha256@^0.10.1: | ||||||
|  |   version "0.10.1" | ||||||
|  |   resolved "https://registry.yarnpkg.com/js-sha256/-/js-sha256-0.10.1.tgz#b40104ba1368e823fdd5f41b66b104b15a0da60d" | ||||||
|  |   integrity sha512-5obBtsz9301ULlsgggLg542s/jqtddfOpV5KJc4hajc9JV8GeY2gZHSVpYBn4nWqAUTJ9v+xwtbJ1mIBgIH5Vw== | ||||||
|  | 
 | ||||||
|  | js-sha256@^0.11.0: | ||||||
|  |   version "0.11.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/js-sha256/-/js-sha256-0.11.0.tgz#256a921d9292f7fe98905face82e367abaca9576" | ||||||
|  |   integrity sha512-6xNlKayMZvds9h1Y1VWc0fQHQ82BxTXizWPEtEeGvmOUYpBRy4gbWroHLpzowe6xiQhHpelCQiE7HEdznyBL9Q== | ||||||
|  | 
 | ||||||
| js-sha256@^0.9.0: | js-sha256@^0.9.0: | ||||||
|   version "0.9.0" |   version "0.9.0" | ||||||
|   resolved "https://registry.yarnpkg.com/js-sha256/-/js-sha256-0.9.0.tgz#0b89ac166583e91ef9123644bd3c5334ce9d0966" |   resolved "https://registry.yarnpkg.com/js-sha256/-/js-sha256-0.9.0.tgz#0b89ac166583e91ef9123644bd3c5334ce9d0966" | ||||||
|  | @ -15076,7 +15255,7 @@ json-stable-stringify-without-jsonify@^1.0.1: | ||||||
|   resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651" |   resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651" | ||||||
|   integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== |   integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== | ||||||
| 
 | 
 | ||||||
| json5@^1.0.2: | json5@^1.0.1, json5@^1.0.2: | ||||||
|   version "1.0.2" |   version "1.0.2" | ||||||
|   resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593" |   resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593" | ||||||
|   integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA== |   integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA== | ||||||
|  | @ -18372,6 +18551,13 @@ react-is@^17.0.1: | ||||||
|   resolved "https://registry.yarnpkg.com/react-is/-/react-is-17.0.2.tgz#e691d4a8e9c789365655539ab372762b0efb54f0" |   resolved "https://registry.yarnpkg.com/react-is/-/react-is-17.0.2.tgz#e691d4a8e9c789365655539ab372762b0efb54f0" | ||||||
|   integrity sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w== |   integrity sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w== | ||||||
| 
 | 
 | ||||||
|  | react-keyed-flatten-children@^3.0.0: | ||||||
|  |   version "3.0.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/react-keyed-flatten-children/-/react-keyed-flatten-children-3.0.0.tgz#b6ad0bde437d3ab86c8af3a1902d164be2a29d67" | ||||||
|  |   integrity sha512-tSH6gvOyQjt3qtjG+kU9sTypclL1672yjpVufcE3aHNM0FhvjBUQZqsb/awIux4zEuVC3k/DP4p0GdTT/QUt/Q== | ||||||
|  |   dependencies: | ||||||
|  |     react-is "^18.2.0" | ||||||
|  | 
 | ||||||
| react-native-appstate-hook@^1.0.6: | react-native-appstate-hook@^1.0.6: | ||||||
|   version "1.0.6" |   version "1.0.6" | ||||||
|   resolved "https://registry.yarnpkg.com/react-native-appstate-hook/-/react-native-appstate-hook-1.0.6.tgz#cbc16e7b89cfaea034cabd999f00e99053cabd06" |   resolved "https://registry.yarnpkg.com/react-native-appstate-hook/-/react-native-appstate-hook-1.0.6.tgz#cbc16e7b89cfaea034cabd999f00e99053cabd06" | ||||||
|  | @ -18410,6 +18596,13 @@ react-native-gesture-handler@~2.14.0: | ||||||
|     lodash "^4.17.21" |     lodash "^4.17.21" | ||||||
|     prop-types "^15.7.2" |     prop-types "^15.7.2" | ||||||
| 
 | 
 | ||||||
|  | react-native-get-random-values@^1.6.0: | ||||||
|  |   version "1.10.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/react-native-get-random-values/-/react-native-get-random-values-1.10.0.tgz#c2c5f12a4ef8b1175145347b4a4b9f9a40d9ffc8" | ||||||
|  |   integrity sha512-gZ1zbXhbb8+Jy9qYTV8c4Nf45/VB4g1jmXuavY5rPfUn7x3ok9Vl3FTl0dnE92Z4FFtfbUNNwtSfcmomdtWg+A== | ||||||
|  |   dependencies: | ||||||
|  |     fast-base64-decode "^1.0.0" | ||||||
|  | 
 | ||||||
| react-native-get-random-values@~1.8.0: | react-native-get-random-values@~1.8.0: | ||||||
|   version "1.8.0" |   version "1.8.0" | ||||||
|   resolved "https://registry.yarnpkg.com/react-native-get-random-values/-/react-native-get-random-values-1.8.0.tgz#1cb4bd4bd3966a356e59697b8f372999fe97cb16" |   resolved "https://registry.yarnpkg.com/react-native-get-random-values/-/react-native-get-random-values-1.8.0.tgz#1cb4bd4bd3966a356e59697b8f372999fe97cb16" | ||||||
|  | @ -19883,6 +20076,49 @@ standard-as-callback@^2.1.0: | ||||||
|   resolved "https://registry.yarnpkg.com/standard-as-callback/-/standard-as-callback-2.1.0.tgz#8953fc05359868a77b5b9739a665c5977bb7df45" |   resolved "https://registry.yarnpkg.com/standard-as-callback/-/standard-as-callback-2.1.0.tgz#8953fc05359868a77b5b9739a665c5977bb7df45" | ||||||
|   integrity sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A== |   integrity sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A== | ||||||
| 
 | 
 | ||||||
|  | statsig-js@4.45.1: | ||||||
|  |   version "4.45.1" | ||||||
|  |   resolved "https://registry.yarnpkg.com/statsig-js/-/statsig-js-4.45.1.tgz#b1f5b9c52adc4a8aece376fb011416c89227932f" | ||||||
|  |   integrity sha512-h94RzFQsJCQCNwQXpZ9OBXcvCxDnkXF6OrCekd81ySvY2l4JSowpxMWX3Iw6IDFzfTfKdER9JQzFLhMSQbT+YQ== | ||||||
|  |   dependencies: | ||||||
|  |     js-sha256 "^0.10.1" | ||||||
|  |     uuid "^8.3.2" | ||||||
|  | 
 | ||||||
|  | statsig-js@4.49.0: | ||||||
|  |   version "4.49.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/statsig-js/-/statsig-js-4.49.0.tgz#8470a9ac218a93d36f4b7b306ff9377e48064740" | ||||||
|  |   integrity sha512-N4drx6fzI168Q4NndFY3IJbSDqpWSBWvS290H/RnT/g3Et58SvtXzG5qqgzmqy4CwcmwH+IL8K15pL7hPnfvUQ== | ||||||
|  |   dependencies: | ||||||
|  |     js-sha256 "^0.11.0" | ||||||
|  |     uuid "^8.3.2" | ||||||
|  | 
 | ||||||
|  | statsig-react-native-expo@^4.6.1: | ||||||
|  |   version "4.6.1" | ||||||
|  |   resolved "https://registry.yarnpkg.com/statsig-react-native-expo/-/statsig-react-native-expo-4.6.1.tgz#0bdf49fee7112f7f28bff2405f4ba0c1727bb3d6" | ||||||
|  |   integrity sha512-rB60c+WSrQPmjW9j75d+acUtwSOe38PE2KTDHiOv1Mf+0TCcFtGYlJmKCibWvbeXR7ZAyjjGeroh23bCSEZauQ== | ||||||
|  |   dependencies: | ||||||
|  |     "@react-native-async-storage/async-storage" "^1.15.2" | ||||||
|  |     expo-constants "^13.0.2" | ||||||
|  |     expo-device "~4.1.1" | ||||||
|  |     js-sha256 "^0.9.0" | ||||||
|  |     react-native-get-random-values "^1.6.0" | ||||||
|  |     statsig-react "^1.21.1" | ||||||
|  |     uuid "^8.3.2" | ||||||
|  | 
 | ||||||
|  | statsig-react@^1.21.1: | ||||||
|  |   version "1.35.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/statsig-react/-/statsig-react-1.35.0.tgz#ad5730b83f564c640623e954fcbcbe848e939946" | ||||||
|  |   integrity sha512-KLN7dhq6FvAl25Z0QN6IINFBgM3yn0GMafoE698tYZqRf911xvevFaR7qUXiTz3W9vmFYrmFRouqVMfCv7DW0A== | ||||||
|  |   dependencies: | ||||||
|  |     statsig-js "4.45.1" | ||||||
|  | 
 | ||||||
|  | statsig-react@^1.36.0: | ||||||
|  |   version "1.36.0" | ||||||
|  |   resolved "https://registry.yarnpkg.com/statsig-react/-/statsig-react-1.36.0.tgz#c2171268a6c76eee534849ec9556b836baba04b6" | ||||||
|  |   integrity sha512-QcTHla3ypfn2RvrnHGNlqWbiC2W/ZjcMM5LT6ExNV4skH7Xhspto3dMS3JVzBhOb74OEDZK4DbxQj9Wdz6XW0w== | ||||||
|  |   dependencies: | ||||||
|  |     statsig-js "4.49.0" | ||||||
|  | 
 | ||||||
| statuses@2.0.1: | statuses@2.0.1: | ||||||
|   version "2.0.1" |   version "2.0.1" | ||||||
|   resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" |   resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" | ||||||
|  | @ -20827,6 +21063,11 @@ typescript@^5.3.3: | ||||||
|   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.3.3.tgz#b3ce6ba258e72e6305ba66f5c9b452aaee3ffe37" |   resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.3.3.tgz#b3ce6ba258e72e6305ba66f5c9b452aaee3ffe37" | ||||||
|   integrity sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw== |   integrity sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw== | ||||||
| 
 | 
 | ||||||
|  | ua-parser-js@^0.7.19: | ||||||
|  |   version "0.7.37" | ||||||
|  |   resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.37.tgz#e464e66dac2d33a7a1251d7d7a99d6157ec27832" | ||||||
|  |   integrity sha512-xV8kqRKM+jhMvcHWUKthV9fNebIzrNy//2O9ZwWcfiBFR5f25XVZPLlEajk/sf3Ra15V92isyQqnIEXRDaZWEA== | ||||||
|  | 
 | ||||||
| ua-parser-js@^0.7.33: | ua-parser-js@^0.7.33: | ||||||
|   version "0.7.35" |   version "0.7.35" | ||||||
|   resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.35.tgz#8bda4827be4f0b1dda91699a29499575a1f1d307" |   resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.35.tgz#8bda4827be4f0b1dda91699a29499575a1f1d307" | ||||||
|  | @ -21108,7 +21349,7 @@ utils-merge@1.0.1: | ||||||
|   resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" |   resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" | ||||||
|   integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA== |   integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA== | ||||||
| 
 | 
 | ||||||
| uuid@^3.0.1: | uuid@^3.0.1, uuid@^3.3.2: | ||||||
|   version "3.4.0" |   version "3.4.0" | ||||||
|   resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.4.0.tgz#b23e4358afa8a202fe7a100af1f5f883f02007ee" |   resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.4.0.tgz#b23e4358afa8a202fe7a100af1f5f883f02007ee" | ||||||
|   integrity sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A== |   integrity sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A== | ||||||
|  | @ -21860,6 +22101,14 @@ xml-name-validator@^4.0.0: | ||||||
|   resolved "https://registry.yarnpkg.com/xml-name-validator/-/xml-name-validator-4.0.0.tgz#79a006e2e63149a8600f15430f0a4725d1524835" |   resolved "https://registry.yarnpkg.com/xml-name-validator/-/xml-name-validator-4.0.0.tgz#79a006e2e63149a8600f15430f0a4725d1524835" | ||||||
|   integrity sha512-ICP2e+jsHvAj2E2lIHxa5tjXRlKDJo4IdvPvCXbXQGdzSfmSpNVyIKMvoZHjDY9DP0zV17iI85o90vRFXNccRw== |   integrity sha512-ICP2e+jsHvAj2E2lIHxa5tjXRlKDJo4IdvPvCXbXQGdzSfmSpNVyIKMvoZHjDY9DP0zV17iI85o90vRFXNccRw== | ||||||
| 
 | 
 | ||||||
|  | xml2js@0.4.23: | ||||||
|  |   version "0.4.23" | ||||||
|  |   resolved "https://registry.yarnpkg.com/xml2js/-/xml2js-0.4.23.tgz#a0c69516752421eb2ac758ee4d4ccf58843eac66" | ||||||
|  |   integrity sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug== | ||||||
|  |   dependencies: | ||||||
|  |     sax ">=0.6.0" | ||||||
|  |     xmlbuilder "~11.0.0" | ||||||
|  | 
 | ||||||
| xml2js@0.6.0: | xml2js@0.6.0: | ||||||
|   version "0.6.0" |   version "0.6.0" | ||||||
|   resolved "https://registry.yarnpkg.com/xml2js/-/xml2js-0.6.0.tgz#07afc447a97d2bd6507a1f76eeadddb09f7a8282" |   resolved "https://registry.yarnpkg.com/xml2js/-/xml2js-0.6.0.tgz#07afc447a97d2bd6507a1f76eeadddb09f7a8282" | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue