This post is all about authenticating firebase phone number authentication and CRUD operations in your cool new app. In this post, you will learn
- How to implement
authenticate phone number
in our app. - How to perform Firebase
CRUD
operations in React-Native app. - How to upload an image on firebase storage
Before start, first you will need a React-Native app to start with, hence you can follow how to create a react-native app for beginners and start after that from here.
Complete source code of this tutorial is available here — react-native-firebase-phone-auth-CRUD
What is React-Native
React Native is a JavaScript framework for writing real, natively rendering mobile applications for iOS and Android. It’s based on React, Facebook’s JavaScript library for building user interfaces, but instead of targeting the browser, it targets mobile platforms. In other words: web developers can now write mobile applications that look and feel truly “native,” all from the comfort of a JavaScript library that we already know and love. Plus, because most of the code you write can be shared between platforms, React Native makes it easy to simultaneously develop for both Android and iOS.
Similar to React for the Web, React Native applications are written using a mixture of JavaScript and XML-Esque markup, known as JSX. Then, under the hood, the React Native “bridge” invokes the native rendering APIs in Objective-C (for iOS) or Java (for Android). Thus, your application will render using real mobile UI components, not webviews, and will look and feel like any other mobile application. React Native also exposes JavaScript interfaces for platform APIs, so your React Native apps can access platform features like the phone camera, or the user’s location.
React Native currently supports both iOS and Android and has the potential to expand to future platforms as well. In this tutorial, we’ll cover firebase phone authentication and CRUD operations. The vast majority of the code we write will be cross-platform. And yes: you can really use React Native to build production-ready mobile applications! Some anecdota: Facebook, Palantir, and TaskRabbit are already using it in production for user-facing applications.
The fact that React Native actually renders using its host platform’s standard rendering APIs enables it to stand out from most existing methods of cross-platform application development, like Cordova or Ionic.
Existing methods of writing mobile applications using combinations of JavaScript, HTML, and CSS typically render using webviews. While this approach can work, it also comes with drawbacks, especially around performance. Additionally, they do not usually have access to the host platform’s set of native UI elements. When these frameworks do try to mimic native UI elements, the results usually “feel” just a little off; reverse-engineering all the fine details of things like animations takes an enormous amount of effort, and they can quickly become out of date.
What and why Firebase?
Firebase is a mobile and web application development platform developed by Firebase Inc. in 2011, then was acquired by Google in 2014. Firebase is a toolset to “build, improve, and grow your app”. It gives you functionality like analytics, databases, messaging and crash reporting so you can move quickly and focus on your users.
Today Firebase is one of the fastest-growing platforms for application development. Some of the reasons are
- You don’t need to write a back-end from scratch. Firebase is a ready-made back-end, with a DB attached to it. You just integrate Firebase SDK in your app and you are good to go.
- It’s R-E-A-L T-I-M-E 🕺.If you are a developer, you understand the importance of a real-time back-end/database in today’s app market. Things like chat, news feeds, ratings, bookings, etc all are very easy if you factor in real-time operations
- Simple Authentication operations.The very first things required in a user-facing application is login/register operations. Firebase handles this very smoothly and with minimum coding effort. You can integrate a number of social authentication services like Facebook, Google etc. with Firebase as well.
- You get tonnes of additional features in-built e.g. AdMob, analytics, etc 😍😍
- It’s free, up to a certain usage limit. But this is pretty awesome for developers who are trying things, or making MVPs, or even for small scale app businesses. 🤑
In this article, we will focus on Firebase with react-native— phone Auth with CRUD operations and image upload to firebase storage.
Before we start I assume you have already created a project and ready to integrate the firebase in that. To add firebase in your project you can just follow this tutorial and start from here.
1. Phone Number Authentication — What and why?
Phone authentication allows users to sign in to Firebase using their phone as the authenticator. An SMS message is sent to the user via their phone number containing a unique code.
Using phone authentication is a little more involved as it is a multi-step process which can differ between iOS and Android. The basic flow for phone authentication is as follows:
- The user enters their phone number
- Firebase sends a verification code to the user’s phone
- The user enters the verification code
- Firebase confirms that the verification code matches and signs the user in
To start with phone authentication you need to install reat-native-firebase
by executing the following command:
npm install --save react-native-firebase
yarn add react-native-firebase
You can follow the official react-native-firebase link for any issue.
let's start with phone authentication, as I have performed all the actions and checked on android but I will cover for the android an IOS both in this tutorial.
Add the dependency
Android:
Add the Firebase Authentication dependency to android/app/build.gradle
:
dependencies {
// ...
implementation "com.google.firebase:firebase-auth:17.0.0"
}
Then, Install the RNFirebase Authentication package
Add the RNFirebaseAuthPackage
to your android/app/src/main/java/com/[app name]/MainApplication.java
:
// ...
import io.invertase.firebase.RNFirebasePackage;
import io.invertase.firebase.auth.RNFirebaseAuthPackage; // <-- Add this line
public class MainApplication extends Application implements ReactApplication {
// ...
@Override
protected List<ReactPackage> getPackages() {
return Arrays.<ReactPackage>asList(
new MainReactPackage(),
new RNFirebasePackage(),
new RNFirebaseAuthPackage() // <-- Add this line
);
}
};
// ...
}
Now you are all set to use authentications in your app.
signInWithPhoneNumber
is a more straight-forward implementation of handling the auth flow, however, it provides less flexibility to handle the various situations which may occur.
i. Trigger phone auth
firebase.auth().signInWithPhoneNumber(phoneNumber)
.then(confirmResult => // save confirm result to use with the manual verification code)
.catch(error => /error);
ii. Confirm verification code
confirmResult.confirm(verificationCode)
.then(user => // User is logged in){
.catch(error => // Error with verification code);
iii. [Android] Handle Auto Verification
To handle auto verification, we need to listen to auth#onAuthStateChanged:
firebase.auth().onAuthStateChanged((user) => {
if (user) // user is verified and logged in
});
Full Working Example:
Logics here:
import React, { Component } from 'react'; | |
import LoginView from './LoginView.js'; | |
import firebase from 'react-native-firebase'; | |
import { Text, Button, Card } from 'native-base'; | |
import { View } from 'react-native'; | |
import { TextInput } from 'react-native-gesture-handler'; | |
import { withNavigation } from 'react-navigation'; | |
import SpinnerComponent from '../Loader/LoaderView.js'; | |
import styles from './Loginstyles'; | |
class LoginContainer extends Component { | |
constructor(props) { | |
super(props); | |
this.state = { | |
user: null, | |
message: '', | |
codeInput: '', | |
phoneNumber: '+91', | |
confirmResult: null, | |
loading: false, | |
userName: '', | |
email: '', | |
phoneNo: '977112345', | |
userId: '' | |
}; | |
} | |
componentDidMount() { | |
this.unsubscribe = firebase.auth().onAuthStateChanged((user) => { | |
if (user) { | |
this.setState({ user: user.toJSON() }); | |
alert(console.log(user)); | |
} else { | |
this.setState({ | |
user: null, | |
message: '', | |
codeInput: '', | |
phoneNumber: '+91', | |
confirmResult: null, | |
}); | |
alert('no user exist'); | |
} | |
}); | |
} | |
componentWillUnmount() { | |
if (this.unsubscribe) this.unsubscribe(); | |
} | |
signIn = async () => { | |
const { phoneNumber } = this.state; | |
this.setState({ message: 'Sending code ...' }); | |
await firebase.auth().signInWithPhoneNumber(phoneNumber) | |
.then(confirmResult => { | |
alert('confirmResult' + confirmResult) | |
this.setState({ confirmResult, message: 'Code has been sent!' }); | |
}) | |
.catch(error => this.setState({ message: `Sign In With Phone Number Error: ${error.message}` })); | |
}; | |
confirmCode = () => { | |
const { codeInput, confirmResult } = this.state; | |
if (confirmResult && codeInput.length) { | |
confirmResult.confirm(codeInput) | |
.then((user) => { | |
alert(JSON.stringify(user.uid)); | |
this.setState( | |
{ | |
message: 'Code Confirmed!', | |
loading: true, | |
userId: user.uid, | |
phoneNo: user.phoneNumber | |
} | |
); | |
this.createUserDatabase() | |
}) | |
.catch(error => { | |
this.setState({ message: `Code Confirm Error: ${error.message}` }); | |
}); | |
} | |
}; | |
createUserDatabase = () => { | |
const { userName, phoneNo, userId } = this.state | |
the_uid = userId | |
const data = { | |
name: userName, | |
contact: phoneNo, | |
} | |
firebase.firestore().doc(`users/${the_uid}`).set(data) | |
.then(() => { | |
console.log("New poll data sent!") | |
}) | |
.catch(error => console.log("Error when creating new poll.", error)); | |
} | |
renderMessage = () => { | |
const { message } = this.state; | |
if (!message.length) return null; | |
return ( | |
<View style={styles.viewCardtype}> | |
<Text style={{ padding: 5, backgroundColor: '#000', color: '#fff' }}>{message}</Text> | |
</View> | |
); | |
} | |
renderVerificationCodeInput = () => { | |
const { codeInput, userName } = this.state; | |
return ( | |
<Card style={styles.boxStyle}> | |
<View style={{ marginTop: 25, padding: 25 }}> | |
<Text style={[styles.textWhite, styles.marginLR]}>Enter verification code below:</Text> | |
<View style={[styles.viewCardtype, styles.marginLR]}> | |
<TextInput | |
autoFocus | |
style={styles.inputText} | |
onChangeText={value => this.setState({ codeInput: value })} | |
placeholder={'Code ... '} | |
value={codeInput} | |
keyboardType='number-pad' | |
placeholderTextColor='white' | |
/> | |
</View> | |
<View style={[styles.viewCardtype, styles.marginLR]}> | |
<TextInput | |
style={styles.inputText} | |
onChangeText={namevalue => this.setState({ userName: namevalue })} | |
placeholder={'Enter your name... '} | |
value={userName} | |
keyboardType='name-phone-pad' | |
placeholderTextColor='white' | |
/> | |
</View> | |
<Button style={[styles.buttonDiv, styles.marginLR]} onPress={this.confirmCode}> | |
<Text style={styles.buttonDivText}> | |
Confirm Code | |
</Text> | |
</Button> | |
</View> | |
</Card> | |
); | |
} | |
valueChange = (textValue) => { | |
this.setState({ phoneNumber: textValue }) | |
} | |
goToMainPage = () => { | |
const { navigation } = this.props; | |
navigation.navigate('PageNavigation'); | |
console.log('skip is working') | |
} | |
renderButton = () => { | |
if (this.state.loading) { | |
return <SpinnerComponent size={'large'} /> | |
} | |
setTimeout(() => { | |
this.setState({ loading: false }) | |
this.goToMainPage() | |
}, 3000) | |
} | |
render() { | |
const { user, confirmResult, codeInput, phoneNumber, loading } = this.state; | |
return ( | |
<LoginView | |
user={user} | |
confirmResult={confirmResult} | |
signOut={this.signOut} | |
renderVerificationCodeInput={() => this.renderVerificationCodeInput()} | |
renderMessage={() => this.renderMessage()} | |
codeInput={codeInput} | |
phoneNumber={phoneNumber} | |
signIn={this.signIn} | |
valueChange={(textValue) => this.valueChange(textValue)} | |
goToMainPage={this.goToMainPage} | |
renderButton={this.renderButton} | |
loading={loading} | |
/> | |
); | |
} | |
} | |
export default withNavigation(LoginContainer); |
View here:
import React, { Component } from 'react'; | |
import { Container, Text, Button, Card, } from 'native-base'; | |
import { TextInput } from 'react-native-gesture-handler'; | |
import { View, ImageBackground } from 'react-native'; | |
import styles from './Loginstyles'; | |
const LoginView = (props) => { | |
const { | |
user, | |
confirmResult, | |
signIn, | |
renderVerificationCodeInput, | |
renderMessage, | |
phoneNumber, | |
valueChange, | |
codeInput | |
} = props; | |
const successImageUri = require('../../assets/1.jpg'); | |
return ( | |
<Container> | |
<ImageBackground source={successImageUri} | |
style={styles.fullwidthHeight} | |
resizeMode='cover'> | |
<View> | |
{!user && !confirmResult && | |
<Card style={styles.boxStyle}> | |
<View> | |
<Text style={{ fontSize: 34, textAlign: 'center', color: 'white', marginVertical: 82 }}>Login</Text> | |
</View> | |
<Text style={[styles.textWhite, styles.marginLR]}>Enter phone number :</Text> | |
<View style={[styles.viewCardtype, styles.marginLR]}> | |
<TextInput | |
autoFocus | |
style={styles.inputText} | |
onChangeText={(value) => valueChange(value)} | |
placeholder={'Enter Phone number ... '} | |
value={phoneNumber} | |
keyboardType='number-pad' | |
maxLength={13} | |
placeholderTextColor='white' | |
/> | |
</View> | |
<Button onPress={signIn} style={[styles.buttonDiv, styles.marginLR]}> | |
<Text style={styles.buttonDivText}>Sign In</Text> | |
</Button> | |
</Card> | |
} | |
{renderMessage()} | |
{!user && confirmResult && renderVerificationCodeInput()} | |
{user && | |
<View style={styles.buttonDiv}> | |
{props.renderButton()} | |
</View> | |
} | |
</View> | |
</ImageBackground> | |
</Container> | |
); | |
} | |
export default LoginView; |
Here I have divided my login flow into three different parts as:
1. LoginContainer contains the logic part
2. LoginView Contains the View Part and
3. LoginStyles contains the CSS of the page
Styles here:
import React, { Component } from 'react' | |
import { Dimensions } from 'react-native'; | |
const height = Dimensions.get('screen').height | |
const width = Dimensions.get('screen').width | |
export default styles = { | |
fullwidthHeight: { | |
width: width, | |
height: height, | |
justifyContent: 'center' | |
}, | |
boxStyle: { | |
shadowColor: "#000", | |
shadowOffset: { | |
width: 0, | |
height: 9, | |
}, | |
shadowOpacity: 0.48, | |
shadowRadius: 11.95, | |
elevation: 18, | |
backgroundColor: 'rgba(0, 0, 0, 0.7)' | |
}, | |
buttonDiv: { | |
backgroundColor: 'white', | |
marginTop: 76, | |
marginBottom: 16, | |
justifyContent: 'center' | |
}, | |
buttonDivText: { | |
color: 'red', | |
textAlign: 'center' | |
}, | |
marginLR: { | |
marginHorizontal: 18 | |
}, | |
textWhite: | |
{ fontSize: 18, color: 'white', paddingBottom: 5 }, | |
viewCardtype: { | |
borderWidth: 1, borderColor: 'white', shadowColor: "#000", | |
shadowOffset: { | |
width: 0, | |
height: 9, | |
}, | |
shadowOpacity: 0.48, | |
shadowRadius: 11.95, | |
elevation: 18, | |
}, | |
inputText: { height: 40, marginTop: 15, marginBottom: 15, color: 'white' } | |
} |
Now you are all set to take the taste of phone authentication in your cool react-native app. Here I have used the latest react-native version 0.60.4. If you get any issues just clone my repository with the link provided above.
Complete source code of this tutorial is available here — react-native-firebase-phone-auth-CRUD
2.CRUD Operations in react-native application
Before you start, you need to add Firebase Firestore in your app.
Add the dependency
Add the Firebase Firestore dependency to android/app/build.gradle
:
dependencies {
// ...
implementation "com.google.firebase:firebase-firestore:19.0.0"
}
Install the RNFirebase Firestore package
Add the RNFirebaseFirestorePackage
to your android/app/src/main/java/com/[app name]/MainApplication.java
:
// ...
import io.invertase.firebase.RNFirebasePackage;
import io.invertase.firebase.firestore.RNFirebaseFirestorePackage; // <-- Add this line
public class MainApplication extends Application implements ReactApplication {
// ...
@Override
protected List<ReactPackage> getPackages() {
return Arrays.<ReactPackage>asList(
new MainReactPackage(),
new RNFirebasePackage(),
new RNFirebaseFirestorePackage() // <-- Add this line
);
}
};
// ...
}
Then,
Enable multidex:
Adding Firestore to your Android app requires multiDexEnabled
to be set to true
in android/app/build.gradle
:
//..
android {
//..
defaultConfig {
//..
multiDexEnabled true // needed for firestore:
}
//..
}
Now you're all set to store and retrieve data in firebase Firestore.
1.Create
You will need to add the createDatabse()
method to persist an item collection in the Firestore database:
createDatabse(){
firebase.firestore().collection('collection name').add({.....your data})}
2. Read
You will need to add the readDatabse()
method to retrieve the available item from the Firestore collection:
readDatabse() {
collection name
return firebase.firestore.collection('').
onSnapshot((data)=>{console.log(data)});
}
3. Update
Next, you need to add the updateDatabse()
method to update an item data by its identifier:
updateDatabse(docId){
collection name
firebase.firestore.doc('/' + docId).update({data to update});
}
Here, docId is the particular id that you get while fetching data from the Firestore, with this docId you will be able to update a particular doc from the list of docs.
4. Delete
Finally, you can add the deleteItem()
method to delete an item by its identifier:
deleteItem(docId){
collection name
firebase.firestore.doc('/' + docId).delete();
}
The complete working example is here:
import React, { Component } from 'react' | |
import { Container, Content, Text, Button, Header, Left, Body, Right, Icon, Form, Item, Input, Card, ListItem, CardItem, Thumbnail, Label, Toast } from 'native-base'; | |
import { Switch } from 'react-native'; | |
import CameraGallery from '../CameraGallery/CameraGallery'; | |
import firebase from 'react-native-firebase'; | |
import SpinnerComponent from '../../src/screens/Loader/LoaderView'; | |
class ModalContainer extends Component { | |
constructor(props) { | |
super(props); | |
this.state = { | |
togVal: true, | |
imageUrl: 'https://scontent-sin2-1.xx.fbcdn.net/v/t1.0-1/c0.0.240.240a/p240x240/65822324_2111355985637711_2881848018242371584_n.jpg?_nc_cat=104&_nc_oc=AQljBZvYRPBQlLX4oEPTOMKK_JRfgJk1vJffBvR_PJLpGyTKeht7zk00VIEOYsbfa1U&_nc_ht=scontent-sin2-1.xx&oh=8ae0df6ebccbd58480935944dd68ac52&oe=5DD9AA1A', | |
taskStatus: null, | |
titleInput: 'Shadman', | |
imageUpload: null, | |
getData: '', | |
receiveUpdatedData: null, | |
docId: null, | |
loader: false | |
}; | |
} | |
async componentDidMount() { | |
const toUpdateData = this.props.sendUpdatedData | |
console.log('didmount called success', toUpdateData); | |
this.setState({ | |
receiveUpdatedData: toUpdateData | |
}, () => { | |
console.log('receivedData', this.state.receiveUpdatedData); | |
if (this.state.receiveUpdatedData !== null && this.state.receiveUpdatedData !== undefined) { | |
const { receiveUpdatedData } = this.state; | |
this.setState({ | |
imageUrl: receiveUpdatedData.Image, | |
togVal: receiveUpdatedData.Status, | |
titleInput: receiveUpdatedData.Title, | |
docId: receiveUpdatedData.id | |
}) | |
} | |
}) | |
} | |
changeValue = (toggleValue) => { | |
console.log('value', toggleValue) | |
this.setState({ | |
togVal: !this.state.togVal | |
}) | |
} | |
getUrl = (image, imageData) => { | |
const fileUri = decodeURI(image) | |
console.log('data coming form gallerypage' + image, imageData) | |
const { togVal } = this.state | |
this.setState({ | |
imageUrl: image, | |
taskStatus: togVal, | |
imageUpload: fileUri | |
}) | |
} | |
textChange = (value) => { | |
this.setState({ | |
titleInput: value, | |
}) | |
} | |
uploadImageTask = (imageBlob) => new Promise((resolve, reject) => { | |
console.log('putFile imageBlob' + imageBlob); | |
const fileName = new Date().getTime().toString(); | |
const userId = firebase.auth().currentUser.uid; | |
console.log('userId' + userId); | |
const ref = firebase.storage().ref(`tasksUpload/${userId}`).child(fileName); | |
ref.putFile(imageBlob).then( | |
() => ref.getDownloadURL().then(url => { | |
const { titleInput, taskStatus } = this.state | |
const taskDetail = { | |
Title: titleInput, | |
Image: url, | |
uid: userId, | |
Status: taskStatus | |
} | |
console.log('url' + url) | |
firebase.firestore().collection('tasks').add(taskDetail) | |
.then(imageData => resolve(imageData)) | |
.catch(e => reject(e)) | |
}) | |
).catch(e => reject(e)); | |
}); | |
addTask = () => { | |
const { imageUrl, imageUpload } = this.state | |
const fileUri = decodeURI(imageUrl) | |
this.uploadImageTask(imageUpload).then((res) => { | |
console.log('task added successfully', res) | |
Toast.show({ | |
text: 'Wrong password!', | |
buttonText: 'Okay', | |
duration: 3000, | |
position: 'bottom' | |
}) | |
this.setState({ | |
imageUrl: 'https://upload.wikimedia.org/wikipedia/commons/thumb/a/ac/No_image_available.svg/450px-No_image_available.svg.png', | |
titleInput: '', | |
taskStatus: null | |
}) | |
}).catch((err) => alert('err' + err)) | |
} | |
updateTask = () => { | |
console.log('task update enter') | |
const { imageUrl, titleInput, taskStatus, docId } = this.state | |
const fileUri = decodeURI(imageUrl); | |
const currentUId = firebase.auth().currentUser.uid | |
const data = { | |
Image: fileUri, | |
Status: taskStatus, | |
Title: titleInput, | |
uid: currentUId, | |
} | |
firebase.firestore().doc(`tasks/${docId}`).update(data).then((res) => { | |
setTimeout(() => { | |
this.setState({ | |
loader: true, | |
imageUrl: 'https://upload.wikimedia.org/wikipedia/commons/thumb/a/ac/No_image_available.svg/450px-No_image_available.svg.png', | |
titleInput: '', | |
taskStatus: null | |
}, () => this.props.onModalClose()) | |
}, 3000) | |
}).catch((err) => console.log('update Eror', err)); | |
this.setState({ | |
loader: false | |
}) | |
} | |
render() { | |
const { togVal, imageUrl, titleInput } = this.state | |
return ( | |
<Container> | |
<Header style={{ backgroundColor: '#6b52ae' }}> | |
<Left> | |
<Button transparent onPress={() => this.props.onModalClose()}> | |
<Icon name='arrow-back' /> | |
</Button> | |
</Left> | |
<Body /> | |
<Right /> | |
</Header> | |
<Content> | |
{this.state.receiveUpdatedData === null && | |
<Card style={{ margin: 32 }}> | |
<CardItem > | |
<Item regular> | |
<Icon name='contact' /> | |
<Input placeholder='Enter Your Title' onChangeText={(value) => this.textChange(value)} /> | |
</Item> | |
</CardItem> | |
<CardItem icon> | |
<Left> | |
<Button transparent> | |
<Icon active name="add" /> | |
</Button> | |
<Body> | |
<Text>Task Status</Text> | |
</Body> | |
</Left> | |
<Right> | |
<Switch onValueChange={(toggleValue) => this.changeValue(toggleValue)} value={togVal} /> | |
</Right> | |
</CardItem> | |
<CardItem> | |
<Left> | |
<Thumbnail square source={{ uri: imageUrl }} /> | |
<Body> | |
<Text>Select image</Text> | |
</Body> | |
</Left> | |
<CameraGallery getImage={(url, base) => this.getUrl(url, base)} /> | |
</CardItem> | |
<Button style={{ margin: 16, borderRadius: 10, justifyContent: 'center' }} onPress={this.addTask}> | |
<Text>Add Task</Text> | |
</Button> | |
</Card> | |
} | |
{this.state.receiveUpdatedData !== null && this.state.receiveUpdatedData !== undefined && | |
<Card style={{ margin: 32 }}> | |
<CardItem > | |
<Item regular> | |
<Icon name='contact' /> | |
<Input onChangeText={(value) => this.textChange(value)} value={titleInput} /> | |
</Item> | |
</CardItem> | |
<CardItem icon> | |
<Left> | |
<Button transparent> | |
<Icon active name="add" /> | |
</Button> | |
<Body> | |
<Text>Task Status</Text> | |
</Body> | |
</Left> | |
<Right> | |
<Switch onValueChange={(toggleValue) => this.changeValue(toggleValue)} value={togVal} /> | |
</Right> | |
</CardItem> | |
<CardItem> | |
<Left> | |
<Thumbnail square source={{ uri: imageUrl }} /> | |
<Body> | |
<Text>Select image</Text> | |
</Body> | |
</Left> | |
<CameraGallery getImage={(url, base) => this.getUrl(url, base)} /> | |
</CardItem> | |
<Button style={{ margin: 16, borderRadius: 10, justifyContent: 'center' }} onPress={this.updateTask}> | |
<Text>Update Task</Text> | |
</Button> | |
</Card> | |
} | |
{this.state.loader && | |
<SpinnerComponent size={42} /> | |
} | |
</Content> | |
</Container> | |
); | |
} | |
} | |
export default ModalContainer; |
import React, { Component } from 'react' | |
import { Container, Content, Text, Button, Header, Left, Body, Right, Icon, Form, Item, Input, Card, ListItem, CardItem, Thumbnail, Label, Toast } from 'native-base'; | |
import { Switch } from 'react-native'; | |
import CameraGallery from '../CameraGallery/CameraGallery'; | |
import firebase from 'react-native-firebase'; | |
import SpinnerComponent from '../../src/screens/Loader/LoaderView'; | |
class ModalContainer extends Component { | |
constructor(props) { | |
super(props); | |
this.state = { | |
togVal: true, | |
imageUrl: 'https://scontent-sin2-1.xx.fbcdn.net/v/t1.0-1/c0.0.240.240a/p240x240/65822324_2111355985637711_2881848018242371584_n.jpg?_nc_cat=104&_nc_oc=AQljBZvYRPBQlLX4oEPTOMKK_JRfgJk1vJffBvR_PJLpGyTKeht7zk00VIEOYsbfa1U&_nc_ht=scontent-sin2-1.xx&oh=8ae0df6ebccbd58480935944dd68ac52&oe=5DD9AA1A', | |
taskStatus: null, | |
titleInput: 'Shadman', | |
imageUpload: null, | |
getData: '', | |
receiveUpdatedData: null, | |
docId: null, | |
loader: false | |
}; | |
} | |
async componentDidMount() { | |
const toUpdateData = this.props.sendUpdatedData | |
console.log('didmount called success', toUpdateData); | |
this.setState({ | |
receiveUpdatedData: toUpdateData | |
}, () => { | |
console.log('receivedData', this.state.receiveUpdatedData); | |
if (this.state.receiveUpdatedData !== null && this.state.receiveUpdatedData !== undefined) { | |
const { receiveUpdatedData } = this.state; | |
this.setState({ | |
imageUrl: receiveUpdatedData.Image, | |
togVal: receiveUpdatedData.Status, | |
titleInput: receiveUpdatedData.Title, | |
docId: receiveUpdatedData.id | |
}) | |
} | |
}) | |
} | |
createUserDatabase = () => { | |
const { userName, phoneNo, userId } = this.state | |
the_uid = userId | |
const data = { | |
name: userName, | |
contact: phoneNo, | |
} | |
firebase.firestore().doc(`users/${the_uid}`).set(data) | |
.then(() => { | |
console.log("New poll data sent!") | |
}) | |
.catch(error => console.log("Error when creating new poll.", error)); | |
} | |
deleteTask = (docId) => { | |
console.log('delete pressed') | |
firebase.firestore().doc(`tasks/${docId}`).delete().then((res) => { | |
console.log('success delete', res) | |
}).catch((err) => { | |
console.log('error delete', err) | |
}) | |
} | |
} | |
export default ModalContainer; |
Complete source code of this tutorial is available here — react-native-firebase-phone-auth-CRUD and image upload
3 Image Upload in react-native-firebase storage
Add the dependency
Add the Firebase Storage dependency to android/app/build.gradle
:
dependencies {
// ...
implementation "com.google.firebase:firebase-storage:17.0.0"
}
Install the RNFirebase Storage package
Add the RNFirebaseStoragePackage
to your android/app/src/main/java/com/[app name]/MainApplication.java
:
// ...
import io.invertase.firebase.RNFirebasePackage;
import io.invertase.firebase.storage.RNFirebaseStoragePackage; // <-- Add this line
public class MainApplication extends Application implements ReactApplication {
// ...
@Override
protected List<ReactPackage> getPackages() {
return Arrays.<ReactPackage>asList(
new MainReactPackage(),
new RNFirebasePackage(),
new RNFirebaseStoragePackage() // <-- Add this line
);
}
};
// ...
}
Now you are all set to upload images on firebase Firestore and update that too. In the above gist, I have used a function uploadImageTask()
. In this function, I have uploaded an image to my firebase storage and received the URL and saved that to my database. In uploadImageTask(), we have used default javascript function decodeURI(imageURl).
Here, The decodeURI() function is used to decode URI generated by encodeURI(). Parameters: This function accepts single parameter complete_encoded_uri_string which holds the encoded string. Return Value: This function returns the decoded string (original string).
import React, { Component } from 'react' | |
import { Container, Content, Text, Button, Header, Left, Body, Right, Icon, Form, Item, Input, Card, ListItem, CardItem, Thumbnail, Label, Toast } from 'native-base'; | |
import { Switch } from 'react-native'; | |
import CameraGallery from '../CameraGallery/CameraGallery'; | |
import firebase from 'react-native-firebase'; | |
import SpinnerComponent from '../../src/screens/Loader/LoaderView'; | |
class ModalContainer extends Component { | |
constructor(props) { | |
super(props); | |
this.state = { | |
togVal: true, | |
imageUrl: 'https://scontent-sin2-1.xx.fbcdn.net/v/t1.0-1/c0.0.240.240a/p240x240/65822324_2111355985637711_2881848018242371584_n.jpg?_nc_cat=104&_nc_oc=AQljBZvYRPBQlLX4oEPTOMKK_JRfgJk1vJffBvR_PJLpGyTKeht7zk00VIEOYsbfa1U&_nc_ht=scontent-sin2-1.xx&oh=8ae0df6ebccbd58480935944dd68ac52&oe=5DD9AA1A', | |
taskStatus: null, | |
titleInput: 'Shadman', | |
imageUpload: null, | |
getData: '', | |
receiveUpdatedData: null, | |
docId: null, | |
loader: false | |
}; | |
} | |
async componentDidMount() { | |
const toUpdateData = this.props.sendUpdatedData | |
console.log('didmount called success', toUpdateData); | |
this.setState({ | |
receiveUpdatedData: toUpdateData | |
}, () => { | |
console.log('receivedData', this.state.receiveUpdatedData); | |
if (this.state.receiveUpdatedData !== null && this.state.receiveUpdatedData !== undefined) { | |
const { receiveUpdatedData } = this.state; | |
this.setState({ | |
imageUrl: receiveUpdatedData.Image, | |
togVal: receiveUpdatedData.Status, | |
titleInput: receiveUpdatedData.Title, | |
docId: receiveUpdatedData.id | |
}) | |
} | |
}) | |
} | |
changeValue = (toggleValue) => { | |
console.log('value', toggleValue) | |
this.setState({ | |
togVal: !this.state.togVal | |
}) | |
} | |
getUrl = (image, imageData) => { | |
const fileUri = decodeURI(image) | |
console.log('data coming form gallerypage' + image, imageData) | |
const { togVal } = this.state | |
this.setState({ | |
imageUrl: image, | |
taskStatus: togVal, | |
imageUpload: fileUri | |
}) | |
} | |
textChange = (value) => { | |
this.setState({ | |
titleInput: value, | |
}) | |
} | |
uploadImageTask = (imageBlob) => new Promise((resolve, reject) => { | |
console.log('putFile imageBlob' + imageBlob); | |
const fileName = new Date().getTime().toString(); | |
const userId = firebase.auth().currentUser.uid; | |
console.log('userId' + userId); | |
const ref = firebase.storage().ref(`tasksUpload/${userId}`).child(fileName); | |
ref.putFile(imageBlob).then( | |
() => ref.getDownloadURL().then(url => { | |
const { titleInput, taskStatus } = this.state | |
const taskDetail = { | |
Title: titleInput, | |
Image: url, | |
uid: userId, | |
Status: taskStatus | |
} | |
console.log('url' + url) | |
firebase.firestore().collection('tasks').add(taskDetail) | |
.then(imageData => resolve(imageData)) | |
.catch(e => reject(e)) | |
}) | |
).catch(e => reject(e)); | |
}); | |
addTask = () => { | |
const { imageUrl, imageUpload } = this.state | |
const fileUri = decodeURI(imageUrl) | |
this.uploadImageTask(imageUpload).then((res) => { | |
console.log('task added successfully', res) | |
Toast.show({ | |
text: 'Wrong password!', | |
buttonText: 'Okay', | |
duration: 3000, | |
position: 'bottom' | |
}) | |
this.setState({ | |
imageUrl: 'https://upload.wikimedia.org/wikipedia/commons/thumb/a/ac/No_image_available.svg/450px-No_image_available.svg.png', | |
titleInput: '', | |
taskStatus: null | |
}) | |
}).catch((err) => alert('err' + err)) | |
} | |
updateTask = () => { | |
console.log('task update enter') | |
const { imageUrl, titleInput, taskStatus, docId } = this.state | |
const fileUri = decodeURI(imageUrl); | |
const currentUId = firebase.auth().currentUser.uid | |
const data = { | |
Image: fileUri, | |
Status: taskStatus, | |
Title: titleInput, | |
uid: currentUId, | |
} | |
firebase.firestore().doc(`tasks/${docId}`).update(data).then((res) => { | |
setTimeout(() => { | |
this.setState({ | |
loader: true, | |
imageUrl: 'https://upload.wikimedia.org/wikipedia/commons/thumb/a/ac/No_image_available.svg/450px-No_image_available.svg.png', | |
titleInput: '', | |
taskStatus: null | |
}, () => this.props.onModalClose()) | |
}, 3000) | |
}).catch((err) => console.log('update Eror', err)); | |
this.setState({ | |
loader: false | |
}) | |
} | |
render() { | |
const { togVal, imageUrl, titleInput } = this.state | |
return ( | |
<Container> | |
<Header style={{ backgroundColor: '#6b52ae' }}> | |
<Left> | |
<Button transparent onPress={() => this.props.onModalClose()}> | |
<Icon name='arrow-back' /> | |
</Button> | |
</Left> | |
<Body /> | |
<Right /> | |
</Header> | |
<Content> | |
{this.state.receiveUpdatedData === null && | |
<Card style={{ margin: 32 }}> | |
<CardItem > | |
<Item regular> | |
<Icon name='contact' /> | |
<Input placeholder='Enter Your Title' onChangeText={(value) => this.textChange(value)} /> | |
</Item> | |
</CardItem> | |
<CardItem icon> | |
<Left> | |
<Button transparent> | |
<Icon active name="add" /> | |
</Button> | |
<Body> | |
<Text>Task Status</Text> | |
</Body> | |
</Left> | |
<Right> | |
<Switch onValueChange={(toggleValue) => this.changeValue(toggleValue)} value={togVal} /> | |
</Right> | |
</CardItem> | |
<CardItem> | |
<Left> | |
<Thumbnail square source={{ uri: imageUrl }} /> | |
<Body> | |
<Text>Select image</Text> | |
</Body> | |
</Left> | |
<CameraGallery getImage={(url, base) => this.getUrl(url, base)} /> | |
</CardItem> | |
<Button style={{ margin: 16, borderRadius: 10, justifyContent: 'center' }} onPress={this.addTask}> | |
<Text>Add Task</Text> | |
</Button> | |
</Card> | |
} | |
{this.state.receiveUpdatedData !== null && this.state.receiveUpdatedData !== undefined && | |
<Card style={{ margin: 32 }}> | |
<CardItem > | |
<Item regular> | |
<Icon name='contact' /> | |
<Input onChangeText={(value) => this.textChange(value)} value={titleInput} /> | |
</Item> | |
</CardItem> | |
<CardItem icon> | |
<Left> | |
<Button transparent> | |
<Icon active name="add" /> | |
</Button> | |
<Body> | |
<Text>Task Status</Text> | |
</Body> | |
</Left> | |
<Right> | |
<Switch onValueChange={(toggleValue) => this.changeValue(toggleValue)} value={togVal} /> | |
</Right> | |
</CardItem> | |
<CardItem> | |
<Left> | |
<Thumbnail square source={{ uri: imageUrl }} /> | |
<Body> | |
<Text>Select image</Text> | |
</Body> | |
</Left> | |
<CameraGallery getImage={(url, base) => this.getUrl(url, base)} /> | |
</CardItem> | |
<Button style={{ margin: 16, borderRadius: 10, justifyContent: 'center' }} onPress={this.updateTask}> | |
<Text>Update Task</Text> | |
</Button> | |
</Card> | |
} | |
{this.state.loader && | |
<SpinnerComponent size={42} /> | |
} | |
</Content> | |
</Container> | |
); | |
} | |
} | |
export default ModalContainer; |
Here all the functions are tested on android and to make it working in IOS please follow the IOS Setup from the react-native-firebase
documentation.
Once you setup then you can execute all the above functions or you can clone the repository.
You can follow all the steps and achieve this very easily or you can clone my repo on Github
Complete source code of this tutorial is available here — react-native-firebase-phone-auth-CRUD and image upload 😎 😎 😎 🕺 🕺 🕺..…
Conclusion
In this post, we learned how to integrate firebase phone auth with CRUD operations and image upload to the firebase storage. Since all the integrations are somehow tricky because of react-native stability but it has great documentation, if you face any issue please do ask me on our discuss section.
Complete source code of this tutorial is available here — react-native-firebase-phone-auth-CRUD and image upload
Next Steps
Now that you have learned the implementation of Firebase CRUD operations,phone auth, and image upload
in react-native, you can also try
- Ionic 4 PayPal payment integration — for Apps and PWA
- Ionic 4 Stripe payment integration — for Apps and PWA
- Ionic 4 Apple Pay integration
- Twitter login in Ionic 4 with Firebase
- Facebook login in Ionic 4 with Firebase
- Geolocation in Ionic 4
- QR Code and scanners in Ionic 4 and
- Translations in Ionic 4
If you need a base to start your next Ionic 4 app, you can make your next awesome app using Ionic 4 Full App
Top comments (0)