Firestore
To correctly begin using firestore, make sure you have the following:
v2.0.0-beta.11
or higher (most like to usenext
tag in package.json)- Install
redux-firestore
in your project usingnpm i --save redux-firestore@latest
firestore
imported withimport 'firebase/firestore'
firestore
initialize withfirebase.firestore()
reduxFirestore
enhancer added to store creatorfirestoreReducer
added to your reducers (will be combinable with main before v2.0.0 release)
Should look something similar to:
import { createStore, combineReducers, compose } from 'redux'
import firebase from 'firebase'
import 'firebase/firestore' // add this to use Firestore
import { reactReduxFirebase, firebaseReducer } from 'react-redux-firebase'
import { reduxFirestore, firestoreReducer } from 'redux-firestore'
const firebaseConfig = {}
// react-redux-firebase config
const rrfConfig = {
userProfile: 'users',
// useFirestoreForProfile: true // Firestore for Profile instead of Realtime DB
}
// initialize firebase instance
firebase.initializeApp(firebaseConfig) // <- new to v2.*.*
// initialize Firestore
firebase.firestore()
// Add reduxReduxFirebase enhancer when making store creator
const createStoreWithFirebase = compose(
reactReduxFirebase(firebase, rrfConfig),
reduxFirestore(firebase)
)(createStore)
// Add Firebase to reducers
const rootReducer = combineReducers({
firebase: firebaseStateReducer,
firestore: firestoreReducer
})
// Create store with reducers and initial state
const initialState = {}
const store = createStoreWithFirebase(rootReducer, initialState)
Profile
If you would like to have your users profiles go to Firestore instead of Real Time Database, you can enable the
useFirestoreForProfile
option when making store creator like so:
// react-redux-firebase config
const rrfConfig = {
userProfile: 'users',
useFirestoreForProfile: true // Firestore for Profile instead of Realtime DB
}
Queries
Firestore queries can be created in two ways:
- Automatically - Using
firestoreConnect
HOC (manages mounting/unmounting) - Manually - Using
get
, or by setting listeners withsetListeners
/setListener
(requires managing of listeners)
Automatic
firestoreConnect
is a React Higher Order component that manages attaching and detaching listeners for you as the component mounts and unmounts. It is possible to roll a similar solution yourself, but can get complex when dealing with advanced situations (queries based on props, props changing, etc.)
Examples
Basic query that will attach/detach as the component passed mounts/unmounts. In this case we are setting a listener for the
'todos'
collection:import { compose } from 'redux' import { connect } from 'react-redux' import { firestoreConnect } from 'react-redux-firebase' export default compose( firestoreConnect(['todos']), // or { collection: 'todos' } connect((state, props) => ({ todos: state.firestore.ordered.todos })) )(SomeComponent)
Create a query based on props by passing a function. In this case we will get a specific todo:
import { compose } from 'redux' import { connect } from 'react-redux' import { firestoreConnect } from 'react-redux-firebase' export default compose( firestoreConnect((props) => [ { collection: 'todos', doc: props.todoId } // or `todos/${props.todoId}` ]), connect(({ firestore: { ordered } }, props) => ({ todos: ordered.todos && ordered.todos[todoId] })) )(SomeComponent)
Manual
If you want to trigger a query based on a click or mange listeners yourself, you can use setListener
or setListeners
. When doing this, make sure you call unsetLister
for each listener you set.
Component Class
import React, { Component } from 'react'
import PropTypes from 'prop-types'
import { connect } from 'react-redux'
class Todos extends Component {
static contextTypes = {
store: PropTypes.object.isRequired
}
componentWillMount () {
const { firebase } = this.context.store
firebase.setListener('todos')
// firebase.setListener({ collection: 'todos' }) // or object notation
}
componentWillUnmount() {
const { firebase } = this.context.store
firebase.unsetListener('todos')
// firebase.unsetListener({ collection: 'todos' }) // or object notation
}
render () {
return (
<div>
{
todos.map(todo => (
<div key={todo.id}>
{JSON.stringify(todo)}
</div>
))
}
</div>
)
}
}
export default connect((state) => ({
todos: state.firestore.ordered.todos
}))(Todos)
Functional Components
It is common to make react components "stateless" meaning that the component is just a function.
import { compose } from 'redux'
import { withFirestore, isLoaded, isEmpty } from 'react-redux-firebase'
const Todos = ({ firestore, todos }) => (
<div>
<button onClick={() => firestore.get('todos')}>Get Todos</button>
{
!isLoaded(todos)
? 'Loading'
: isEmpty(todos)
? 'Todo list is empty'
: todos.map((todo) =>
<TodoItem key={todo.id} todo={todo} />
)
}
<div>
)
export default compose(
withFirestore,
connect((state) => ({
todos: state.firestore.ordered.todos
}))
)(Todos)
This can be useful, but then can limit usage of lifecycle hooks and other features of Component Classes.
recompose
helps solve this by providing Higher Order Component functions such as lifecycle
, and withHandlers
.
import { connect } from 'react-redux'
import { withFirestore } from 'react-redux-firebase'
import { compose, withHandlers, lifecycle } from 'recompose'
const enhance = compose(
withFirestore, // add firestore to props
withHandlers({
loadData: props => path => props.firestore.get(path)
}),
lifecycle({
componentWillMount() {
this.props.loadData('todos')
// this.props.firestore.get('todos') // equivalent without withHandlers
}
}),
connect((state) => ({
todos: state.firestore.ordered.todos,
}))
)
export default enhance(SomeComponent)
For more information on using recompose visit the docs
storeAs
storeAs
is not yet supported for the Firestore integration, but will be coming soon.
Populate
Populate is not yet supported for the Firestore integration, but will be coming soon.
More Info
The Firestore integration is build on redux-firestore
.