npm i react-redux-ui-tools -S
yarn add react-redux-ui-tools
- Implementation of UI state should be easy to achieve.
- UI state should be global.
- UI state should be easily managed from each component via an action.
- UI state should be easy to reset (manually and when unmounting).
In this example, each block-scope represents a component and each variable represents a UI state key:
{
// Everything inside this scope has access to filter and tags. This is our root UI component.
let filter = ''
let tags = []
// Imagine the following scopes are a list of to-do task components:
[
{
// Everything inside this scope has access to isSelected - plus all parent variables.
let isSelected = true
},
{
// This also has isSelected inside its scope and access to parent variables, but
// we want isSelected to be in a separate scope that can be manipulated independently from other
// siblings.
let isSelected = false
}
]
}
Wrap your root component with the react-redux-ui-tools ui
HOC. It's given a new scope for temporary UI variables which:
- are automatically bound to
this.props.ui
. - are automatically passed to any child component wrapped with the
ui
HOC. - will be automatically reset on componentWillUnmount (preventable via options).
- can be reset manually via the
this.props.ui.resetUI
function. - are updatable by any child component wrapped with the
ui
HOC.
This is powerful. Each component is reusable and can still affect UI state for parent components.
Specifically, the ui
HOC injects four props into your components:
uiKey
: The key passed to the HOC.ui
: The UI state for the component'suiKey
.updateUI
: A function accepting either a name/value pair or object which updates state withinuiKey
.resetUI
: A function which resets the state withinuiKey
to its default.
On instantiation and after calls to resetUI
, the HOC will set any default state specified.
On componentWillUnmount
the entire state in uiKey
will be set to undefined.
The ui
HOC takes a configuration object with the following keys:
The name of the key used in the UI reducer under which we store all state.
If specified as a function, will be passed props as the first argument and will only generate the key when the component is mounted. If not specified it will be autogenerated based on the component name suffixed with a random hex code. Components using the same key will share the same UI context!
Set to true if the UI state for this component should persist after componentWillUnmount
.
You must also explicitly define a key
for this component, otherwise the component will randomize the key and load new UI state on instantiation.
Note: All parent UI components also need to set this to true for this to take effect. Think of block-level scoping again — if a parent scope quits all child scopes are also out of context!
All UI variables need to be explicitly defined in the state object. This allows us to determine which scope a variable belongs to, as scope is inherited in the component tree.
Think of this as using let
inside your block scopes.
An additional reducer for this component's UI state.
mergeProps passed to react-redux's connect
.
options passed to react-redux's connect
.
const uiConfig = {
key: 'SomeComponent', // or as a function, e.g. props => `SomeComponent:${props.id}`
persist: true,
state: {
uiVar1: '',
uiVar2: (props, state) => state.router.location.query.searchTerm
},
reducer: (state, action) => {
switch( action.type ) {
case '@@reduxReactRouter/routerDidChange':
if ( action.payload.location.query.extra_filters ) {
return {
...state,
extraFilters: true
}
}
return state
default:
return state
}
},
mergeProps: () => ({}),
options: {}
}
// rootReducer.js
import { combineReducers } from 'redux'
import { reducer as ui } from 'react-redux-ui-tools'
const createRootReducer = reducers => {
return combineReducers( {
...reducers,
ui,
} )
}
export default createRootReducer
// SomeComponent.js
import React, { Component } from 'react'
import ui from 'react-redux-ui-tools'
class SomeComponent extends Component {
render() {
const {
someUIFlag,
updateUI,
resetUI
} = this.props.ui
return (
<button onClick={updateUI('someUiFlag', !someUiFlag)}>
Toggle flag in UI state
</button>
<button onClick={resetUI()}>
Reset UI state
</button>
)
}
}
const uiConfig = { /* as above */ }
export default ui( uiConfig )( SomeComponent )
NOTE: If you wish to use Immutable.JS for UI state, add the react-redux-ui-tools reducer with a third parameter set to Immutable.JS' Map class, e.g.:
import { combineReducers } from 'redux'
import { Map } from 'immutable'
import { reducer as ui } from 'react-redux-ui-tools'
const createRootReducer = reducers => {
return combineReducers( {
...reducers,
ui: (state, action) => ui( state, action, Map )
} )
}
export default createRootReducer
NOTE: You may also use the HOC as a decorator:
// SomeComponent.js
import React, { Component } from 'react'
import ui from 'react-redux-ui-tools'
const uiConfig = { /* as above */ }
@ui( uiConfig )
class SomeComponent extends Component {
render() {
const {
someUIFlag,
updateUI,
resetUI
} = this.props.ui
return (
<button onClick={updateUI('someUiFlag', !someUiFlag)}>
Toggle flag in UI state
</button>
<button onClick={resetUI()}>
Reset UI state
</button>
)
}
}
export default SomeComponent
import React, { Component } from 'react'
import ui from 'react-redux-ui-tools'
// Component A gets its own context with the default UI state below.
// this.props.ui will contain this state map.
@ui( {
state: {
// use the filter query parma via redux-router as the default
filter: (props, state) => state.router.location.query.filter,
isFormVisible: true,
isBackgroundRed: false
}
} )
class A extends Component {
render() {
return (
<div>
// This will render '{ "filter": '', isFormVisible: true, isBackgroundRed: false }'
<pre><code>{ this.props.ui }</code></pre>
// Render child B
<B />
</div>
)
}
}
// B inherits context from A and adds its own context.
// This means that this.props.ui still contains A's state map.
@ui()
class B extends Component {
render() {
return (
<C />
)
}
}
// C inherits context from its parent B. This works recursively,
// therefore C's this.props.ui has the state map from A plus someChildProp.
// Setting variables within C updates within the context of A; all UI
// components connected to this UI key will receive the new props.
@ui( {
state: {
someChildProp: 'foo'
}
} )
class C extends Component {
render() {
return (
<div>
<p>I have my own UI state C and inherit UI state from B and A</p>
<p>
If I define variables which collide with B or A mine will
be used, as it is the most specific context.
</p>
</div>
)
}
}
MIT license.
Developed by Jonathan Horowitz.
Based on the original redux-ui code written by Franklin Ta and Tony Holdstock-Brown.