How To Integrate Firebase Authentication With an Expo App
Originally Published at Jscrambler.com.
Firebase is a Backend as a Service (BaaS) that provides a variety of services for web and mobile app development. Most of the mobile apps built using React Native and Expo require knowing the identity of a user. This allows an app to securely save user data in the cloud and provide more personalized functionalities.
Firebase has an Authentication service that integrates well in a React Native and Expo app. It has a ready-to-use SDK and supports many authentication providers such as email/password, phone numbers, and federated providers (Google, Facebook, Twitter, etc.).
In this tutorial, let's take a look at how as a mobile developer building applications using Expo SDK, you can integrate and use Firebase Authentication. You are going to:
- create some sample screens to display forms (login, sign-up);
- create a home screen that only a logged-in user can access;
- create different navigators using the react-navigation library;
- create an auth flow by conditionally rendering between these navigators when a user is logged in or not;
- and integrate Firebase Auth with the email/password method.
The source code for this tutorial is available on GitHub.
Prerequisites
🔗To follow this tutorial, please make sure you have the following tools and utilities installed on your local development environment and have access to the services mentioned below:
- Nodejs (>= 12.x.x) with a package manager installed such as npm or yarn
- expo-cli (>= 4.x.x)
- Firebase account (a free “Spark” plan is enough).
Creating a React Native app with expo-cli
🔗The initial step is to either create a new React Native project using expo-cli by following the steps mentioned below or, if you know the lexicons of creating projects with Expo, integrate Firebase JS SDK.
Yes, this guide is using Firebase JS SDK and Expo managed workflow.
Open your favorite terminal window, and execute the following command (where firebase-auth
is an example name of the project directory)
expo init firebase-auth# navigate inside the directorycd firebase-auth
Then, install the following libraries:
npm install @react-navigation/native @react-navigation/stack# OR is using yarnyarn add @react-navigation/native @react-navigation/stack# after the above dependencies install successfullyexpo install firebase dotenv expo-constants react-native-gesture-handler react-native-reanimated react-native-screens react-native-safe-area-context @react-native-community/masked-view
Side-note: This example guide is using React Navigation library version 5. Make sure to check out the official documentation, as some of the installation instructions might have changed since the writing of this tutorial.
Create a Firebase Project
🔗To get started, you’re going to need a Firebase app. Once you’ve created a new account with Firebase and logged in, create a new project by clicking on the Add Project button.
Next, add the name of the new Firebase project and then click Continue.
You can disable Google Analytics as it won't be used in this example. Then click Create Project
Expo Managed workflow apps can run inside a client app Expo Go (in development mode). The Expo Go app currently supports Firebase JS SDK and not the react-native-firebase library. More information in official Expo documentation.
On the Dashboard screen, in the left side menu, click the settings icon, and then go to the Project Settings page and then look for the section General > Your apps. If it's a new project, there won't be any apps.
Click the Web button. It will prompt you to enter the details of your app. Enter the app’s nickname, and then click the Register app button.
Then, Firebase will provide configuration objects with API keys and other keys that are required to use different Firebase services.
These API keys can be included in your React Native app as they are not used to access Firebase services’ backend resources. That can only be done by Firebase security rules.
This does not mean that you should expose these keys to a version control host such as GitHub. We will learn how to set up environment variables in an Expo app in the next section.
Let's enable the email/password sign-in method. From the left side menu, go to the Authentication page. If you are using this service for the first time in your Firebase project, click the Get Started button.
Then, in the Sign-in method tab, click the status of Email/Password, enable it, and then click Save.
Using Environment Variables
🔗To add environment variables to an Expo app, the initial step is to install the dotenv package (which should be already installed if you have been following along).
Create a .env
file at the root of your project and add the following:
API_KEY=XXXXAUTH_DOMAIN=XXXXPROJECT_ID=XXXXSTORAGE_BUCKET=XXXXMESSAGING_SENDER_ID=XXXXAPP_ID=XXXX
Replace all X's
in the above file with actual values for each key you get from the firebaseConfig
object.
Next, rename the app.json
file to app.config.js
at the root of your project. Add the import statement to use the dotenv
configuration. Since it's a JSON file, you will have to export all Expo configuration variables and also add an extra
object that contains Firebase configuration keys. Here is how the file should look like after this step:
1import 'dotenv/config';23export default {4 expo: {5 name: 'expo-firebase-auth-example',6 slug: 'expo-firebase-auth-example',7 version: '1.0.0',8 orientation: 'portrait',9 icon: './assets/icon.png',10 splash: {11 image: './assets/splash.png',12 resizeMode: 'contain',13 backgroundColor: '#ffffff'14 },15 updates: {16 fallbackToCacheTimeout: 017 },18 assetBundlePatterns: ['**/*'],19 ios: {20 supportsTablet: true21 },22 android: {23 adaptiveIcon: {24 foregroundImage: './assets/adaptive-icon.png',25 backgroundColor: '#FFFFFF'26 }27 },28 web: {29 favicon: './assets/favicon.png'30 },31 extra: {32 apiKey: process.env.API_KEY,33 authDomain: process.env.AUTH_DOMAIN,34 projectId: process.env.PROJECT_ID,35 storageBucket: process.env.STORAGE_BUCKET,36 messagingSenderId: process.env.MESSAGING_SENDER_ID,37 appId: process.env.APP_ID38 }39 }40};
Now, all the keys inside the extra
object are readable app-wide using expo-constants
. This package allows reading values from app.json
- or in this case, the app.config.js
file.
Open the Expo-generated project in your code editor, create a new directory in the root called config/
and add a file called firebase.js
. Edit the file as shown below:
1import firebase from 'firebase/app';2import 'firebase/auth';3import Constants from 'expo-constants';45// Initialize Firebase6const firebaseConfig = {7 apiKey: Constants.manifest.extra.apiKey,8 authDomain: Constants.manifest.extra.authDomain,9 projectId: Constants.manifest.extra.projectId,10 storageBucket: Constants.manifest.extra.storageBucket,11 messagingSenderId: Constants.manifest.extra.messagingSenderId,12 appId: Constants.manifest.extra.appId13};1415let Firebase;1617if (firebase.apps.length === 0) {18 Firebase = firebase.initializeApp(firebaseConfig);19}2021export default Firebase;
Creating reusable components
🔗The example app we are building in this tutorial will require some reusable components. These are visual components that can be used on different screens. Instead of writing them from scratch on every screen inside the app, let's create them once and re-use them whenever required.
Create a new directory called components/
and the following files:
- Button.js: contains a configurable
<Pressable/>
component; - IconButton.js: contains an icon button composed of a
<Pressable/>
component and the@expo/vector-icons
library; - ErrorMessage.js: a text component that is used to display an error message when authenticating a user;
- InputField.js: contains a configurable
<TextInput />
component.
Add the following code snippet to the Button.js
file:
1// components/Button.js2import React from 'react';3import { StyleSheet, Pressable, Text } from 'react-native';45const Button = ({6 title,7 backgroundColor = '#000',8 titleColor = '#fff',9 titleSize = 14,10 onPress,11 width = '100%',12 containerStyle13}) => {14 return (15 <Pressable16 onPress={onPress}17 style={args => {18 if (args.pressed) {19 return [20 styles.base,21 {22 opacity: 0.5,23 backgroundColor,24 width25 },26 containerStyle27 ];28 }2930 return [31 styles.base,32 {33 opacity: 1,34 backgroundColor,35 width36 },37 containerStyle38 ];39 }}40 >41 <Text style={[styles.text, { color: titleColor, fontSize: titleSize }]}>42 {title}43 </Text>44 </Pressable>45 );46};4748const styles = StyleSheet.create({49 text: {50 fontWeight: '600'51 },52 base: {53 alignItems: 'center',54 justifyContent: 'center',55 minHeight: 42,56 borderRadius: 4,57 paddingHorizontal: 1258 }59});6061export default Button;
Add the following code snippet in IconButton.js
:
1// components/IconButton.js23import React from 'react';4import { Pressable, StyleSheet } from 'react-native';5import { AntDesign } from '@expo/vector-icons';67const IconButton = ({ color, size, onPress, name }) => {8 return (9 <Pressable10 style={args => {11 if (args.pressed) {12 return [13 styles.base,14 {15 opacity: 0.5,16 backgroundColor: 'transparent'17 }18 ];19 }2021 return [styles.base, { opacity: 1, backgroundColor: 'transparent' }];22 }}23 onPress={onPress}24 >25 <AntDesign name={name} size={size} color={color} />26 </Pressable>27 );28};2930const styles = StyleSheet.create({31 base: {32 alignItems: 'center',33 justifyContent: 'center'34 }35});3637export default IconButton;
Add the following code snippet in ErrorMessage.js
. This component will be used to display error messages either when signing up or logging in to the app. These messages are human-readable and thrown by the Firebase Auth service. You can go through the complete list of messages in the Firebase official documentation.
1// components/ErrorMessage.js23import React from 'react';4import { StyleSheet, Text } from 'react-native';56const ErrorMessage = ({ error, visible }) => {7 if (!error || !visible) {8 return null;9 }1011 return <Text style={styles.errorText}>⚠️ {error}</Text>;12};1314const styles = StyleSheet.create({15 errorText: {16 color: '#fdca40',17 fontSize: 20,18 marginBottom: 10,19 fontWeight: '600'20 }21});2223export default ErrorMessage;
Add the following code snippet in InputField.js
:
1// components/InputField.js2import React from 'react';3import { View, StyleSheet, TextInput, TouchableOpacity } from 'react-native';4import { MaterialCommunityIcons } from '@expo/vector-icons';56const InputField = ({7 leftIcon,8 iconColor = '#000',9 rightIcon,10 inputStyle,11 containerStyle,12 placeholderTextColor = '#444',13 handlePasswordVisibility,14 ...rest15}) => {16 return (17 <View style={[styles.container, containerStyle]}>18 {leftIcon ? (19 <MaterialCommunityIcons20 name={leftIcon}21 size={20}22 color={iconColor}23 style={styles.leftIcon}24 />25 ) : null}26 <TextInput27 {...rest}28 placeholderTextColor={placeholderTextColor}29 style={[styles.input, inputStyle]}30 />31 {rightIcon ? (32 <TouchableOpacity onPress={handlePasswordVisibility}>33 <MaterialCommunityIcons34 name={rightIcon}35 size={20}36 color={iconColor}37 style={styles.rightIcon}38 />39 </TouchableOpacity>40 ) : null}41 </View>42 );43};4445const styles = StyleSheet.create({46 container: {47 borderRadius: 4,48 flexDirection: 'row',49 padding: 1250 },51 leftIcon: {52 marginRight: 1053 },54 input: {55 flex: 1,56 width: '100%',57 fontSize: 1858 },59 rightIcon: {60 alignSelf: 'center',61 marginLeft: 1062 }63});6465export default InputField;
Lastly, create an index.js
file that will expose all these components from the directory itself:
1import IconButton from './IconButton';2import Button from './Button';3import ErrorMessage from './ErrorMessage';4import InputField from './InputField';56export { IconButton, Button, ErrorMessage, InputField };
Creating screens in the app
🔗The sole focus of this tutorial is to integrate Firebase SDK and not to teach how to create app screens in React Native from scratch. While we go briefly over which screen is going to be composed of what React Native elements, please make sure you have basic knowledge of what core components are included in React Native.
Let's start by creating the structure of the screens directory. Once you have opened the Expo project in your preferred code editor, you will be welcomed by the default directory structure as shown below:
Create a new directory called /screens
and add the following screen files:
- HomeScreen.js
- LoginScreen.js
- SignupScreen.js
After creating these screen files, let's create the screens one by one. Start by modifying HomeScreen.js
. This screen will show the user's email and their UID when the user has either successfully signed up or logged in.
The UID is generated and assigned to every user who registers with the Firebase Auth service.
Both the user's email and UID will come from AuthenticatedUserContext
. We will get into those details later.
The firebase.auth().signOut()
function is a method provided by the Firebase auth service to log out the user from the app.
Add the following code snippet to HomeScreen.js
.
1import { StatusBar } from 'expo-status-bar';2import React, { useContext } from 'react';3import { StyleSheet, Text, View } from 'react-native';45import { IconButton } from '../components';6import Firebase from '../config/firebase';7import { AuthenticatedUserContext } from '../navigation/AuthenticatedUserProvider';89const auth = Firebase.auth();1011export default function HomeScreen() {12 const { user } = useContext(AuthenticatedUserContext);13 const handleSignOut = async () => {14 try {15 await auth.signOut();16 } catch (error) {17 console.log(error);18 }19 };20 return (21 <View style={styles.container}>22 <StatusBar style="dark-content" />23 <View style={styles.row}>24 <Text style={styles.title}>Welcome {user.email}!</Text>25 <IconButton26 name="logout"27 size={24}28 color="#fff"29 onPress={handleSignOut}30 />31 </View>32 <Text style={styles.text}>Your UID is: {user.uid} </Text>33 </View>34 );35}3637const styles = StyleSheet.create({38 container: {39 flex: 1,40 backgroundColor: '#e93b81',41 paddingTop: 50,42 paddingHorizontal: 1243 },44 row: {45 flexDirection: 'row',46 justifyContent: 'space-between',47 alignItems: 'center',48 marginBottom: 2449 },50 title: {51 fontSize: 24,52 fontWeight: '600',53 color: '#fff'54 },55 text: {56 fontSize: 16,57 fontWeight: 'normal',58 color: '#fff'59 }60});
Next, let's create the login screen. Add the code snippet below inside LoginScreen.js
. It contains two input fields and a button. Each input field represents the field where the user will enter their email
and password
. The value of each input field is stored inside two namesake state variables using the useState
hook.
Initially, the value for each variable is an empty string. When the user provides the value in the input field, the current value for each of these variables is updated using the corresponding update function setEmail
and setPassword
. The values stored by these variables will be used when sending login information to Firebase.
The three other state variables defined inside the LoginScreen
component are:
passwordVisibility
: to show/hide password on the input fieldrightIcon
: to set a default icon for thepasswordVisibility
functionalityloginError
: to store any incoming error when logging in from Firebase.
onLogin
is an asynchronous method that handles whether to log in the user or not based on their email
and password
values. These values are passed as arguments to a method called signInWithEmailAndPassword
provided by Firebase Auth.
1import { StatusBar } from 'expo-status-bar';2import React from 'react';3import { useState } from 'react';4import { StyleSheet, Text, View, Button as RNButton } from 'react-native';56import { Button, InputField, ErrorMessage } from '../components';7import Firebase from '../config/firebase';89const auth = Firebase.auth();1011export default function LoginScreen({ navigation }) {12 const [email, setEmail] = useState('');13 const [password, setPassword] = useState('');14 const [passwordVisibility, setPasswordVisibility] = useState(true);15 const [rightIcon, setRightIcon] = useState('eye');16 const [loginError, setLoginError] = useState('');1718 const handlePasswordVisibility = () => {19 if (rightIcon === 'eye') {20 setRightIcon('eye-off');21 setPasswordVisibility(!passwordVisibility);22 } else if (rightIcon === 'eye-off') {23 setRightIcon('eye');24 setPasswordVisibility(!passwordVisibility);25 }26 };2728 const onLogin = async () => {29 try {30 if (email !== '' && password !== '') {31 await auth.signInWithEmailAndPassword(email, password);32 }33 } catch (error) {34 setLoginError(error.message);35 }36 };3738 return (39 <View style={styles.container}>40 <StatusBar style="dark-content" />41 <Text style={styles.title}>Login</Text>42 <InputField43 inputStyle={{44 fontSize: 1445 }}46 containerStyle={{47 backgroundColor: '#fff',48 marginBottom: 2049 }}50 leftIcon="email"51 placeholder="Enter email"52 autoCapitalize="none"53 keyboardType="email-address"54 textContentType="emailAddress"55 autoFocus={true}56 value={email}57 onChangeText={text => setEmail(text)}58 />59 <InputField60 inputStyle={{61 fontSize: 1462 }}63 containerStyle={{64 backgroundColor: '#fff',65 marginBottom: 2066 }}67 leftIcon="lock"68 placeholder="Enter password"69 autoCapitalize="none"70 autoCorrect={false}71 secureTextEntry={passwordVisibility}72 textContentType="password"73 rightIcon={rightIcon}74 value={password}75 onChangeText={text => setPassword(text)}76 handlePasswordVisibility={handlePasswordVisibility}77 />78 {loginError ? <ErrorMessage error={loginError} visible={true} /> : null}79 <Button80 onPress={onLogin}81 backgroundColor="#f57c00"82 title="Login"83 tileColor="#fff"84 titleSize={20}85 containerStyle={{86 marginBottom: 2487 }}88 />89 <RNButton90 onPress={() => navigation.navigate('Signup')}91 title="Go to Signup"92 color="#fff"93 />94 </View>95 );96}9798const styles = StyleSheet.create({99 container: {100 flex: 1,101 backgroundColor: '#e93b81',102 paddingTop: 50,103 paddingHorizontal: 12104 },105 title: {106 fontSize: 24,107 fontWeight: '600',108 color: '#fff',109 alignSelf: 'center',110 paddingBottom: 24111 }112});
The signup screen is similar to the login screen. It uses onHandleSignup
, which is an asynchronous method that handles the action of registering a user or not based on their email
and password
values. These values are passed as arguments to a method called createUserWithEmailAndPassword
provided by Firebase Auth. Add the following code snippet to the SignupScreen.js
file:
1import { StatusBar } from 'expo-status-bar';2import React from 'react';3import { useState } from 'react';4import { StyleSheet, Text, View, Button as RNButton } from 'react-native';56import { Button, InputField, ErrorMessage } from '../components';7import Firebase from '../config/firebase';89const auth = Firebase.auth();1011export default function SignupScreen({ navigation }) {12 const [email, setEmail] = useState('');13 const [password, setPassword] = useState('');14 const [passwordVisibility, setPasswordVisibility] = useState(true);15 const [rightIcon, setRightIcon] = useState('eye');16 const [signupError, setSignupError] = useState('');1718 const handlePasswordVisibility = () => {19 if (rightIcon === 'eye') {20 setRightIcon('eye-off');21 setPasswordVisibility(!passwordVisibility);22 } else if (rightIcon === 'eye-off') {23 setRightIcon('eye');24 setPasswordVisibility(!passwordVisibility);25 }26 };2728 const onHandleSignup = async () => {29 try {30 if (email !== '' && password !== '') {31 await auth.createUserWithEmailAndPassword(email, password);32 }33 } catch (error) {34 setSignupError(error.message);35 }36 };3738 return (39 <View style={styles.container}>40 <StatusBar style="dark-content" />41 <Text style={styles.title}>Create new account</Text>42 <InputField43 inputStyle={{44 fontSize: 1445 }}46 containerStyle={{47 backgroundColor: '#fff',48 marginBottom: 2049 }}50 leftIcon="email"51 placeholder="Enter email"52 autoCapitalize="none"53 keyboardType="email-address"54 textContentType="emailAddress"55 autoFocus={true}56 value={email}57 onChangeText={text => setEmail(text)}58 />59 <InputField60 inputStyle={{61 fontSize: 1462 }}63 containerStyle={{64 backgroundColor: '#fff',65 marginBottom: 2066 }}67 leftIcon="lock"68 placeholder="Enter password"69 autoCapitalize="none"70 autoCorrect={false}71 secureTextEntry={passwordVisibility}72 textContentType="password"73 rightIcon={rightIcon}74 value={password}75 onChangeText={text => setPassword(text)}76 handlePasswordVisibility={handlePasswordVisibility}77 />78 {signupError ? <ErrorMessage error={signupError} visible={true} /> : null}79 <Button80 onPress={onHandleSignup}81 backgroundColor="#f57c00"82 title="Signup"83 tileColor="#fff"84 titleSize={20}85 containerStyle={{86 marginBottom: 2487 }}88 />89 <RNButton90 onPress={() => navigation.navigate('Login')}91 title="Go to Login"92 color="#fff"93 />94 </View>95 );96}9798const styles = StyleSheet.create({99 container: {100 flex: 1,101 backgroundColor: '#e93b81',102 paddingTop: 50,103 paddingHorizontal: 12104 },105 title: {106 fontSize: 24,107 fontWeight: '600',108 color: '#fff',109 alignSelf: 'center',110 paddingBottom: 24111 }112});
Create an authenticated user provider
🔗In this section, you are going to create an authentication provider to check whether the user is logged in or not and access them if they are logged in.
Create a new directory called navigation/
and inside it, create a file called AuthenticatedUserProvider.js
.
When a user is authenticated using a sign-in method in Firebase, it returns a user object with various properties such as email, photo URL, UID, display name, etc. To create the auth flow in the example app we are building, we need a way of knowing whether this user object exists or not. Thus, we conditionally render two different stack navigators (we will create them in the next section). So, a user will only be able to log in and access HomeScreen
if their respective user object exists.
One way to share data that is considered global in a React app is to use the React Context API. When creating a context, we must pass a default value. This value is used when a component has a matching Provider.
The Provider allows the React components to subscribe to the context changes. It wraps all other components in the React or React Native app.
To create an authenticated user provider, export a function called AuthenticatedUserProvider
. This provider is going to allow the screen components to access the logged-in or logged-out state of a user in the application. So, in the code snippet below, we define a state variable called user
.
1import React, { useState, createContext } from 'react';23export const AuthenticatedUserContext = createContext({});45export const AuthenticatedUserProvider = ({ children }) => {6 const [user, setUser] = useState(null);78 return (9 <AuthenticatedUserContext.Provider value={{ user, setUser }}>10 {children}11 </AuthenticatedUserContext.Provider>12 );13};
Creating Home and Auth stacks
🔗In this example app, there are two different stack navigator files to create:
HomeStack.js
: composed ofHomeScreen
AuthStack.js
: composed ofLoginScreen
andSignupScreen
Create these new files inside the navigation/
directory.
Add the following code snippet inside HomeStack.js
:
1import React from 'react';2import { createStackNavigator } from '@react-navigation/stack';34import HomeScreen from '../screens/HomeScreen';56const Stack = createStackNavigator();78export default function HomeStack() {9 return (10 <Stack.Navigator headerMode="none">11 <Stack.Screen name="Home" component={HomeScreen} />12 </Stack.Navigator>13 );14}
Next, add the following code snippet inside AuthStack.js
:
1import React from 'react';2import { createStackNavigator } from '@react-navigation/stack';34import LoginScreen from '../screens/LoginScreen';5import SignupScreen from '../screens/SignupScreen';67const Stack = createStackNavigator();89export default function AuthStack() {10 return (11 <Stack.Navigator headerMode="none">12 <Stack.Screen name="Login" component={LoginScreen} />13 <Stack.Screen name="Signup" component={SignupScreen} />14 </Stack.Navigator>15 );16}
Check a user's authenticated state
🔗The Firebase Auth service provides a listener called onAuthStateChanged
to detect changes to a user's logged-in state. It subscribes to a user's current authenticated state and receives an event whenever that state changes.
Using this listener, if the returned state of a user is null
, it means that the user is currently logged out. If it does not return null
, it will return a user object. This helps in persisting a user's authentication state in the app.
We will use this listener method at the top of our navigator. Create a new file called RootNavigator.js
inside the navigation/
directory. Start by importing the following statements:
1import React, { useContext, useEffect, useState } from 'react';2import { NavigationContainer } from '@react-navigation/native';3import { View, ActivityIndicator } from 'react-native';45import Firebase from '../config/firebase';6import { AuthenticatedUserContext } from './AuthenticatedUserProvider';7import AuthStack from './AuthStack';8import HomeStack from './HomeStack';910const auth = Firebase.auth();
Next, create a function component called RootNavigator
. Inside it, define a state variable called isLoading
when a user's authenticated state is currently being checked with the Firebase Auth service.
Using the useContext
hook, get the current value of user
and the method setUser
to update that value from AuthenticatedUserContext
.
This hook will trigger a re-render whenever the value of user
changes from the AuthenticatedUserContext
.
The onAuthStateChanged
listener will trigger inside the useEffect
hook. It also returns an unsubscriber function which allows the app to stop listening for events whenever the hook is no longer in use.
Add the following code snippet inside the RootNavigator.js
file:
1export default function RootNavigator() {2 const { user, setUser } = useContext(AuthenticatedUserContext);3 const [isLoading, setIsLoading] = useState(true);45 useEffect(() => {6 // onAuthStateChanged returns an unsubscriber7 const unsubscribeAuth = auth.onAuthStateChanged(async authenticatedUser => {8 try {9 await (authenticatedUser ? setUser(authenticatedUser) : setUser(null));10 setIsLoading(false);11 } catch (error) {12 console.log(error);13 }14 });1516 // unsubscribe auth listener on unmount17 return unsubscribeAuth;18 }, []);1920 if (isLoading) {21 return (22 <View style={{ flex: 1, justifyContent: 'center', alignItems: 'center' }}>23 <ActivityIndicator size="large" />24 </View>25 );26 }2728 return (29 <NavigationContainer>30 {user ? <HomeStack /> : <AuthStack />}31 </NavigationContainer>32 );33}
In the above code snippet, note that both the stack navigators are conditionally rendered depending on the state of the user.
Wrapping RootNavigator with AuthenticatedUserProvider
🔗Now that RootNavigator
is defined, the question remains on how to use AuthenticatedUserProvider
to wrap a set of components in the current app tree.
Well, you have to wrap this provider around the RootNavigator
in order to use the helper functions as well as the value of the current user in the screen components.
Create an index.js
file inside the navigation/
directory and add the following code snippet:
1import React from 'react';23import { AuthenticatedUserProvider } from './AuthenticatedUserProvider';4import RootNavigator from './RootNavigator';56/**7 * Wrap all providers here8 */910export default function Routes() {11 return (12 <AuthenticatedUserProvider>13 <RootNavigator />14 </AuthenticatedUserProvider>15 );16}
Also, modify the App.js
file to return Routes
.
Here is the demo of the complete authentication flow you will get after this step:
If you head over to the Firebase console and go to the Users tab on the Authentication page, you will see the details of the signed up user.
Conclusion
🔗You have now successfully integrated the Firebase Auth service in a React Native app using Expo SDK.
Using Firebase JS SDK allows integrating other sign-in providers such as Phone authentication, Facebook and Google. Refer to Expo’s official documentation to try out other login providers.
If you need to take a look at the code for this tutorial, you can refer to this GitHub repo.
More Posts
Browse all posts