forked from Expensify/App
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[TS migration] Migrate 'ScreenWrapper' component to TypeScript
- Loading branch information
1 parent
27a51a4
commit 3409b2c
Showing
3 changed files
with
252 additions
and
263 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,252 @@ | ||
import {useNavigation} from '@react-navigation/native'; | ||
import {StackNavigationProp} from '@react-navigation/stack'; | ||
import React, {ForwardedRef, forwardRef, ReactNode, useEffect, useRef, useState} from 'react'; | ||
import {DimensionValue, Keyboard, PanResponder, StyleProp, View, ViewStyle} from 'react-native'; | ||
import {PickerAvoidingView} from 'react-native-picker-select'; | ||
import {EdgeInsets} from 'react-native-safe-area-context'; | ||
import useEnvironment from '@hooks/useEnvironment'; | ||
import useInitialDimensions from '@hooks/useInitialWindowDimensions'; | ||
import useKeyboardState from '@hooks/useKeyboardState'; | ||
import useNetwork from '@hooks/useNetwork'; | ||
import useThemeStyles from '@hooks/useThemeStyles'; | ||
import useWindowDimensions from '@hooks/useWindowDimensions'; | ||
import * as Browser from '@libs/Browser'; | ||
import {RootStackParamList} from '@libs/Navigation/types'; | ||
import toggleTestToolsModal from '@userActions/TestTool'; | ||
import CONST from '@src/CONST'; | ||
import CustomDevMenu from './CustomDevMenu'; | ||
import HeaderGap from './HeaderGap'; | ||
import KeyboardAvoidingView from './KeyboardAvoidingView'; | ||
import OfflineIndicator from './OfflineIndicator'; | ||
import SafeAreaConsumer from './SafeAreaConsumer'; | ||
import TestToolsModal from './TestToolsModal'; | ||
|
||
type ChildrenProps = { | ||
insets?: EdgeInsets; | ||
safeAreaPaddingBottomStyle?: { | ||
paddingBottom?: DimensionValue; | ||
}; | ||
didScreenTransitionEnd: boolean; | ||
}; | ||
|
||
type ScreenWrapperProps = { | ||
/** Returns a function as a child to pass insets to or a node to render without insets */ | ||
children: ReactNode | React.FC<ChildrenProps>; | ||
|
||
/** A unique ID to find the screen wrapper in tests */ | ||
testID: string; | ||
|
||
/** Additional styles to add */ | ||
style?: StyleProp<ViewStyle>; | ||
|
||
/** Additional styles for header gap */ | ||
headerGapStyles?: StyleProp<ViewStyle>; | ||
|
||
/** Styles for the offline indicator */ | ||
offlineIndicatorStyle?: StyleProp<ViewStyle>; | ||
|
||
/** Whether to include padding bottom */ | ||
includeSafeAreaPaddingBottom?: boolean; | ||
|
||
/** Whether to include padding top */ | ||
includePaddingTop?: boolean; | ||
|
||
/** Called when navigated Screen's transition is finished. It does not fire when user exit the page. */ | ||
onEntryTransitionEnd?: () => void; | ||
|
||
/** The behavior to pass to the KeyboardAvoidingView, requires some trial and error depending on the layout/devices used. | ||
* Search 'switch(behavior)' in ./node_modules/react-native/Libraries/Components/Keyboard/KeyboardAvoidingView.js for more context */ | ||
keyboardAvoidingViewBehavior?: 'padding' | 'height' | 'position'; | ||
|
||
/** Whether KeyboardAvoidingView should be enabled. Use false for screens where this functionality is not necessary */ | ||
shouldEnableKeyboardAvoidingView?: boolean; | ||
|
||
/** Whether picker modal avoiding should be enabled. Should be enabled when there's a picker at the bottom of a | ||
* scrollable form, gives a subtly better UX if disabled on non-scrollable screens with a submit button */ | ||
shouldEnablePickerAvoiding?: boolean; | ||
|
||
/** Whether to dismiss keyboard before leaving a screen */ | ||
shouldDismissKeyboardBeforeClose?: boolean; | ||
|
||
/** Whether to use the maxHeight (true) or use the 100% of the height (false) */ | ||
shouldEnableMaxHeight?: boolean; | ||
|
||
/** Whether to use the minHeight. Use true for screens where the window height are changing because of Virtual Keyboard */ | ||
shouldEnableMinHeight?: boolean; | ||
|
||
/** Whether to show offline indicator */ | ||
shouldShowOfflineIndicator?: boolean; | ||
|
||
/** | ||
* The navigation prop is passed by the navigator. It is used to trigger the onEntryTransitionEnd callback | ||
* when the screen transition ends. | ||
* | ||
* This is required because transitionEnd event doesn't trigger in the testing environment. | ||
*/ | ||
navigation?: StackNavigationProp<RootStackParamList>; | ||
}; | ||
|
||
function ScreenWrapper( | ||
{ | ||
shouldEnableMaxHeight = false, | ||
shouldEnableMinHeight, | ||
includePaddingTop = true, | ||
keyboardAvoidingViewBehavior = 'padding', | ||
includeSafeAreaPaddingBottom = true, | ||
shouldEnableKeyboardAvoidingView = true, | ||
shouldEnablePickerAvoiding = true, | ||
headerGapStyles, | ||
children, | ||
shouldShowOfflineIndicator = true, | ||
offlineIndicatorStyle, | ||
style, | ||
shouldDismissKeyboardBeforeClose = true, | ||
onEntryTransitionEnd, | ||
testID, | ||
navigation: navigationProp, | ||
}: ScreenWrapperProps, | ||
ref: ForwardedRef<View>, | ||
) { | ||
/** | ||
* We are only passing navigation as prop from | ||
* ReportScreenWrapper -> ReportScreen -> ScreenWrapper | ||
* | ||
* so in other places where ScreenWrapper is used, we need to | ||
* fallback to useNavigation. | ||
*/ | ||
const navigationFallback = useNavigation<StackNavigationProp<RootStackParamList>>(); | ||
const navigation = navigationProp ?? navigationFallback; | ||
const {windowHeight, isSmallScreenWidth} = useWindowDimensions(); | ||
const {initialHeight} = useInitialDimensions(); | ||
const styles = useThemeStyles(); | ||
const keyboardState = useKeyboardState(); | ||
const {isDevelopment} = useEnvironment(); | ||
const {isOffline} = useNetwork(); | ||
const [didScreenTransitionEnd, setDidScreenTransitionEnd] = useState(false); | ||
const maxHeight = shouldEnableMaxHeight ? windowHeight : undefined; | ||
const minHeight = shouldEnableMinHeight && !Browser.isSafari() ? initialHeight : undefined; | ||
const isKeyboardShown = keyboardState?.isKeyboardShown ?? false; | ||
|
||
const isKeyboardShownRef = useRef<boolean>(false); | ||
|
||
isKeyboardShownRef.current = keyboardState?.isKeyboardShown ?? false; | ||
|
||
const panResponder = useRef( | ||
PanResponder.create({ | ||
// eslint-disable-next-line @typescript-eslint/naming-convention | ||
onStartShouldSetPanResponderCapture: (_e, gestureState) => gestureState.numberActiveTouches === CONST.TEST_TOOL.NUMBER_OF_TAPS, | ||
onPanResponderRelease: toggleTestToolsModal, | ||
}), | ||
).current; | ||
|
||
const keyboardDissmissPanResponder = useRef( | ||
PanResponder.create({ | ||
// eslint-disable-next-line @typescript-eslint/naming-convention | ||
onMoveShouldSetPanResponderCapture: (_e, gestureState) => { | ||
const isHorizontalSwipe = Math.abs(gestureState.dx) > Math.abs(gestureState.dy); | ||
const shouldDismissKeyboard = shouldDismissKeyboardBeforeClose && isKeyboardShown && Browser.isMobile(); | ||
|
||
return isHorizontalSwipe && shouldDismissKeyboard; | ||
}, | ||
onPanResponderGrant: Keyboard.dismiss, | ||
}), | ||
).current; | ||
|
||
useEffect(() => { | ||
const unsubscribeTransitionEnd = navigation.addListener('transitionEnd', (event) => { | ||
// Prevent firing the prop callback when user is exiting the page. | ||
if (event?.data?.closing) { | ||
return; | ||
} | ||
|
||
setDidScreenTransitionEnd(true); | ||
onEntryTransitionEnd?.(); | ||
}); | ||
|
||
// We need to have this prop to remove keyboard before going away from the screen, to avoid previous screen look weird for a brief moment, | ||
// also we need to have generic control in future - to prevent closing keyboard for some rare cases in which beforeRemove has limitations | ||
// described here https://reactnavigation.org/docs/preventing-going-back/#limitations | ||
const beforeRemoveSubscription = shouldDismissKeyboardBeforeClose | ||
? navigation.addListener('beforeRemove', () => { | ||
if (!isKeyboardShownRef.current) { | ||
return; | ||
} | ||
Keyboard.dismiss(); | ||
}) | ||
: undefined; | ||
|
||
return () => { | ||
unsubscribeTransitionEnd(); | ||
|
||
if (beforeRemoveSubscription) { | ||
beforeRemoveSubscription(); | ||
} | ||
}; | ||
// Rule disabled because this effect is only for component did mount & will component unmount lifecycle event | ||
// eslint-disable-next-line react-hooks/exhaustive-deps | ||
}, []); | ||
|
||
return ( | ||
<SafeAreaConsumer> | ||
{({insets, paddingTop, paddingBottom, safeAreaPaddingBottomStyle}) => { | ||
const paddingStyle: StyleProp<ViewStyle> = {}; | ||
|
||
if (includePaddingTop) { | ||
paddingStyle.paddingTop = paddingTop; | ||
} | ||
|
||
// We always need the safe area padding bottom if we're showing the offline indicator since it is bottom-docked. | ||
if (includeSafeAreaPaddingBottom || (isOffline && shouldShowOfflineIndicator)) { | ||
paddingStyle.paddingBottom = paddingBottom; | ||
} | ||
|
||
return ( | ||
<View | ||
ref={ref} | ||
style={[styles.flex1, {minHeight}]} | ||
// eslint-disable-next-line react/jsx-props-no-spreading | ||
{...(isDevelopment ? panResponder.panHandlers : {})} | ||
testID={testID} | ||
> | ||
<View | ||
style={[styles.flex1, paddingStyle, style]} | ||
// eslint-disable-next-line react/jsx-props-no-spreading | ||
{...keyboardDissmissPanResponder.panHandlers} | ||
> | ||
<KeyboardAvoidingView | ||
style={[styles.w100, styles.h100, {maxHeight}]} | ||
behavior={keyboardAvoidingViewBehavior} | ||
enabled={shouldEnableKeyboardAvoidingView} | ||
> | ||
<PickerAvoidingView | ||
// @ts-expect-error Remove once react-native-picker-select is updated | ||
style={styles.flex1} | ||
enabled={shouldEnablePickerAvoiding} | ||
> | ||
<HeaderGap styles={headerGapStyles} /> | ||
{isDevelopment && <TestToolsModal />} | ||
{isDevelopment && <CustomDevMenu />} | ||
{ | ||
// If props.children is a function, call it to provide the insets to the children. | ||
typeof children === 'function' | ||
? children({ | ||
insets, | ||
safeAreaPaddingBottomStyle, | ||
didScreenTransitionEnd, | ||
}) | ||
: children | ||
} | ||
{isSmallScreenWidth && shouldShowOfflineIndicator && <OfflineIndicator style={offlineIndicatorStyle} />} | ||
</PickerAvoidingView> | ||
</KeyboardAvoidingView> | ||
</View> | ||
</View> | ||
); | ||
}} | ||
</SafeAreaConsumer> | ||
); | ||
} | ||
|
||
ScreenWrapper.displayName = 'ScreenWrapper'; | ||
|
||
export default forwardRef(ScreenWrapper); |
Oops, something went wrong.