This module merges the logic of loading data on the client side and the server side.
With react, we use to do async things (generally loading data etc...) in component life cycles. When using redux with react, your connected components will generally be instances of "PureComponent", that means they are stateless. (read more about react pure component and why this concept was introduced) This difference is a big thing, With redux, you store your data into the state (that is connected to props), not into component state.
When using SSR (Server side rendering), components do not have life cycles, the "render" only occurs once (renderToString). To have the state totally hydrated, we need to render multiple times. During each render, components might dispatch actions to the store and the redux state might change, changing the result of the next render.
On client side, this is not a problem, when the state changes, connected components receives new props and will update.
So we need a module that can re-render until the state is ready without implementing a specific server logic. This module use the "componentWillMount" event on both env to dispatch loading actions, and trigger events to mark async components as loaded in the state. On server side, each time the store receives a event like this, he will re render, until all the components are ready, then send the response to browser.
- Server is rendering
- Component "Foo" dispatch an action to tell he is loading, and start doing jobs, result is going to the state to be connected later.
- Data are loaded, "Foo" dispatch an action to tell he is ready
- Store receive this action and examine the list of remaining actions
- The list of remaining actions appears to be empty, that mean the state should have all required data to render "foo", do a final render.
- OR
- The list of remaining actions is not empty, "foo" or an other component is loading => we wait for next state notification then re render => (Go to 1.)
This allow us to have a deep three rendering on server side.
With the new react router v4, routes are loaded in components, if you have some routes inside a component that is deep in your three, react won't see this route during the renderToString. react-async-load is a tool to build you state data using re-render on server side.
npm install redux-async-load --save
For Server side only
- store: Object
- the redux store implementing the reducer from this module and the rest of your data
- render: Function
- no args
- return string
- This method will be invoked for each render on server side, you must return the html of your app using renderToString from react-dom
- stateKey: string
- default: 'asyncLoad'
- This is the key of the reducer in your state
A Promise with the final html string of the application
stateSelector is used to find the asyncState inside the redux state.
- key: string
- The key of the asyncState in the redux state (default = 'asyncLoad')
- state: Object
- The redux state
isReady will check the asyncState to know if all components are loaded
- asyncState: Object
- This is the state of async components. Use the stateSelector to get it from the redux state
- return boolean true if all components are loaded
- loadId: string
- This is unique identifier of the operation in the asyncState
- return an action
- loadId: string The Id of the connected element
- shouldLoad:(props) => boolean
- props: Object The props passed to the component
- If this method return true, and if the component is not marked as loaded in state, the component will call the load function passed in props
- load: (props) => Promise<any> | null
- props Object The props passed to the component
- If this method returns a promise once load has been done, the loaded status of the component will be set automatically to false. If not, you will have to dispatch the action asyncSetStatus Normally, It should returns the result of an action
- shouldReload: (newProps, oldProps) => boolean
- newProps: Object The new props of the component
- oldProps: Object The old props of the component
- If this method return true, the component will call the load function passed in props
- render: (props) => React.Element If this property is present, It'll be used as the default render, and must return a single valid component.
- props: Object The props passed to the component
import {createStore, combineReducers} from 'redux'
import {reducer as asyncLoad} from 'redux-async-load'
export default createStore(combineReducers({
asyncLoad, // <-- add the reducer to your reducers
//[your reducers]
}), yourInitialState)
import React, {Component} from 'react'
import {connect} from 'react-redux'
import {ReduxLoader} from 'redux-async-load'
//this is a redux action creator to load data that will use index to load the user in state.
import {myLoadAction} from './action'
const User = props => <p>{props.user && props.user.name}</p>
/*
* Props that represent loading logic are implemented in the ReduxLoader
* If your props are dynamics and represent data or identifiers, you should implement them directly on the AsyncUser
* Model: AsyncUser(props) => ReduxLoader({...logicProps, props})
*/
const AsyncUser = props => <ReduxLoader
{/* Do load only if we do not have data */}
shouldLoad={props => !props.data}
{/* This tells the component how to load your data */}
{/* This is not invoked if you pass an action creator named "load" to connect because it will replace this props */}
load={props => myLoadAction(props.userId)}
{/* The component receives props, tell him if he must reload data when userId change */}
shouldReload={(props, oldProps) => (props.userId !== oldProps.userId)}
{/* This is the render method, use it to an altenrative way to render children */}
{/* If the component contains chilren, they will be present in props.children */}
render={props => <User {...props} />}
>
{/* OR */}
{/* This is the normal way of rendering an element */}
{/* <User /> will be cloned with all props */}
{/* If the prop render is present, children will be passed as props */}
<User />
</ReduxLoader>
//connect our component to the state
export default connect((state, {userId}) => {
return {
//loadId is required by Redux loader to set the loading flag in redux async state
loadId: userId
//We suppose your load action and his reducer will hydrate this part of the state with data (used buy AsyncUser)
data: state.user[userId]
}
}, ({load}))(AsyncUser)
export default props => (
<div>
<p>This component was preloaded on server, see the source code...</p>
<AsyncUser userId={12345}/>
</div>
)
import {renderToString} from 'react-dom'
import {renderAsync} from 'redux-async-load'
import {Provider} from 'react-redux'
import store from './store'
import App from './app'
import Html from './html' // <= your html template
import express from 'express'
const app = new express()
app.use((req, res, next) => {
//Your other jobs
//render async will subscribe to the store, and knows when the data are loaded
renderAsync(store, () => renderToString(<Provider store={store}><App /></Provider>))
//Render the generated html as you do normally
.then(appHtml => {
// Build your html structure with your favorite tool (like Helmet)
const html = <Html content={appHtml} />
res.send('<!doctype html>\n' + html)
})
.catch(next)
})
import {render} from 'react-dom'
import {Provider} from 'react-redux'
import store from './store'
import App from './app'
render(<Provider store={store}><App /></Provider>, document.getElementById('app-root'))
- add examples
- add tests
This is clearly experimental, and maybe removed if a better solution is found. You should take all precautions to prevent huge loading. If the schema of your app is very deep, you might want to preload data in a high order component that will prevent re rendering one more time