π΄ React components which can have their state hoisted into Redux.
As I search for components with which to build an application, I frequently find otherwise excellent components which have inaccessible state. Often this means complications when integrating with Redux, due to how the state is passed in and emitted. Sometimes I will try to find a react-redux-
variant of a component, however these unfortunately lose the ease of integration of plain React components.
This is a proposal to build components out of reducers and actions and a library to help do so. The intention is to make it easy to write standardised components which (1) can be quickly installed into an app, and (2) can have their state hoisted into another state management library (Redux, MobX, etc) if this is required.
It's loosely inspired from the conventions within erikras/ducks-modular-redux
. It also has some similarities to multireducer
however due to its use of convention it's decoupled from redux.
- Chore: Finish unit tests. (NOTE: It's already tested via an example within
example/src
).
export {
actions,
reducer,
withLogic,
Template,
Component,
REDUCER_NAME,
COMPONENT_NAME,
COMPONENT_KEY
}
export default Component
A Component
...
- MUST
export default
andexport
itself.- MAY
export
the name of the component asCOMPONENT_NAME
. - MAY
export
the primary key of each of the components asCOMPONENT_KEY
(e.g.id
,name
).
- MAY
- MUST store its state using a reducer and some actions.
- MAY use the higher-order component (HOC)
connectToState(reducer, actions)
to achieve this.
- MAY use the higher-order component (HOC)
- MUST
export
its component logic as a higher-order component (HOC)withLogic(Template)
.- MUST dispatch actions during the lifecycle of the component to describe its state.
- MUST dispatch an
init(identity, props)
action on either construction orcomponentWillMount
. - MAY dispatch a
receiveNextProps(identity, props)
action oncomponentWillReceiveProps
. - MUST dispatch a
destroy(identity)
action oncomponentWillUnmount
. - MAY use the higher-order component (HOC)
withLifecycleStateLogic
to achieve this.
- MUST dispatch an
- MAY implement
withRenderProp
in order to render a user-specified render prop but otherwise fallback to rendering theTemplate
.
- MUST dispatch actions during the lifecycle of the component to describe its state.
- MUST
export
its action creator functions asactions
.- MUST either wrap each of its actions with
withActionIdentity(actionCreator)
or use action creators with the same signature.
- MUST either wrap each of its actions with
- MUST
export
its reducer asreducer(state, action)
.- MAY
export
the default name for its reducer asREDUCER_NAME
.
- MAY
- MUST
export
its component template asTemplate
.
The best way to understand the convention is to read some example code for an Input
component.
import Input, { COMPONENT_NAME, COMPONENT_KEY } from './Input'
import Template from './InputDisplay'
import withLogic from './withLogic'
import reducer, { REDUCER_NAME } from './reducer'
import * as actions from './actions'
export {
actions,
reducer,
withLogic,
Template,
REDUCER_NAME,
COMPONENT_NAME,
COMPONENT_KEY
}
export default Input
import React, { Component } from 'react'
import {
withRenderProp,
withLifecycleStateLogic
} from 'conventional-component'
import InputDisplay from './InputDisplay'
function withLogic(Template = InputDisplay) {
class Input extends Component {
onBlur = event => {
event.preventDefault()
return this.props.setFocus(false)
}
onChange = event => {
event.preventDefault()
return this.props.setValue(event.target.value)
}
onFocus = event => {
event.preventDefault()
return this.props.setFocus(true)
}
reset = event => {
event.preventDefault()
return this.props.setValue('')
}
render() {
const templateProps = {
...this.props,
onBlur: this.onBlur,
onChange: this.onChange,
onFocus: this.onFocus,
reset: this.reset
}
if (
typeof this.props.render === 'function' ||
typeof this.props.children === 'function'
) {
return withRenderProp(templateProps)
}
if (Template) {
return <Template {...templateProps} />
}
return null
}
}
return withLifecycleStateLogic({
shouldDispatchReceiveNextProps: false
})(Input)
}
export default withLogic
import { compose } from 'recompose'
import { connectToState } from 'conventional-component'
import InputDisplay from './InputDisplay'
import withLogic from './withLogic'
import reducer from './reducer'
import * as actions from './actions'
const COMPONENT_NAME = 'Input'
const COMPONENT_KEY = 'name'
const enhance = compose(connectToState(reducer, actions), withLogic)
const Input = enhance(InputDisplay)
export { COMPONENT_NAME, COMPONENT_KEY }
export default Input
The best way to understand how the state can be hoisted into Redux is to read some example code in which this is done.
import { asConnectedComponent } from 'conventional-component'
import {
actions,
withLogic,
Template,
REDUCER_NAME,
COMPONENT_NAME,
COMPONENT_KEY
} from '../../Input'
export default asConnectedComponent({
actions,
withLogic,
Template,
REDUCER_NAME,
COMPONENT_NAME,
COMPONENT_KEY
})
import { withReducerIdentity } from 'conventional-component'
import { reducer, COMPONENT_NAME, REDUCER_NAME } from '../../Input'
export { REDUCER_NAME }
export default withReducerIdentity(COMPONENT_NAME, reducer)
yarn add conventional-component
This function allows a reducer
to be used in place of standard this.setState
calls. It passes through the reducer state and the actions into a component.
It's implemented as a higher-order component (HOC) and therefore returns a function which takes a Component
. In fact it might look familiar as it is an analogue to react-redux#connect(mapStateToProps, actions)
, however with first argument being a standard redux reducer
and the second argument being an object of identity-receiving action creators (these could possibly have been created by wrapping normal action creators with withActionIdentity
).
withLifecycleStateLogic({ shouldDispatchReceiveNextProps }) => LogicComponent => LifecycleLogicComponent
This higher-order component (HOC) is provided to help dispatch the correct lifecycle
actions (e.g. init
and destroy
when a component is added or removed from the screen.)
It should be used within withLogic
to wrap any other component logic.
By default shouldDispatchReceiveNextProps
is false.
This must be called by conventional components during either the constructor
or the componentWillMount
lifecycle methods.
INIT
is also exported alongside this.
This may be called by conventional components during the componentWillReceiveProps
lifecycle method.
RECEIVE_NEXT_PROPS
is also exported alongside this.
This must be called by conventional components during the componentWillUnmount
lifecycle method.
DESTROY
is also exported alongside this.
If we choose to store the state within a redux store, we need to make sure that we can identify the state of each component by a key. Therefore, we should ensure that all actions contain an identity
property.
This is a helper which can be used to wrap normal action creators with this extra property. The first argument of these wrapped action creators is always the identity
property.
If you are using thunked actions or need more control for whatever reason, you can just conform to this type signature yourself. All you need to do is make sure that the first argument of each of your action creators is identity
and that the action which is returned contains this value within an identity
property.
This is just a helper to improve the readability of the render prop and function-as-a-child patterns.
To generate a redux ConnectedComponent
you just pass in the named exports of your conventional component.
The functions defined below are used internally by this to ensure that there is a mapping between a particular copy of the Component and its state.
As we need to be able to store the state of more than one copy of a particular component at a time, we need to make sure that the reducer which was previously written for a singular component is wrapped to understand the identity
property of our actions. We pass this reducer in as the second argument (e.g. identifiedReducer
).
Since many actions could contain an identity
property, we also need to make sure that we don't call the reducer unless the identity
matches a predicate. Therefore the first argument (e.g. identifierPredicate
) should either be the name of the component (e.g. COMPONENT_NAME
) or a predicate function that returns a boolean.