conventional-component v0.5.1
conventional-component

🏴 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.
:warning: :construction_worker: :wrench: Ready-for-use yet WIP :hammer: :construction: :warning:
- Chore: Finish unit tests. (NOTE: It's already tested via an example within
example/src).
Convention
export {
actions,
reducer,
withLogic,
Template,
Component,
REDUCER_NAME,
COMPONENT_NAME,
COMPONENT_KEY
}
export default ComponentRules
A Component...
- MUST
export defaultandexportitself.- MAY
exportthe name of the component asCOMPONENT_NAME. - MAY
exportthe 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
exportits 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)
withLifecycleStateLogicto achieve this.
- MUST dispatch an
- MAY implement
withRenderPropin 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
exportits 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
exportits reducer asreducer(state, action).- MAY
exportthe default name for its reducer asREDUCER_NAME.
- MAY
- MUST
exportits component template asTemplate.
Example
Component
The best way to understand the convention is to read some example code for an Input component.
Index
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 InputwithLogic(Template)
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 withLogicInput
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 InputRedux
The best way to understand how the state can be hoisted into Redux is to read some example code in which this is done.
Component
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
})Reducer
import { withReducerIdentity } from 'conventional-component'
import { reducer, COMPONENT_NAME, REDUCER_NAME } from '../../Input'
export { REDUCER_NAME }
export default withReducerIdentity(COMPONENT_NAME, reducer)Install
yarn add conventional-componentAPI
Component
connectToState(reducer, actionCreators) => Component => ConnectedComponent
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.
init(identity, props)
This must be called by conventional components during either the constructor or the componentWillMount lifecycle methods.
INIT is also exported alongside this.
receiveNextProps(identity, props)
This may be called by conventional components during the componentWillReceiveProps lifecycle method.
RECEIVE_NEXT_PROPS is also exported alongside this.
destroy(identity)
This must be called by conventional components during the componentWillUnmount lifecycle method.
DESTROY is also exported alongside this.
withActionIdentity(actionCreator) => IdentityReceivingActionCreator
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.
withRenderProp(props)
This is just a helper to improve the readability of the render prop and function-as-a-child patterns.
Redux
asConnectedComponent(conventionalComponentConfiguration) => ConnectedComponent
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.
createIdentifier(componentName, componentKey)
createIdentifiedActionCreators(identifier, actionCreators) => Props => IdentifiedActionCreators
createMapStateToProps(reducerName, identifier, structuredSelector)
withReducerIdentity(identifierPredicate, identifiedReducer) => IdentifiedReducer
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.