2023-03-13 22:01:43 +01:00
import * as React from 'react'
import { StyleSheet } from 'react-native'
2023-04-21 19:21:38 +02:00
import { observer } from 'mobx-react-lite'
2023-03-13 22:01:43 +01:00
import {
NavigationContainer ,
createNavigationContainerRef ,
2023-04-21 19:21:38 +02:00
CommonActions ,
2023-03-13 22:01:43 +01:00
StackActions ,
2023-04-24 23:36:05 +02:00
DefaultTheme ,
DarkTheme ,
2023-03-13 22:01:43 +01:00
} from '@react-navigation/native'
import { createNativeStackNavigator } from '@react-navigation/native-stack'
2023-06-22 18:40:32 +02:00
import {
BottomTabBarProps ,
createBottomTabNavigator ,
} from '@react-navigation/bottom-tabs'
2023-03-13 22:01:43 +01:00
import {
HomeTabNavigatorParams ,
SearchTabNavigatorParams ,
2023-05-26 03:02:37 +02:00
FeedsTabNavigatorParams ,
2023-03-13 22:01:43 +01:00
NotificationsTabNavigatorParams ,
FlatNavigatorParams ,
AllNavigatorParams ,
2023-04-18 18:19:37 +02:00
MyProfileTabNavigatorParams ,
BottomTabNavigatorParams ,
2023-03-13 22:01:43 +01:00
} from 'lib/routes/types'
2023-04-13 03:27:55 +02:00
import { BottomBar } from './view/shell/bottom-bar/BottomBar'
2023-03-13 22:01:43 +01:00
import { buildStateObject } from 'lib/routes/helpers'
import { State , RouteParams } from 'lib/routes/types'
import { colors } from 'lib/styles'
import { isNative } from 'platform/detection'
import { useColorSchemeStyle } from 'lib/hooks/useColorSchemeStyle'
import { router } from './routes'
2023-04-29 03:03:13 +02:00
import { usePalette } from 'lib/hooks/usePalette'
import { useStores } from './state'
2023-09-19 21:24:58 +02:00
import { getRoutingInstrumentation } from 'lib/sentry'
import { bskyTitle } from 'lib/strings/headings'
import { JSX } from 'react/jsx-runtime'
import { timeout } from 'lib/async/timeout'
2023-03-13 22:01:43 +01:00
import { HomeScreen } from './view/screens/Home'
import { SearchScreen } from './view/screens/Search'
2023-05-26 03:02:37 +02:00
import { FeedsScreen } from './view/screens/Feeds'
2023-03-13 22:01:43 +01:00
import { NotificationsScreen } from './view/screens/Notifications'
2023-05-11 23:08:21 +02:00
import { ModerationScreen } from './view/screens/Moderation'
import { ModerationMuteListsScreen } from './view/screens/ModerationMuteLists'
2023-03-13 22:01:43 +01:00
import { NotFoundScreen } from './view/screens/NotFound'
import { SettingsScreen } from './view/screens/Settings'
2023-09-21 20:33:19 +02:00
import { LanguageSettingsScreen } from './view/screens/LanguageSettings'
2023-03-13 22:01:43 +01:00
import { ProfileScreen } from './view/screens/Profile'
import { ProfileFollowersScreen } from './view/screens/ProfileFollowers'
import { ProfileFollowsScreen } from './view/screens/ProfileFollows'
2023-05-18 05:12:14 +02:00
import { CustomFeedScreen } from './view/screens/CustomFeed'
import { CustomFeedLikedByScreen } from './view/screens/CustomFeedLikedBy'
2023-05-11 23:08:21 +02:00
import { ProfileListScreen } from './view/screens/ProfileList'
2023-03-13 22:01:43 +01:00
import { PostThreadScreen } from './view/screens/PostThread'
2023-03-31 20:17:26 +02:00
import { PostLikedByScreen } from './view/screens/PostLikedBy'
2023-03-13 22:01:43 +01:00
import { PostRepostedByScreen } from './view/screens/PostRepostedBy'
import { DebugScreen } from './view/screens/Debug'
import { LogScreen } from './view/screens/Log'
2023-03-14 02:34:01 +01:00
import { SupportScreen } from './view/screens/Support'
import { PrivacyPolicyScreen } from './view/screens/PrivacyPolicy'
2023-04-07 05:53:58 +02:00
import { TermsOfServiceScreen } from './view/screens/TermsOfService'
import { CommunityGuidelinesScreen } from './view/screens/CommunityGuidelines'
import { CopyrightPolicyScreen } from './view/screens/CopyrightPolicy'
2023-04-22 01:55:29 +02:00
import { AppPasswords } from 'view/screens/AppPasswords'
2023-05-11 23:08:21 +02:00
import { ModerationMutedAccounts } from 'view/screens/ModerationMutedAccounts'
import { ModerationBlockedAccounts } from 'view/screens/ModerationBlockedAccounts'
2023-05-18 05:04:01 +02:00
import { SavedFeeds } from 'view/screens/SavedFeeds'
2023-08-31 00:21:12 +02:00
import { PreferencesHomeFeed } from 'view/screens/PreferencesHomeFeed'
2023-09-19 21:24:58 +02:00
import { PreferencesThreads } from 'view/screens/PreferencesThreads'
2023-03-13 22:01:43 +01:00
const navigationRef = createNavigationContainerRef < AllNavigatorParams > ( )
const HomeTab = createNativeStackNavigator < HomeTabNavigatorParams > ( )
const SearchTab = createNativeStackNavigator < SearchTabNavigatorParams > ( )
2023-05-26 03:02:37 +02:00
const FeedsTab = createNativeStackNavigator < FeedsTabNavigatorParams > ( )
2023-03-13 22:01:43 +01:00
const NotificationsTab =
createNativeStackNavigator < NotificationsTabNavigatorParams > ( )
2023-04-18 18:19:37 +02:00
const MyProfileTab = createNativeStackNavigator < MyProfileTabNavigatorParams > ( )
2023-03-13 22:01:43 +01:00
const Flat = createNativeStackNavigator < FlatNavigatorParams > ( )
2023-04-18 18:19:37 +02:00
const Tab = createBottomTabNavigator < BottomTabNavigatorParams > ( )
2023-03-13 22:01:43 +01:00
/ * *
* These "common screens" are reused across stacks .
* /
2023-05-16 20:13:05 +02:00
function commonScreens ( Stack : typeof HomeTab , unreadCountLabel? : string ) {
const title = ( page : string ) = > bskyTitle ( page , unreadCountLabel )
2023-03-13 22:01:43 +01:00
return (
< >
2023-05-16 20:13:05 +02:00
< Stack.Screen
name = "NotFound"
component = { NotFoundScreen }
options = { { title : title ( 'Not Found' ) } }
/ >
< Stack.Screen
name = "Moderation"
component = { ModerationScreen }
options = { { title : title ( 'Moderation' ) } }
/ >
2023-05-11 23:08:21 +02:00
< Stack.Screen
name = "ModerationMuteLists"
component = { ModerationMuteListsScreen }
2023-05-16 20:13:05 +02:00
options = { { title : title ( 'Mute Lists' ) } }
2023-05-11 23:08:21 +02:00
/ >
< Stack.Screen
name = "ModerationMutedAccounts"
component = { ModerationMutedAccounts }
2023-05-16 20:13:05 +02:00
options = { { title : title ( 'Muted Accounts' ) } }
2023-05-11 23:08:21 +02:00
/ >
< Stack.Screen
name = "ModerationBlockedAccounts"
component = { ModerationBlockedAccounts }
2023-05-16 20:13:05 +02:00
options = { { title : title ( 'Blocked Accounts' ) } }
/ >
< Stack.Screen
name = "Settings"
component = { SettingsScreen }
options = { { title : title ( 'Settings' ) } }
/ >
2023-09-21 20:33:19 +02:00
< Stack.Screen
name = "LanguageSettings"
component = { LanguageSettingsScreen }
options = { { title : title ( 'Language Settings' ) } }
/ >
2023-05-16 20:13:05 +02:00
< Stack.Screen
name = "Profile"
component = { ProfileScreen }
2023-08-03 19:25:17 +02:00
options = { ( { route } ) = > ( {
title : title ( ` @ ${ route . params . name } ` ) ,
animation : 'none' ,
} ) }
2023-05-11 23:08:21 +02:00
/ >
2023-03-13 22:01:43 +01:00
< Stack.Screen
name = "ProfileFollowers"
component = { ProfileFollowersScreen }
2023-05-16 20:13:05 +02:00
options = { ( { route } ) = > ( {
title : title ( ` People following @ ${ route . params . name } ` ) ,
} ) }
/ >
< Stack.Screen
name = "ProfileFollows"
component = { ProfileFollowsScreen }
options = { ( { route } ) = > ( {
title : title ( ` People followed by @ ${ route . params . name } ` ) ,
} ) }
/ >
< Stack.Screen
name = "ProfileList"
component = { ProfileListScreen }
options = { { title : title ( 'Mute List' ) } }
/ >
< Stack.Screen
name = "PostThread"
component = { PostThreadScreen }
options = { ( { route } ) = > ( { title : title ( ` Post by @ ${ route . params . name } ` ) } ) }
/ >
< Stack.Screen
name = "PostLikedBy"
component = { PostLikedByScreen }
options = { ( { route } ) = > ( { title : title ( ` Post by @ ${ route . params . name } ` ) } ) }
/ >
< Stack.Screen
name = "PostRepostedBy"
component = { PostRepostedByScreen }
options = { ( { route } ) = > ( { title : title ( ` Post by @ ${ route . params . name } ` ) } ) }
/ >
2023-05-18 06:33:59 +02:00
< Stack.Screen
name = "CustomFeed"
component = { CustomFeedScreen }
2023-05-18 06:36:36 +02:00
options = { { title : title ( 'Feed' ) } }
2023-05-18 06:33:59 +02:00
/ >
2023-05-18 05:12:14 +02:00
< Stack.Screen
name = "CustomFeedLikedBy"
component = { CustomFeedLikedByScreen }
2023-05-18 06:36:36 +02:00
options = { { title : title ( 'Liked by' ) } }
2023-05-18 05:12:14 +02:00
/ >
2023-05-16 20:13:05 +02:00
< Stack.Screen
name = "Debug"
component = { DebugScreen }
options = { { title : title ( 'Debug' ) } }
/ >
< Stack.Screen
name = "Log"
component = { LogScreen }
options = { { title : title ( 'Log' ) } }
/ >
< Stack.Screen
name = "Support"
component = { SupportScreen }
options = { { title : title ( 'Support' ) } }
/ >
< Stack.Screen
name = "PrivacyPolicy"
component = { PrivacyPolicyScreen }
options = { { title : title ( 'Privacy Policy' ) } }
/ >
< Stack.Screen
name = "TermsOfService"
component = { TermsOfServiceScreen }
options = { { title : title ( 'Terms of Service' ) } }
2023-03-13 22:01:43 +01:00
/ >
2023-04-07 05:53:58 +02:00
< Stack.Screen
name = "CommunityGuidelines"
component = { CommunityGuidelinesScreen }
2023-05-16 20:13:05 +02:00
options = { { title : title ( 'Community Guidelines' ) } }
/ >
< Stack.Screen
name = "CopyrightPolicy"
component = { CopyrightPolicyScreen }
options = { { title : title ( 'Copyright Policy' ) } }
/ >
< Stack.Screen
name = "AppPasswords"
component = { AppPasswords }
options = { { title : title ( 'App Passwords' ) } }
2023-04-07 05:53:58 +02:00
/ >
2023-05-18 06:33:59 +02:00
< Stack.Screen
name = "SavedFeeds"
component = { SavedFeeds }
options = { { title : title ( 'Edit My Feeds' ) } }
/ >
2023-08-31 00:21:12 +02:00
< Stack.Screen
name = "PreferencesHomeFeed"
component = { PreferencesHomeFeed }
options = { { title : title ( 'Home Feed Preferences' ) } }
/ >
2023-09-19 21:24:58 +02:00
< Stack.Screen
name = "PreferencesThreads"
component = { PreferencesThreads }
options = { { title : title ( 'Threads Preferences' ) } }
/ >
2023-03-13 22:01:43 +01:00
< / >
)
}
/ * *
* The TabsNavigator is used by native mobile to represent the routes
* in 3 distinct tab - stacks with a different root screen on each .
* /
function TabsNavigator() {
2023-06-22 18:40:32 +02:00
const tabBar = React . useCallback (
( props : JSX.IntrinsicAttributes & BottomTabBarProps ) = > (
< BottomBar { ...props } / >
) ,
[ ] ,
)
2023-09-28 21:41:44 +02:00
2023-03-13 22:01:43 +01:00
return (
< Tab.Navigator
initialRouteName = "HomeTab"
backBehavior = "initialRoute"
2023-05-31 03:16:30 +02:00
screenOptions = { { headerShown : false , lazy : true } }
2023-03-13 22:01:43 +01:00
tabBar = { tabBar } >
< Tab.Screen name = "HomeTab" component = { HomeTabNavigator } / >
2023-05-26 03:02:37 +02:00
< Tab.Screen name = "SearchTab" component = { SearchTabNavigator } / >
< Tab.Screen name = "FeedsTab" component = { FeedsTabNavigator } / >
2023-03-13 22:01:43 +01:00
< Tab.Screen
name = "NotificationsTab"
component = { NotificationsTabNavigator }
/ >
2023-04-18 18:19:37 +02:00
< Tab.Screen name = "MyProfileTab" component = { MyProfileTabNavigator } / >
2023-03-13 22:01:43 +01:00
< / Tab.Navigator >
)
}
function HomeTabNavigator() {
const contentStyle = useColorSchemeStyle ( styles . bgLight , styles . bgDark )
2023-08-28 22:37:44 +02:00
2023-03-13 22:01:43 +01:00
return (
< HomeTab.Navigator
screenOptions = { {
gestureEnabled : true ,
fullScreenGestureEnabled : true ,
headerShown : false ,
animationDuration : 250 ,
contentStyle ,
} } >
< HomeTab.Screen name = "Home" component = { HomeScreen } / >
{ commonScreens ( HomeTab ) }
< / HomeTab.Navigator >
)
}
function SearchTabNavigator() {
const contentStyle = useColorSchemeStyle ( styles . bgLight , styles . bgDark )
return (
< SearchTab.Navigator
screenOptions = { {
gestureEnabled : true ,
fullScreenGestureEnabled : true ,
headerShown : false ,
animationDuration : 250 ,
contentStyle ,
} } >
< SearchTab.Screen name = "Search" component = { SearchScreen } / >
{ commonScreens ( SearchTab as typeof HomeTab ) }
< / SearchTab.Navigator >
)
}
2023-05-26 03:02:37 +02:00
function FeedsTabNavigator() {
const contentStyle = useColorSchemeStyle ( styles . bgLight , styles . bgDark )
return (
< FeedsTab.Navigator
screenOptions = { {
gestureEnabled : true ,
fullScreenGestureEnabled : true ,
headerShown : false ,
animationDuration : 250 ,
contentStyle ,
} } >
< FeedsTab.Screen name = "Feeds" component = { FeedsScreen } / >
{ commonScreens ( FeedsTab as typeof HomeTab ) }
< / FeedsTab.Navigator >
)
}
2023-03-13 22:01:43 +01:00
function NotificationsTabNavigator() {
const contentStyle = useColorSchemeStyle ( styles . bgLight , styles . bgDark )
return (
< NotificationsTab.Navigator
screenOptions = { {
gestureEnabled : true ,
fullScreenGestureEnabled : true ,
headerShown : false ,
animationDuration : 250 ,
contentStyle ,
} } >
< NotificationsTab.Screen
name = "Notifications"
component = { NotificationsScreen }
/ >
{ commonScreens ( NotificationsTab as typeof HomeTab ) }
< / NotificationsTab.Navigator >
)
}
2023-09-08 02:36:08 +02:00
const MyProfileTabNavigator = observer ( function MyProfileTabNavigatorImpl() {
2023-04-18 18:19:37 +02:00
const contentStyle = useColorSchemeStyle ( styles . bgLight , styles . bgDark )
const store = useStores ( )
return (
< MyProfileTab.Navigator
screenOptions = { {
gestureEnabled : true ,
fullScreenGestureEnabled : true ,
headerShown : false ,
animationDuration : 250 ,
contentStyle ,
} } >
< MyProfileTab.Screen
name = "MyProfile"
// @ts-ignore // TODO: fix this broken type in ProfileScreen
component = { ProfileScreen }
initialParams = { {
2023-04-21 19:21:38 +02:00
name : store.me.did ,
2023-04-18 18:19:37 +02:00
} }
/ >
{ commonScreens ( MyProfileTab as typeof HomeTab ) }
< / MyProfileTab.Navigator >
)
2023-04-21 19:21:38 +02:00
} )
2023-04-18 18:19:37 +02:00
2023-03-13 22:01:43 +01:00
/ * *
* The FlatNavigator is used by Web to represent the routes
* in a single ( "flat" ) stack .
* /
2023-09-08 02:36:08 +02:00
const FlatNavigator = observer ( function FlatNavigatorImpl() {
2023-04-13 03:49:40 +02:00
const pal = usePalette ( 'default' )
2023-09-15 01:10:59 +02:00
const store = useStores ( )
const unreadCountLabel = store . me . notifications . unreadCountLabel
2023-05-16 20:13:05 +02:00
const title = ( page : string ) = > bskyTitle ( page , unreadCountLabel )
2023-03-13 22:01:43 +01:00
return (
< Flat.Navigator
screenOptions = { {
gestureEnabled : true ,
fullScreenGestureEnabled : true ,
headerShown : false ,
animationDuration : 250 ,
2023-04-13 03:49:40 +02:00
contentStyle : [ pal . view ] ,
2023-03-13 22:01:43 +01:00
} } >
2023-05-16 20:13:05 +02:00
< Flat.Screen
name = "Home"
component = { HomeScreen }
options = { { title : title ( 'Home' ) } }
/ >
< Flat.Screen
name = "Search"
component = { SearchScreen }
options = { { title : title ( 'Search' ) } }
/ >
2023-05-26 03:02:37 +02:00
< Flat.Screen
name = "Feeds"
component = { FeedsScreen }
options = { { title : title ( 'Feeds' ) } }
/ >
2023-05-16 20:13:05 +02:00
< Flat.Screen
name = "Notifications"
component = { NotificationsScreen }
options = { { title : title ( 'Notifications' ) } }
/ >
{ commonScreens ( Flat as typeof HomeTab , unreadCountLabel ) }
2023-03-13 22:01:43 +01:00
< / Flat.Navigator >
)
2023-05-17 16:50:28 +02:00
} )
2023-03-13 22:01:43 +01:00
/ * *
* The RoutesContainer should wrap all components which need access
* to the navigation context .
* /
const LINKING = {
prefixes : [ 'bsky://' , 'https://bsky.app' ] ,
getPathFromState ( state : State ) {
// find the current node in the navigation tree
let node = state . routes [ state . index || 0 ]
while ( node . state ? . routes && typeof node . state ? . index === 'number' ) {
node = node . state ? . routes [ node . state ? . index ]
}
// build the path
const route = router . matchName ( node . name )
if ( typeof route === 'undefined' ) {
return '/' // default to home
}
return route . build ( ( node . params || { } ) as RouteParams )
} ,
getStateFromPath ( path : string ) {
const [ name , params ] = router . matchPath ( path )
if ( isNative ) {
if ( name === 'Search' ) {
return buildStateObject ( 'SearchTab' , 'Search' , params )
}
if ( name === 'Notifications' ) {
return buildStateObject ( 'NotificationsTab' , 'Notifications' , params )
}
2023-05-31 04:16:29 +02:00
if ( name === 'Home' ) {
return buildStateObject ( 'HomeTab' , 'Home' , params )
}
// if the path is something else, like a post, profile, or even settings, we need to initialize the home tab as pre-existing state otherwise the back button will not work
return buildStateObject ( 'HomeTab' , name , params , [
{
name : 'Home' ,
params : { } ,
} ,
] )
2023-03-13 22:01:43 +01:00
} else {
return buildStateObject ( 'Flat' , name , params )
}
} ,
}
function RoutesContainer ( { children } : React . PropsWithChildren < { } > ) {
2023-04-24 23:36:05 +02:00
const theme = useColorSchemeStyle ( DefaultTheme , DarkTheme )
2023-03-13 22:01:43 +01:00
return (
2023-05-01 21:42:31 +02:00
< NavigationContainer
ref = { navigationRef }
linking = { LINKING }
theme = { theme }
onReady = { ( ) = > {
// Register the navigation container with the Sentry instrumentation (only works on native)
if ( isNative ) {
const routingInstrumentation = getRoutingInstrumentation ( )
routingInstrumentation . registerNavigationContainer ( navigationRef )
}
} } >
2023-03-13 22:01:43 +01:00
{ children }
< / NavigationContainer >
)
}
/ * *
* These helpers can be used from outside of the RoutesContainer
* ( eg in the state models ) .
* /
function navigate < K extends keyof AllNavigatorParams > (
name : K ,
params? : AllNavigatorParams [ K ] ,
) {
if ( navigationRef . isReady ( ) ) {
2023-10-13 21:10:15 +02:00
return Promise . race ( [
new Promise < void > ( resolve = > {
const handler = ( ) = > {
resolve ( )
navigationRef . removeListener ( 'state' , handler )
}
navigationRef . addListener ( 'state' , handler )
// @ts-ignore I dont know what would make typescript happy but I have a life -prf
navigationRef . navigate ( name , params )
} ) ,
timeout ( 1 e3 ) ,
] )
2023-03-13 22:01:43 +01:00
}
2023-10-13 21:10:15 +02:00
return Promise . resolve ( )
2023-03-13 22:01:43 +01:00
}
function resetToTab ( tabName : 'HomeTab' | 'SearchTab' | 'NotificationsTab' ) {
if ( navigationRef . isReady ( ) ) {
navigate ( tabName )
2023-05-04 23:18:27 +02:00
if ( navigationRef . canGoBack ( ) ) {
navigationRef . dispatch ( StackActions . popToTop ( ) ) //we need to check .canGoBack() before calling it
}
2023-03-13 22:01:43 +01:00
}
}
2023-07-20 08:50:42 +02:00
// returns a promise that resolves after the state reset is complete
function reset ( ) : Promise < void > {
2023-04-21 19:21:38 +02:00
if ( navigationRef . isReady ( ) ) {
navigationRef . dispatch (
CommonActions . reset ( {
index : 0 ,
routes : [ { name : isNative ? 'HomeTab' : 'Home' } ] ,
} ) ,
)
2023-07-20 08:50:42 +02:00
return Promise . race ( [
timeout ( 1 e3 ) ,
new Promise < void > ( resolve = > {
const handler = ( ) = > {
resolve ( )
navigationRef . removeListener ( 'state' , handler )
}
navigationRef . addListener ( 'state' , handler )
} ) ,
] )
} else {
return Promise . resolve ( )
2023-04-21 19:21:38 +02:00
}
}
2023-03-13 22:01:43 +01:00
function handleLink ( url : string ) {
let path
if ( url . startsWith ( '/' ) ) {
path = url
} else if ( url . startsWith ( 'http' ) ) {
try {
path = new URL ( url ) . pathname
} catch ( e ) {
console . error ( 'Invalid url' , url , e )
return
}
} else {
console . error ( 'Invalid url' , url )
return
}
const [ name , params ] = router . matchPath ( path )
if ( isNative ) {
if ( name === 'Search' ) {
resetToTab ( 'SearchTab' )
} else if ( name === 'Notifications' ) {
resetToTab ( 'NotificationsTab' )
} else {
resetToTab ( 'HomeTab' )
// @ts-ignore matchPath doesnt give us type-checked output -prf
navigate ( name , params )
}
} else {
// @ts-ignore matchPath doesnt give us type-checked output -prf
navigate ( name , params )
}
}
const styles = StyleSheet . create ( {
bgDark : {
backgroundColor : colors.black ,
} ,
bgLight : {
2023-04-24 23:36:05 +02:00
backgroundColor : colors.white ,
2023-03-13 22:01:43 +01:00
} ,
} )
export {
navigate ,
resetToTab ,
2023-04-21 19:21:38 +02:00
reset ,
2023-03-13 22:01:43 +01:00
handleLink ,
TabsNavigator ,
FlatNavigator ,
RoutesContainer ,
}