* Fix (#4430): Use separate hooks for shell mode animated styles * Consolidate in one file --------- Co-authored-by: Dan Abramov <dan.abramov@gmail.com>
370 lines
13 KiB
TypeScript
370 lines
13 KiB
TypeScript
import React, {ComponentProps} from 'react'
|
|
import {GestureResponderEvent, TouchableOpacity, View} from 'react-native'
|
|
import Animated from 'react-native-reanimated'
|
|
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
|
import {msg, Trans} from '@lingui/macro'
|
|
import {useLingui} from '@lingui/react'
|
|
import {BottomTabBarProps} from '@react-navigation/bottom-tabs'
|
|
import {StackActions} from '@react-navigation/native'
|
|
|
|
import {useAnalytics} from '#/lib/analytics/analytics'
|
|
import {useHaptics} from '#/lib/haptics'
|
|
import {useDedupe} from '#/lib/hooks/useDedupe'
|
|
import {useMinimalShellFooterTransform} from '#/lib/hooks/useMinimalShellTransform'
|
|
import {useNavigationTabState} from '#/lib/hooks/useNavigationTabState'
|
|
import {usePalette} from '#/lib/hooks/usePalette'
|
|
import {clamp} from '#/lib/numbers'
|
|
import {getTabState, TabState} from '#/lib/routes/helpers'
|
|
import {s} from '#/lib/styles'
|
|
import {emitSoftReset} from '#/state/events'
|
|
import {useUnreadMessageCount} from '#/state/queries/messages/list-converations'
|
|
import {useUnreadNotifications} from '#/state/queries/notifications/unread'
|
|
import {useProfileQuery} from '#/state/queries/profile'
|
|
import {useSession} from '#/state/session'
|
|
import {useLoggedOutViewControls} from '#/state/shell/logged-out'
|
|
import {useShellLayout} from '#/state/shell/shell-layout'
|
|
import {useCloseAllActiveElements} from '#/state/util'
|
|
import {Button} from '#/view/com/util/forms/Button'
|
|
import {Text} from '#/view/com/util/text/Text'
|
|
import {UserAvatar} from '#/view/com/util/UserAvatar'
|
|
import {Logo} from '#/view/icons/Logo'
|
|
import {Logotype} from '#/view/icons/Logotype'
|
|
import {useDialogControl} from '#/components/Dialog'
|
|
import {SwitchAccountDialog} from '#/components/dialogs/SwitchAccount'
|
|
import {
|
|
Bell_Filled_Corner0_Rounded as BellFilled,
|
|
Bell_Stroke2_Corner0_Rounded as Bell,
|
|
} from '#/components/icons/Bell'
|
|
import {
|
|
HomeOpen_Filled_Corner0_Rounded as HomeFilled,
|
|
HomeOpen_Stoke2_Corner0_Rounded as Home,
|
|
} from '#/components/icons/HomeOpen'
|
|
import {MagnifyingGlass_Filled_Stroke2_Corner0_Rounded as MagnifyingGlassFilled} from '#/components/icons/MagnifyingGlass'
|
|
import {MagnifyingGlass2_Stroke2_Corner0_Rounded as MagnifyingGlass} from '#/components/icons/MagnifyingGlass2'
|
|
import {
|
|
Message_Stroke2_Corner0_Rounded as Message,
|
|
Message_Stroke2_Corner0_Rounded_Filled as MessageFilled,
|
|
} from '#/components/icons/Message'
|
|
import {styles} from './BottomBarStyles'
|
|
|
|
type TabOptions =
|
|
| 'Home'
|
|
| 'Search'
|
|
| 'Notifications'
|
|
| 'MyProfile'
|
|
| 'Feeds'
|
|
| 'Messages'
|
|
|
|
export function BottomBar({navigation}: BottomTabBarProps) {
|
|
const {hasSession, currentAccount} = useSession()
|
|
const pal = usePalette('default')
|
|
const {_} = useLingui()
|
|
const safeAreaInsets = useSafeAreaInsets()
|
|
const {track} = useAnalytics()
|
|
const {footerHeight} = useShellLayout()
|
|
const {isAtHome, isAtSearch, isAtNotifications, isAtMyProfile, isAtMessages} =
|
|
useNavigationTabState()
|
|
const numUnreadNotifications = useUnreadNotifications()
|
|
const numUnreadMessages = useUnreadMessageCount()
|
|
const footerMinimalShellTransform = useMinimalShellFooterTransform()
|
|
const {data: profile} = useProfileQuery({did: currentAccount?.did})
|
|
const {requestSwitchToAccount} = useLoggedOutViewControls()
|
|
const closeAllActiveElements = useCloseAllActiveElements()
|
|
const dedupe = useDedupe()
|
|
const accountSwitchControl = useDialogControl()
|
|
const playHaptic = useHaptics()
|
|
const iconWidth = 28
|
|
|
|
const showSignIn = React.useCallback(() => {
|
|
closeAllActiveElements()
|
|
requestSwitchToAccount({requestedAccount: 'none'})
|
|
}, [requestSwitchToAccount, closeAllActiveElements])
|
|
|
|
const showCreateAccount = React.useCallback(() => {
|
|
closeAllActiveElements()
|
|
requestSwitchToAccount({requestedAccount: 'new'})
|
|
// setShowLoggedOut(true)
|
|
}, [requestSwitchToAccount, closeAllActiveElements])
|
|
|
|
const onPressTab = React.useCallback(
|
|
(tab: TabOptions) => {
|
|
track(`MobileShell:${tab}ButtonPressed`)
|
|
const state = navigation.getState()
|
|
const tabState = getTabState(state, tab)
|
|
if (tabState === TabState.InsideAtRoot) {
|
|
emitSoftReset()
|
|
} else if (tabState === TabState.Inside) {
|
|
dedupe(() => navigation.dispatch(StackActions.popToTop()))
|
|
} else {
|
|
dedupe(() => navigation.navigate(`${tab}Tab`))
|
|
}
|
|
},
|
|
[track, navigation, dedupe],
|
|
)
|
|
const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab])
|
|
const onPressSearch = React.useCallback(
|
|
() => onPressTab('Search'),
|
|
[onPressTab],
|
|
)
|
|
const onPressNotifications = React.useCallback(
|
|
() => onPressTab('Notifications'),
|
|
[onPressTab],
|
|
)
|
|
const onPressProfile = React.useCallback(() => {
|
|
onPressTab('MyProfile')
|
|
}, [onPressTab])
|
|
const onPressMessages = React.useCallback(() => {
|
|
onPressTab('Messages')
|
|
}, [onPressTab])
|
|
|
|
const onLongPressProfile = React.useCallback(() => {
|
|
playHaptic()
|
|
accountSwitchControl.open()
|
|
}, [accountSwitchControl, playHaptic])
|
|
|
|
return (
|
|
<>
|
|
<SwitchAccountDialog control={accountSwitchControl} />
|
|
|
|
<Animated.View
|
|
style={[
|
|
styles.bottomBar,
|
|
pal.view,
|
|
pal.border,
|
|
{paddingBottom: clamp(safeAreaInsets.bottom, 15, 30)},
|
|
footerMinimalShellTransform,
|
|
]}
|
|
onLayout={e => {
|
|
footerHeight.value = e.nativeEvent.layout.height
|
|
}}>
|
|
{hasSession ? (
|
|
<>
|
|
<Btn
|
|
testID="bottomBarHomeBtn"
|
|
icon={
|
|
isAtHome ? (
|
|
<HomeFilled
|
|
width={iconWidth + 1}
|
|
style={[styles.ctrlIcon, pal.text, styles.homeIcon]}
|
|
/>
|
|
) : (
|
|
<Home
|
|
width={iconWidth + 1}
|
|
style={[styles.ctrlIcon, pal.text, styles.homeIcon]}
|
|
/>
|
|
)
|
|
}
|
|
onPress={onPressHome}
|
|
accessibilityRole="tab"
|
|
accessibilityLabel={_(msg`Home`)}
|
|
accessibilityHint=""
|
|
/>
|
|
<Btn
|
|
testID="bottomBarSearchBtn"
|
|
icon={
|
|
isAtSearch ? (
|
|
<MagnifyingGlassFilled
|
|
width={iconWidth + 2}
|
|
style={[styles.ctrlIcon, pal.text, styles.searchIcon]}
|
|
/>
|
|
) : (
|
|
<MagnifyingGlass
|
|
width={iconWidth + 2}
|
|
style={[styles.ctrlIcon, pal.text, styles.searchIcon]}
|
|
/>
|
|
)
|
|
}
|
|
onPress={onPressSearch}
|
|
accessibilityRole="search"
|
|
accessibilityLabel={_(msg`Search`)}
|
|
accessibilityHint=""
|
|
/>
|
|
<Btn
|
|
testID="bottomBarMessagesBtn"
|
|
icon={
|
|
isAtMessages ? (
|
|
<MessageFilled
|
|
width={iconWidth - 1}
|
|
style={[styles.ctrlIcon, pal.text, styles.feedsIcon]}
|
|
/>
|
|
) : (
|
|
<Message
|
|
width={iconWidth - 1}
|
|
style={[styles.ctrlIcon, pal.text, styles.feedsIcon]}
|
|
/>
|
|
)
|
|
}
|
|
onPress={onPressMessages}
|
|
notificationCount={numUnreadMessages.numUnread}
|
|
accessible={true}
|
|
accessibilityRole="tab"
|
|
accessibilityLabel={_(msg`Chat`)}
|
|
accessibilityHint={
|
|
numUnreadMessages.count > 0
|
|
? `${numUnreadMessages.numUnread} unread`
|
|
: ''
|
|
}
|
|
/>
|
|
<Btn
|
|
testID="bottomBarNotificationsBtn"
|
|
icon={
|
|
isAtNotifications ? (
|
|
<BellFilled
|
|
width={iconWidth}
|
|
style={[styles.ctrlIcon, pal.text, styles.bellIcon]}
|
|
/>
|
|
) : (
|
|
<Bell
|
|
width={iconWidth}
|
|
style={[styles.ctrlIcon, pal.text, styles.bellIcon]}
|
|
/>
|
|
)
|
|
}
|
|
onPress={onPressNotifications}
|
|
notificationCount={numUnreadNotifications}
|
|
accessible={true}
|
|
accessibilityRole="tab"
|
|
accessibilityLabel={_(msg`Notifications`)}
|
|
accessibilityHint={
|
|
numUnreadNotifications === ''
|
|
? ''
|
|
: `${numUnreadNotifications} unread`
|
|
}
|
|
/>
|
|
<Btn
|
|
testID="bottomBarProfileBtn"
|
|
icon={
|
|
<View style={styles.ctrlIconSizingWrapper}>
|
|
{isAtMyProfile ? (
|
|
<View
|
|
style={[
|
|
styles.ctrlIcon,
|
|
pal.text,
|
|
styles.profileIcon,
|
|
styles.onProfile,
|
|
{borderColor: pal.text.color},
|
|
]}>
|
|
<UserAvatar
|
|
avatar={profile?.avatar}
|
|
size={iconWidth - 3}
|
|
// See https://github.com/bluesky-social/social-app/pull/1801:
|
|
usePlainRNImage={true}
|
|
type={profile?.associated?.labeler ? 'labeler' : 'user'}
|
|
/>
|
|
</View>
|
|
) : (
|
|
<View
|
|
style={[styles.ctrlIcon, pal.text, styles.profileIcon]}>
|
|
<UserAvatar
|
|
avatar={profile?.avatar}
|
|
size={iconWidth - 3}
|
|
// See https://github.com/bluesky-social/social-app/pull/1801:
|
|
usePlainRNImage={true}
|
|
type={profile?.associated?.labeler ? 'labeler' : 'user'}
|
|
/>
|
|
</View>
|
|
)}
|
|
</View>
|
|
}
|
|
onPress={onPressProfile}
|
|
onLongPress={onLongPressProfile}
|
|
accessibilityRole="tab"
|
|
accessibilityLabel={_(msg`Profile`)}
|
|
accessibilityHint=""
|
|
/>
|
|
</>
|
|
) : (
|
|
<>
|
|
<View
|
|
style={{
|
|
width: '100%',
|
|
flexDirection: 'row',
|
|
alignItems: 'center',
|
|
justifyContent: 'space-between',
|
|
paddingTop: 14,
|
|
paddingBottom: 2,
|
|
paddingLeft: 14,
|
|
paddingRight: 6,
|
|
gap: 8,
|
|
}}>
|
|
<View
|
|
style={{flexDirection: 'row', alignItems: 'center', gap: 8}}>
|
|
<Logo width={28} />
|
|
<View style={{paddingTop: 4}}>
|
|
<Logotype width={80} fill={pal.text.color} />
|
|
</View>
|
|
</View>
|
|
|
|
<View
|
|
style={{flexDirection: 'row', alignItems: 'center', gap: 4}}>
|
|
<Button
|
|
onPress={showCreateAccount}
|
|
accessibilityHint={_(msg`Sign up`)}
|
|
accessibilityLabel={_(msg`Sign up`)}>
|
|
<Text type="md" style={[{color: 'white'}, s.bold]}>
|
|
<Trans>Sign up</Trans>
|
|
</Text>
|
|
</Button>
|
|
|
|
<Button
|
|
type="default"
|
|
onPress={showSignIn}
|
|
accessibilityHint={_(msg`Sign in`)}
|
|
accessibilityLabel={_(msg`Sign in`)}>
|
|
<Text type="md" style={[pal.text, s.bold]}>
|
|
<Trans>Sign in</Trans>
|
|
</Text>
|
|
</Button>
|
|
</View>
|
|
</View>
|
|
</>
|
|
)}
|
|
</Animated.View>
|
|
</>
|
|
)
|
|
}
|
|
|
|
interface BtnProps
|
|
extends Pick<
|
|
ComponentProps<typeof TouchableOpacity>,
|
|
| 'accessible'
|
|
| 'accessibilityRole'
|
|
| 'accessibilityHint'
|
|
| 'accessibilityLabel'
|
|
> {
|
|
testID?: string
|
|
icon: JSX.Element
|
|
notificationCount?: string
|
|
onPress?: (event: GestureResponderEvent) => void
|
|
onLongPress?: (event: GestureResponderEvent) => void
|
|
}
|
|
|
|
function Btn({
|
|
testID,
|
|
icon,
|
|
notificationCount,
|
|
onPress,
|
|
onLongPress,
|
|
accessible,
|
|
accessibilityHint,
|
|
accessibilityLabel,
|
|
}: BtnProps) {
|
|
return (
|
|
<TouchableOpacity
|
|
testID={testID}
|
|
style={styles.ctrl}
|
|
onPress={onLongPress ? onPress : undefined}
|
|
onPressIn={onLongPress ? undefined : onPress}
|
|
onLongPress={onLongPress}
|
|
accessible={accessible}
|
|
accessibilityLabel={accessibilityLabel}
|
|
accessibilityHint={accessibilityHint}>
|
|
{icon}
|
|
{notificationCount ? (
|
|
<View style={[styles.notificationCount]}>
|
|
<Text style={styles.notificationCountLabel}>{notificationCount}</Text>
|
|
</View>
|
|
) : undefined}
|
|
</TouchableOpacity>
|
|
)
|
|
}
|