Manage component-specific state as you would global state via redux
The redux
library has changed the way that we manage state within our JavaScript applications, and is a wonderful tool. Like many great tools, though, developers see it as a hammer in a world of nails, using redux
for all state regardless of whether it should be global or not. The creator of redux
himself has even written about how redux
should be used selectively, because redux
is meant for global state.
react-local-redux
tries to strike a balance, leveraging the powers of the redux
paradigm but keeping state that does not need to be global scoped to the component that it should live in by creating a higher-order component that operates like a local store. It's usage should come naturally to those who have used react-redux
, you can use redux
middlewares and enhancers, and there are even helpers to remove boilerplate related to building action creators and reducers.
import { connectLocal } from "react-local-redux";
const actionCreators = {
decrement: () => ({ type: "DECREMENT" }),
increment: () => ({ type: "INCREMENT" })
};
const reducer = (state = { count: 0 }, action) => {
switch (action.type) {
case "DECREMENT":
return {
...state,
count: state.count - 1
};
case "INCREMENT":
return {
...state,
count: state.count + 1
};
default:
return state;
}
};
@connectLocal(reducer, actionCreators)
class App extends Component {
render() {
const {count, decrement, increment} = this.props;
return (
<div>
<div>Count: {count}</div>
<button onClick={decrement}>Decrement</button>
<button onClick={increment}>Increment</button>
</div>
);
}
}
connectLocal(reducer: function[, actionCreators: (function|Object)[, options: Object]]) => (ComponentToWrap: ReactComponent): ReactComponent
Decorator that accepts a reducer
function and actionCreators
, returning a function that accepts a ReactComponent
and returns a higher-order ReactComponent
.
If actionCreators
is an object of functions, it will automatically wrap those functions in dispatch
, and if it is a function then it will handle the following contract:
actionCreators(dispatch: function, ownProps: Object): Object
This should operate the exact same as mapDispatchToProps
in react-redux
.
Internally the reducer
will be used as local state, with each of the actionCreators
being used to update that state. Both the local state and the wrapped actionCreators
will be passed to the ComponentToWrap
as props, very much align the lines of connect
in the react-redux
package. If no actionCreators
are passed, the dispatch
method itself will be passed as a prop to the ComponentToWrap
.
Like connect
in react-redux
, you can pass an object of options
to customize how connectLocal
will operate:
pure: boolean
- is the component considered a "pure" component, meaning does it only update when props / state / context has changed based on strict equality
areOwnPropsEqual(currentProps: Object, nextProps: Object): boolean
- custom props equality comparator
areStatesEqual(currentState: Object, nextState: Object): boolean
- custom states equality comparator
areStatesEqual(mergedProps: Object, nextMergedProps: Object): boolean
- custom equality comparator for the merged state, actionCreators, and props passed to the component
Additionally, there are some options specific to connectLocal
that provide more functionality:
enhancer(fn: function): function
- store enhancer used in
redux.createStore()
- store enhancer used in
@connectLocal(reducer, actionCreators, {
enhancer: window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__({
name: 'MySpecialComponent'
})
})
middlewares: Array<function>
- array of redux middlewares to be appled to the
dispatch
method
- array of redux middlewares to be appled to the
@connectLocal(reducer, actionCreators, {middlewares: [reduxThunk]})
mergeProps(state: Object, actionCreators: Object, props: Object): Object
- custom method to merge state, actionCreators, and props into the props passed to the component
Additional imports are available for handling construction of action creators and reducers. The paradigm will be familiar to those who have used the redux-actions
library.
createActionCreator(type: string[, payloadHandler: function = identity[, metaHandler: function = noop]]): function
Create a Flux Standard Action in the same vein as redux-actions
.
const addStuff = createActionCreator("ADD_STUFF");
addStuff({ added: "stuff" });
// {payload: {added: 'stuff'}, type: 'ADD_STUFF'}
By default, the first parameter passed to the action creator will be used as the payload, but you can provide custom handlers for the payload and the meta if desired.
const addStuff = createActionCreator(
"ADD_STUFF",
({ added }) => added,
({ added }) => added === "stuff"
);
addStuff({ added: "stuff" });
// {meta: true, payload: 'stuff', type: 'ADD_STUFF'}
Additionally, if an error is provided as the payload, an error
property will be automatically set to true
on the action.
createReducer(handlers: Object, initialState: Object): function
Create a reducer based on a map of handlers, each themselves a reducer.
const handlers = {
SET_THING; (state, {payload}) => ({
...state,
thing: payload,
}),
};
const initialState = {
thing: null,
};
const reducer = createReducer(handlers, initialState);
const result = reducer(initialState, {payload: 'new thing', type: 'SET_THING'});
// {thing: 'new thing'}
Standard stuff, clone the repo and npm install
dependencies. The npm scripts available:
build
=> run webpack to build developmentdist
file with NODE_ENV=developmentbuild:minified
=> run webpack to build productiondist
file with NODE_ENV=productiondev
=> run webpack dev server to run example app / playgrounddist
=> runsbuild
andbuild:minified
lint
=> run ESLint against all files in thesrc
folderprepublish
=> runsprepublish:compile
when publishingprepublish:compile
=> runlint
,test:coverage
,transpile:es
,transpile:lib
,dist
test
=> run AVA test functions withNODE_ENV=test
test:coverage
=> runtest
but withnyc
for coverage checkertest:watch
=> runtest
, but with persistent watchertranspile:lib
=> run babel against all files insrc
to create files inlib
transpile:es
=> run babel against all files insrc
to create files ines
, preserving ES2015 modules (forpkg.module
)