import React, {ComponentProps} from 'react' import { Linking, SafeAreaView, ScrollView, StyleProp, StyleSheet, TouchableOpacity, View, ViewStyle, } from 'react-native' import {useNavigation, StackActions} from '@react-navigation/native' import {observer} from 'mobx-react-lite' import { FontAwesomeIcon, FontAwesomeIconStyle, } from '@fortawesome/react-native-fontawesome' import {s, colors} from 'lib/styles' import {FEEDBACK_FORM_URL, HELP_DESK_URL} from 'lib/constants' import {useStores} from 'state/index' import { HomeIcon, HomeIconSolid, BellIcon, BellIconSolid, UserIcon, CogIcon, MagnifyingGlassIcon2, MagnifyingGlassIcon2Solid, UserIconSolid, HashtagIcon, ListIcon, HandIcon, } from 'lib/icons' import {UserAvatar} from 'view/com/util/UserAvatar' import {Text} from 'view/com/util/text/Text' import {useTheme} from 'lib/ThemeContext' import {usePalette} from 'lib/hooks/usePalette' import {useAnalytics} from 'lib/analytics/analytics' import {pluralize} from 'lib/strings/helpers' import {getTabState, TabState} from 'lib/routes/helpers' import {NavigationProp} from 'lib/routes/types' import {useNavigationTabState} from 'lib/hooks/useNavigationTabState' import {isWeb} from 'platform/detection' import {formatCount, formatCountShortOnly} from 'view/com/util/numeric/format' import {Trans, msg} from '@lingui/macro' import {useLingui} from '@lingui/react' import {useSetDrawerOpen} from '#/state/shell' import {useModalControls} from '#/state/modals' import {useSession, SessionAccount} from '#/state/session' import {useProfileQuery} from '#/state/queries/profile' import {useUnreadNotifications} from '#/state/queries/notifications/unread' export function DrawerProfileCard({ account, onPressProfile, }: { account: SessionAccount onPressProfile: () => void }) { const {_} = useLingui() const pal = usePalette('default') const {data: profile} = useProfileQuery({did: account.did}) return ( {profile?.displayName || account.handle} @{account.handle} {formatCountShortOnly(profile?.followersCount ?? 0)} {' '} {pluralize(profile?.followersCount || 0, 'follower')} ·{' '} {formatCountShortOnly(profile?.followsCount ?? 0)} {' '} following ) } export const DrawerContent = observer(function DrawerContentImpl() { const theme = useTheme() const pal = usePalette('default') const store = useStores() const {_} = useLingui() const setDrawerOpen = useSetDrawerOpen() const navigation = useNavigation() const {track} = useAnalytics() const {isAtHome, isAtSearch, isAtFeeds, isAtNotifications, isAtMyProfile} = useNavigationTabState() const {currentAccount} = useSession() const numUnreadNotifications = useUnreadNotifications() // events // = const onPressTab = React.useCallback( (tab: string) => { track('Menu:ItemClicked', {url: tab}) const state = navigation.getState() setDrawerOpen(false) if (isWeb) { // hack because we have flat navigator for web and MyProfile does not exist on the web navigator -ansh if (tab === 'MyProfile') { navigation.navigate('Profile', {name: store.me.handle}) } else { // @ts-ignore must be Home, Search, Notifications, or MyProfile navigation.navigate(tab) } } else { const tabState = getTabState(state, tab) if (tabState === TabState.InsideAtRoot) { store.emitScreenSoftReset() } else if (tabState === TabState.Inside) { navigation.dispatch(StackActions.popToTop()) } else { // @ts-ignore must be Home, Search, Notifications, or MyProfile navigation.navigate(`${tab}Tab`) } } }, [store, track, navigation, setDrawerOpen], ) 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 onPressMyFeeds = React.useCallback( () => onPressTab('Feeds'), [onPressTab], ) const onPressLists = React.useCallback(() => { track('Menu:ItemClicked', {url: 'Lists'}) navigation.navigate('Lists') setDrawerOpen(false) }, [navigation, track, setDrawerOpen]) const onPressModeration = React.useCallback(() => { track('Menu:ItemClicked', {url: 'Moderation'}) navigation.navigate('Moderation') setDrawerOpen(false) }, [navigation, track, setDrawerOpen]) const onPressSettings = React.useCallback(() => { track('Menu:ItemClicked', {url: 'Settings'}) navigation.navigate('Settings') setDrawerOpen(false) }, [navigation, track, setDrawerOpen]) const onPressFeedback = React.useCallback(() => { track('Menu:FeedbackClicked') Linking.openURL( FEEDBACK_FORM_URL({ email: currentAccount?.email, handle: currentAccount?.handle, }), ) }, [track, currentAccount]) const onPressHelp = React.useCallback(() => { track('Menu:HelpClicked') Linking.openURL(HELP_DESK_URL) }, [track]) // rendering // = return ( {currentAccount && ( )} } size={24} strokeWidth={1.7} /> ) : ( } size={24} strokeWidth={1.7} /> ) } label="Search" accessibilityLabel={_(msg`Search`)} accessibilityHint="" bold={isAtSearch} onPress={onPressSearch} /> } size="24" strokeWidth={3.25} /> ) : ( } size="24" strokeWidth={3.25} /> ) } label="Home" accessibilityLabel={_(msg`Home`)} accessibilityHint="" bold={isAtHome} onPress={onPressHome} /> } size="24" strokeWidth={1.7} /> ) : ( } size="24" strokeWidth={1.7} /> ) } label="Notifications" accessibilityLabel={_(msg`Notifications`)} accessibilityHint={ numUnreadNotifications === '' ? '' : `${numUnreadNotifications} unread` } count={numUnreadNotifications} bold={isAtNotifications} onPress={onPressNotifications} /> ) : ( ) } label="Feeds" accessibilityLabel={_(msg`Feeds`)} accessibilityHint="" bold={isAtFeeds} onPress={onPressMyFeeds} /> } label="Lists" accessibilityLabel={_(msg`Lists`)} accessibilityHint="" onPress={onPressLists} /> } label="Moderation" accessibilityLabel={_(msg`Moderation`)} accessibilityHint="" onPress={onPressModeration} /> } size="26" strokeWidth={1.5} /> ) : ( } size="26" strokeWidth={1.5} /> ) } label="Profile" accessibilityLabel={_(msg`Profile`)} accessibilityHint="" onPress={onPressProfile} /> } size="26" strokeWidth={1.75} /> } label="Settings" accessibilityLabel={_(msg`Settings`)} accessibilityHint="" onPress={onPressSettings} /> Feedback Help ) }) interface MenuItemProps extends ComponentProps { icon: JSX.Element label: string count?: string bold?: boolean } function MenuItem({ icon, label, accessibilityLabel, count, bold, onPress, }: MenuItemProps) { const pal = usePalette('default') return ( {icon} {count ? ( 2 ? styles.menuItemCountHundreds : count.length > 1 ? styles.menuItemCountTens : undefined, ]}> {count} ) : undefined} {label} ) } const InviteCodes = observer(function InviteCodesImpl({ style, }: { style?: StyleProp }) { const {track} = useAnalytics() const store = useStores() const setDrawerOpen = useSetDrawerOpen() const pal = usePalette('default') const {invitesAvailable} = store.me const {openModal} = useModalControls() const onPress = React.useCallback(() => { track('Menu:ItemClicked', {url: '#invite-codes'}) setDrawerOpen(false) openModal({name: 'invite-codes'}) }, [openModal, track, setDrawerOpen]) return ( 0 ? pal.link : pal.textLight, ]} size={18} /> 0 ? pal.link : pal.textLight}> {formatCount(store.me.invitesAvailable)} invite{' '} {pluralize(store.me.invitesAvailable, 'code')} ) }) const styles = StyleSheet.create({ view: { flex: 1, paddingBottom: 50, maxWidth: 300, }, viewDarkMode: { backgroundColor: '#1B1919', }, main: { paddingLeft: 20, paddingTop: 20, }, smallSpacer: { height: 20, }, profileCardDisplayName: { marginTop: 20, paddingRight: 30, }, profileCardHandle: { marginTop: 4, paddingRight: 30, }, profileCardFollowers: { marginTop: 16, paddingRight: 10, }, menuItem: { flexDirection: 'row', alignItems: 'center', paddingVertical: 16, paddingRight: 10, }, menuItemIconWrapper: { width: 24, height: 24, alignItems: 'center', justifyContent: 'center', marginRight: 12, }, menuItemCount: { position: 'absolute', width: 'auto', right: -6, top: -4, backgroundColor: colors.blue3, paddingHorizontal: 4, paddingBottom: 1, borderRadius: 6, }, menuItemCountTens: { width: 25, }, menuItemCountHundreds: { right: -12, width: 34, }, menuItemCountLabel: { fontSize: 12, fontWeight: 'bold', fontVariant: ['tabular-nums'], color: colors.white, }, inviteCodes: { paddingLeft: 22, paddingVertical: 8, flexDirection: 'row', alignItems: 'center', }, inviteCodesIcon: { marginRight: 6, }, footer: { flexWrap: 'wrap', flexDirection: 'row', gap: 8, paddingRight: 20, paddingTop: 20, paddingLeft: 20, }, footerBtn: { flexDirection: 'row', alignItems: 'center', padding: 10, borderRadius: 25, }, footerBtnFeedback: { paddingHorizontal: 20, }, footerBtnFeedbackLight: { backgroundColor: '#DDEFFF', }, footerBtnFeedbackDark: { backgroundColor: colors.blue6, }, })