import {useCallback, useRef, useState} from 'react'
import {Pressable, View} from 'react-native'
import Animated from 'react-native-reanimated'
import {useSafeAreaInsets} from 'react-native-safe-area-context'
import {sanitizeUrl} from '@braintree/sanitize-url'
import {msg, plural} from '@lingui/core/macro'
import {useLingui} from '@lingui/react'
import {Trans} from '@lingui/react/macro'
import {StackActions, useNavigationState} from '@react-navigation/native'
import {useHideBottomBarBorder} from '#/lib/hooks/useHideBottomBarBorder'
import {useMinimalShellFooterTransform} from '#/lib/hooks/useMinimalShellTransform'
import {useNavigationDeduped} from '#/lib/hooks/useNavigationDeduped'
import {
getCurrentRoute,
getTabState,
isTab,
TabState,
} from '#/lib/routes/helpers'
import {makeProfileLink} from '#/lib/routes/links'
import {type CommonNavigatorParams} from '#/lib/routes/types'
import {convertBskyAppUrlIfNeeded} from '#/lib/strings/url-helpers'
import {emitSoftReset} from '#/state/events'
import {useModalControls} from '#/state/modals'
import {useUnreadMessageCount} from '#/state/queries/messages/list-conversations'
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 {Link} from '#/view/com/util/Link'
import {UserAvatar} from '#/view/com/util/UserAvatar'
import {Logo} from '#/view/icons/Logo'
import {Logotype} from '#/view/icons/Logotype'
import {atoms as a, useTheme} from '#/alf'
import {Button, ButtonText} from '#/components/Button'
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,
MagnifyingGlass_Stroke2_Corner0_Rounded as MagnifyingGlass,
} from '#/components/icons/MagnifyingGlass'
import {
Message_Stroke2_Corner0_Rounded as Message,
Message_Stroke2_Corner0_Rounded_Filled as MessageFilled,
} from '#/components/icons/Message'
import {Text} from '#/components/Typography'
import {useAgeAssurance} from '#/ageAssurance'
import {IS_WEB_TOUCH_DEVICE} from '#/env'
import {router} from '#/routes'
import {styles} from './BottomBarStyles'
export function BottomBarWeb() {
const {_} = useLingui()
const {hasSession, currentAccount} = useSession()
const t = useTheme()
const {bottom: bottomInset} = useSafeAreaInsets()
const footerMinimalShellTransform = useMinimalShellFooterTransform()
const {requestSwitchToAccount} = useLoggedOutViewControls()
const closeAllActiveElements = useCloseAllActiveElements()
const {footerHeight} = useShellLayout()
const hideBorder = useHideBottomBarBorder()
const accountSwitchControl = useDialogControl()
const {data: profile} = useProfileQuery({did: currentAccount?.did})
const iconWidth = 26
const unreadMessageCount = useUnreadMessageCount()
const notificationCountStr = useUnreadNotifications()
const aa = useAgeAssurance()
const showSignIn = useCallback(() => {
closeAllActiveElements()
requestSwitchToAccount({requestedAccount: 'none'})
}, [requestSwitchToAccount, closeAllActiveElements])
const showCreateAccount = useCallback(() => {
closeAllActiveElements()
requestSwitchToAccount({requestedAccount: 'new'})
// setShowLoggedOut(true)
}, [requestSwitchToAccount, closeAllActiveElements])
const onLongPressProfile = useCallback(() => {
accountSwitchControl.open()
}, [accountSwitchControl])
return (
<>
footerHeight.set(event.nativeEvent.layout.height)}>
{hasSession ? (
<>
{({isActive}) => {
const Icon = isActive ? HomeFilled : Home
return (
)
}}
{({isActive}) => {
const Icon = isActive ? MagnifyingGlassFilled : MagnifyingGlass
return (
)
}}
{hasSession && (
<>
{({isActive}) => {
const Icon = isActive ? MessageFilled : Message
return (
)
}}
{({isActive}) => {
const Icon = isActive ? BellFilled : Bell
return (
)
}}
{({isActive}) => (
)}
>
)}
>
) : (
<>
>
)}
>
)
}
const NavItem: React.FC<{
children: (props: {isActive: boolean}) => React.ReactNode
href: string
routeName: string
hasNew?: boolean
notificationCount?: string
onLongPress?: () => void
}> = ({children, href, routeName, hasNew, notificationCount, onLongPress}) => {
const t = useTheme()
const {_} = useLingui()
const {bottom: bottomInset} = useSafeAreaInsets()
const {currentAccount} = useSession()
const currentRoute = useNavigationState(state => {
if (!state) {
return {name: 'Home'}
}
return getCurrentRoute(state)
})
// Checks whether we're on someone else's profile
const isOnDifferentProfile =
currentRoute.name === 'Profile' &&
routeName === 'Profile' &&
(currentRoute.params as CommonNavigatorParams['Profile']).name !==
currentAccount?.handle
const isActive =
currentRoute.name === 'Profile'
? isTab(currentRoute.name, routeName) &&
(currentRoute.params as CommonNavigatorParams['Profile']).name ===
(routeName === 'Profile'
? currentAccount?.handle
: (currentRoute.params as CommonNavigatorParams['Profile']).name)
: isTab(currentRoute.name, routeName)
if (IS_WEB_TOUCH_DEVICE) {
return (
{children}
)
}
return (
{children({isActive})}
{notificationCount ? (
{notificationCount}
) : hasNew ? (
) : null}
)
}
function TouchNavItem({
children,
href,
routeName,
isActive,
isOnDifferentProfile,
hasNew,
notificationCount,
onLongPress,
}: {
children: (props: {isActive: boolean}) => React.ReactNode
href: string
routeName: string
isActive: boolean
isOnDifferentProfile: boolean
hasNew?: boolean
notificationCount?: string
onLongPress?: () => void
}) {
const t = useTheme()
const {_} = useLingui()
const {bottom: bottomInset} = useSafeAreaInsets()
const navigation = useNavigationDeduped()
const {closeModal} = useModalControls()
// CSS transition press animation — runs on compositor thread
// so navigation re-renders don't cause jank
const [pressed, setPressed] = useState(false)
const pressInTime = useRef(0)
const pressOutTimer = useRef | undefined>(
undefined,
)
const ANIM_MS = 100
const handlePressIn = () => {
if (pressOutTimer.current) {
clearTimeout(pressOutTimer.current)
pressOutTimer.current = undefined
}
pressInTime.current = Date.now()
setPressed(true)
}
const handlePressOut = () => {
const elapsed = Date.now() - pressInTime.current
const remaining = Math.max(0, ANIM_MS - elapsed)
// Wait for scale-down to finish before starting scale-up
pressOutTimer.current = setTimeout(() => {
setPressed(false)
pressOutTimer.current = undefined
}, remaining)
}
const onPress = () => {
closeModal()
const sanitizedHref = convertBskyAppUrlIfNeeded(sanitizeUrl(href))
const [resolvedRouteName, params] = router.matchPath(sanitizedHref)
if (isOnDifferentProfile) {
// @ts-ignore we're not able to type check on this one -prf
navigation.dispatch(StackActions.push(resolvedRouteName, params))
} else {
const state = navigation.getState()
const tabState = getTabState(state, resolvedRouteName)
if (tabState === TabState.InsideAtRoot) {
emitSoftReset()
} else {
// @ts-ignore we're not able to type check on this one -prf
navigation.navigate(resolvedRouteName, params, {pop: true})
}
}
}
return (
{children({isActive})}
{notificationCount ? (
{notificationCount}
) : hasNew ? (
) : null}
)
}