The navigation in the App consists of a top-level Stack Navigator (called RootStack
) with each of its Screen
components handling different high-level flow. All those flows can be seen in AuthScreens.tsx
file.
RHP
- Right hand panel that shows content inside a dismissible modal that takes up a partial portion of the screen on large format devices e.g. desktop/web/tablets. On smaller screens the content shown in the RHP fills the entire screen.
Navigation Actions - User actions correspond to resulting navigation actions that we will define now. The navigation actions are: Back
, Up
, Dismiss
, Forward
and Push
.
-
Back
- Moves the user “back” in the history stack by popping the screen or stack of screens. Note: This can potentially make the user exit the app itself (native) or display a previous app (not Expensify), or just the empty state of the browser. -
Up
- Pops the current screen off the current stack. This action is very easy to confuse withBack
. Unless you’ve navigated from one screen to a nested screen in a stack of screens these actions will almost always be the same. Unlike a “back” action, “up” should never result in the user exiting the app and should only be an option if there is somewhere to go “up” to. -
Dismiss
- Closes any modals (outside the navigation hierarchy) or pops a nested stack of screens off returning the user to the previous screen in the main stack. -
Forward
- This will take you forward in the history stack. Can only be invoked after you have goneBack
at least once. Note: Only possible on web. -
Push
- Either adds a new individual screen to the main stack or a nested stack of screens to the main stack with the user pointed at the last index of the pushed stack.
Most of the time, if you want to add some of the flows concerning one of your reports, e.g. Money Request
from a user, you will most probably use RightModalNavigator.tsx
and ModalStackNavigators.tsx
file:
-
Since each of those flows is kind of a modal stack, if you want to add a page to the existing flow, you should just add a page to the correct stack in
ModalStackNavigators.tsx
. -
If you want to create new flow, add a
Screen
inRightModalNavigator.tsx
and make new modal inModalStackNavigators.tsx
with chosen pages.
When creating RHP flows, you have to remember a couple things:
-
Since you can deeplink to different pages inside the RHP navigator, it is important to provide the possibility for the user to properly navigate back from any page with UP press (
HeaderWithBackButton
component). -
An example can be deeplinking to
/settings/profile/personal-details
. From there, when pressing the UP button, you should navigate to/settings/profile
, so in order for it to work, you should provide the correct route inonBackButtonPress
prop ofHeaderWithBackButton
(Navigation.goBack(ROUTES.SETTINGS_PROFILE)
in this example). -
We use a custom
goBack
function to handle the browser and thereact-navigation
history stack. Under the hood, it resolves to either replacing the current screen with the one we navigate to (deeplinking scenario) or just going back if we reached the current page by navigating in App (pops the screen). It ensures the requested behaviors on web, which is navigating back to the place from where you deeplinked when going into the RHP flow by it. -
If you want to navigate to a certain report after completing a flow related to it, e.g.
RequestMoney
flow with a certain group/user, you should useNavigation.dismissModal
with thisreportID
as an argument. If, in the future, we would like to navigate to something different than the report after such flows, the API should be rather easy to change. We do it like that in order to replace the RHP flow with the new report instead of pushing it, so pressing the back button does not navigate back to the ending page of the flow. If we were to navigate to the same report, we just pop the RHP modal.
An example of adding Settings_Workspaces
page:
- Add the page name to
SCREENS.ts
which will be reused throughout the app (linkingConfig, navigators, etc.):
const SCREENS = {
SETTINGS: {
WORKSPACES: 'Settings_Workspaces',
},
} as const;
- Add path to
ROUTES.ts
: https://github.com/Expensify/App/blob/main/src/ROUTES.ts
export const ROUTES = {
// static route
SETTINGS_WORKSPACES: 'settings/workspaces',
// dynamic route
SETTINGS_WORKSPACES: {
route: 'settings/:accountID',
getRoute: (accountID: number) => `settings/${accountID}` as const,
},
};
-
Add
Settings_Workspaces
page to proper RHP flow inlinkingConfig.ts
: https://github.com/Expensify/App/blob/fbc11ca729ffa4676fb3bc8cd110ac3890debff6/src/libs/Navigation/linkingConfig.ts#L47-L50 -
Add your page to proper navigator (it should be aligned with where you've put it in the previous step) https://github.com/Expensify/App/blob/fbc11ca729ffa4676fb3bc8cd110ac3890debff6/src/libs/Navigation/AppNavigator/ModalStackNavigators.js#L141
-
Make sure
HeaderWithBackButton
leads to the previous page in navigation flow of your page: https://github.com/Expensify/App/blob/3531af22dcadaa94ed11eccf370517dca0b8c305/src/pages/workspace/WorkspacesListPage.js#L186
Using react-freeze allows us to increase performance by avoiding unnecessary re-renders of screens that aren’t visible to the user anyway.
-
To ensure that the user doesn't ever see frozen report content, we are freezing the screens from 2 levels down the
RootStack
(which contains aScreen
for each report), so when dismissing with a swipe, the user always sees the content of the previous report. -
We want to freeze as high in the view hierarchy as we can, so we do it in a
Screen
ofRootStack
, beingCentralPaneNavigator
andSidebarScreen
. -
We want the report content visible as fast as possible, and at the same time we want the navigation animation to trigger instantly. To do so, we do a hack with
firstRenderRef
which rendersReportActionsSkeletonView
instead of the messages at the first render, and the proper content afterward. It works since there are always more renders ofReportScreen
before the content shows up (hopefully).
-
The wide and narrow layouts are conditionally rendered with different components in
createResponsiveNavigator
depending on screen size (isSmallScreen
prop from thewithWindowDimension.js
). -
The wide layout is rendered with our custom
ThreePaneView.js
and the narrow layout is rendered withStackView
from@react-navigation/stack
-
To make sure that we have the correct navigation state after changing the layout we need to force react to create new instance of the
NavigationContainer
. Without this, the navigation state could be broken after changing the type of layout. -
We are getting the new instance by changing the
key
prop ofNavigationContainer
that depends on theisSmallScreenWidth
. -
To keep the navigation state that was present before changing the layout, we save the state on every change and use it for the
initialState
prop. Changing the layout means that every component insideNavigationContainer
is mounted anew.
Let's assume that the user wants to navigate like this:
1. Settings_root - navigate > Profile
2. Profile - UP > Settings_root
3. Settings_root - navigate > About
4. About - browser back button > Settings_root
Without minimal action, expected behavior won't be achieved and the final screen will be Profile
.
Broken behavior is the outcome of two things:
-
Back button in the browser resets the navigation state with the state saved in step two.
-
Navigation.navigate
creates action withgetActionFromState
dispatched at the top level of the navigation hierarchy.
The reason why getActionFromState
provided by react-navigation
is dispatched at the top level of the navigation hierarchy is that it doesn't know about current navigation state, only about desired one.
In this example it doesn't know if the RightModalNavigator
and Settings
are already mounted.
The action for the first step looks like that:
{
"type": "NAVIGATE",
"payload": {
"name": "RightModalNavigator",
"params": {
"initial": true,
"screen": "Settings",
"params": {
"initial": true,
"screen": "Profile",
}
}
}
}
That means, the params for the RightModalNavigator
and Settings
(also a navigator) will be filled with the information that we want to have the Profile
screen in the state.
{
"index": 2,
"routes": [
{ "name": "Home", },
{
"name": "RightModalNavigator",
// here you can see that the params are filled with the information about structure that should be in the state.
"params": {
"initial": true,
"screen": "Settings",
"params": {
"initial": true,
"screen": "Settings_Profile",
"path": "/settings/profile"
}
},
"state": {
"index": 0,
"routes": [
{
"name": "Settings",
// Same here
"params": {
"initial": true,
"screen": "Settings_Profile",
"path": "/settings/profile"
},
"state": {
"index": 0,
"routes": [
{
"name": "Settings_Profile"
}
]
}
}
]
}
}
]
}
This information will stay here even if we pop the Profile
screen and navigate to About
screen.
Later on, when the user presses the browser back button expecting that the Settings_root
screen will appear, the navigation state will be reset with information about the Profile
screen in the params and this will be used as a source of truth for the navigation.
If we can create simple action that will only push one screen to the existing navigator, we won't fill any params of the navigators.
The getMinimalAction
compares action generated by the getActionFromState
with the current navigation state and tries to find the smallest action possible.
The action for the first step created with getMinimalAction
looks like this:
{
"type": "NAVIGATE",
"payload": {
"name": "Settings_Profile"
},
"target": "Settings-stack-key-xyz"
}
There is no minimal action for deeplinking directly to the Profile
screen. But because the Settings_root
is not on the stack, pressing UP will reset the params for navigators to the correct ones.