state-decorator
v7.3.0-optim.1
Published
React state management library
Downloads
1,124
Maintainers
Readme
The StateDecorator is a set of React hooks that manage a complex component state in an easy, testable and deterministic way.
Features
- Deterministic state changes (clear separation of effects and side effects)
- Ease asynchronous actions state changes (loading / error states, success & error handlers, parallel actions management, optimistic updates...)
- Easily and efficiently share slices of state.
- Easily testable (test framework provided)
- Easily update state from or react to props changes
- Ease debugging (trace state changes, dev tools)
- Improve code conciseness (no boiler plate code)
- Strongly typed (Typescript)
V7: Partial state and new hooks
- React 18 support
- All effects now return a partial state that will be merged with current state.
- More documentation on global and local store use cases.
- useLocalStore now have a parameter to specify if the React component must re-rerender after a state change.
- New useStoreContextSlice hook was added to get a slice of a store in a context.
- New onMountDeferred to execute initialization code after initial render
- New effect helpers
- See migration details in migration section.
Getting started
Installation
npm install state-decorator
or
yarn add state-decorator
Getting started
Example
import { useLocalStore, StoreConfig } from "state-decorator";
// Typings
type User = {
firstName: string;
lastName: string;
};
type State = {
selectedUserId: string;
user: User;
};
type Actions = {
selectUser: (id: string) => void;
selectAndLoadUser: (id: string) => void;
loadUser: () => Promise<User>;
};
type Props = {
userIds: string[];
};
// Initial state & actions
const config: StoreConfig<State, Actions, Props> = {
getInitialState: () => ({
selectedUserId: null,
user: null,
}),
actions: {
// synchronous action (simple form)
selectUser: ({ args: [id] }) => ({ selectedUserId: id }),
// a synchronous action with side effects
selectAndLoadUser: {
// effects contain only immutable state changes
effects: ({ args: [id] }) => ({ selectedUserId: id }),
// side effects (ie. other effects than state changes), using "actions" alias
sideEffects: ({ a }) => {
a.loadUser();
},
},
// asynchronous action
loadUser: {
// promise provider, can use action context (ctx) to get current state, args, props etc.
getPromise: (ctx) =>
Promise.resolve({ firstName: "John", lastName: "Doe" }),
// state changes
effects: ({ res }) => ({ user: res }),
// state changes if error
errorEffects: ({ error }) => ({ user: null }),
// side effects here
sideEffects: ({ res, actions }) => {},
// side effects if error
errorSideEffects: ({ err, actions }) => {},
// many more options available here, read the doc!
},
},
};
// Bind to react component and use state/actions
export function App(props: Props) => {
const { state, actions } = useLocalStore(config);
return (
<div>
{state.user && (
<div>
{state.user.firstName} {state.user.lastName}
</div>
)}
<div>
{props.userIds.map((userId) => (
<button
onClick={() => {
actions.selectAndLoadUser(userId);
}}
>
{userId}
</button>
))}
</div>
</div>
);
};
React hooks
| Hook | Purpose | Returns | Is component refreshed on store change? | Is store destroyed on unmount? | | -------------------- | ------------------------------------------------------------- | ------- | --------------------------------------- | ------------------------------ | | useLocalStore | Create a store and binds it to the react component. | Store | if refreshOnUpdate is set | Y | | useStore | Binds a store instance to the react component. | Store | yes | N | | useBindStore | Binds a store instance to the react component (inject props). | Store | yes | N | | useStoreSlice | Get a slice of a store. | Slice | yes | N | | useStoreContextSlice | Get a slice of a store stored in a context | Slice | If slice has changed only | N |
Initial state
The initial state is build using a function provided to the store.
const config: StoreConfig<State, Actions, Props> = {
getInitialState: (p) => ({ counter: 0 }),
actions: {
/* ... */
},
};
export const CounterContainer = () => {
const { state, actions } = useLocalStore(config);
return <CounterView {...state} {...actions} />;
};
Action context
The action context contain all the data needed to implement the actions. It contains the action arguments and the injected data.
| Property | Alias | Availability | Description | | ------------- | ----- | ------------------------------ | ------------------------------------------------------- | | state | s | everywhere | Current state | | args | | action related | Action arguments | | props | p | everywhere | Current bound component props | | actions | a | side effects / onMount | Store actions (decorated) | | result | res | promise effects / side effects | The promise result | | error | err | error effects / side effects | The promise error | | derived | ds | everywhere | The derived state (from props/state) | | promiseId | | asynchronous actions callbacks | Parallel asynchronous actions promise identifier | | notifyWarning | | side effects | Injected function if set at store or global level | | clearError | | side effects | Clear an error from ErrorMap | | indices | | onPropsChange | Indices in the getDeps array of changed props | | isInit | | onPropsChange | Whether the onPropsCHange is executed during init phase |
Actions
The StateDecorator is taking a list of actions and decorate them to inject state, props, actions etc. and then expose the decorated actions that simply takes the function arguments defined in the type.
Warning: The actions are considered static and cannot be changed dynamically. Use state/props in your actions implementation to have different logic depending on some inputs.
Synchronous actions
Short form
action: (ctx) => ctx.state, // changes to be merged to current state
Example:
const config: StoreConfig<State, Actions, Props> = {
actions: {
setText: ({ args: [newText] }) => ({ text: newText }),
},
};
Regular form
action: {
// debounce effect and side effects if > 0
debounceTimeout: 0,
// debounce side effects if > 0
debounceSideEffectsTimeout: 0,
effects: (ctx) => ctx.state, // state changes
sideEffects: (ctx) => {
// side effect code
},
}
Example:
const config: StoreConfig<State, Actions, Props> = {
actions: {
setText: {
effects: ({ args: [newText] }) => ({ text: newText }),
// use either debounceTimeout or debounceSideEffectsTimeout
// debounceTimeout: 5000,
debounceSideEffectsTimeout: 5000,
sideEffects: ({ a }) => {
a.saveDocument();
},
},
},
};
Notes
- If the effects function returns
null
, the side effects, if any, are not executed when the action is called. - If no effects function is defined but a side effects function is provided, the side effects will be executed when the action is called.
Cancel action
To cancel effects/side effects just return null
instead of new state in effects.
Asynchronous actions
action: {
preEffects: (ctx) => ctx.state, // state changes before promise
optimisticEffects: (ctx) => ctx.state, // optimistic state changes before promise, will be reverted if promise fails. Be sure to install optimistic middleware !
getPromise: (ctx) => Promise.resolve(), // promise provider
effects: (ctx) => ctx.state, // state changes if success
errorEffects: (ctx) => ctx.state, // state changes if error
sideEffects: (ctx) => { /* side effect code if success */ },
errorSideEffects: (ctx) => { /* side effect code if failure */ },
getSuccessMessage: (ctx) => 'Success', // if a notifySuccess is set locally or globally
getErrorMessage: (ctx) => 'Error', // if a notifyError is set locally or globally
abortable: true, // inject a abort signal in the promise invocation context and allow to abort promise
conflictPolicy: ConflictPolicy.KEEP_LAST, // behavior if action is called while a promise is ongoing
getPromiseId: (args) => null, // parallel actions promise identifier (see conflicting actions)
retryCount: 1,
retryDelaySeed: 1000,
}
Example:
const config: StoreConfig<State, Actions, Props> = {
actions: {
loadList: {
getPromise: () => fetch('https://myapi/myservices').then((res) => (res.ok ? res.json() : Promise.reject())),
effects: ({ res }) => ({ list: res }),
sideEffects: ({ a }) => {
a.otherAction();
},
getErrorMessage: () => 'Cannot load list',
},
},
};
If
null
is returned by getPromise, the action is not executed. It allows to cancel an action depending on state or props etc.getGetPromise is a shortcut that sets the conflict policy to ConflictPolicy.REUSE and retryCount to 3.
To cancel any effects just return
null
instead of new state in xxxEffects.
Asynchronous action lifecycle
graph TD
A(Call action) --> preEffects
preEffects --> optimisticEffects
optimisticEffects --> getPromise
getPromise --> promiseSuccess{Success?}
promiseSuccess -->|Yes| effects[effects]
effects --> notifySuccess[notifySuccess]
notifySuccess --> sideEffects[sideEffects]
sideEffects --> promiseResolved[Promise is resolved]
promiseResolved --> Z(End)
promiseSuccess -->|No| errorEffects[errorEffects]
errorEffects --> notifyError[notifyError]
notifyError --> errSideEffetcs[errorSideEffects]
errSideEffetcs --> errorManaged{Error managed?}
errorManaged --> |Yes| promiseResolved
errorManaged --> |No| promiseRejected
promiseRejected --> Z
check if a previous call to this action is ongoing,
- if yes, check if the conflict policy is parallel,
- if yes, let the flow continue.
- if no, return a new promise according to the conflict policy.
- if yes, check if the conflict policy is parallel,
preEffects: state changes before promise execution.
optimisticEffects: optimistic state changes before promise execution, will be rollback if promise fails.
getPromise: get a promise from action context:
- if promise is resolved:
- effects: update state from action contexts and the result of the promise
- if notifySuccess is set, call it with getSuccessMessage()
- sideEffects: trigger a side effect with no change on state.
- if promise is rejected (or aborted):
- if the action was optimistic, revert the optimisticEffects change and replay all following actions.
- errorEffects: change state from promise arguments and returned error.
- if notifyError is set, call it with getErrorMessage().
- errorSideEffects: trigger a side effect with no change on state.
- Global asyncErrorHandler is called (with a flag indicating that the error was managed or not, see Error management for more details).
- if promise is resolved:
If a conflicting action is stored, process it.
Loading state
The loading state of asynchronous actions is automatically computed.
const { isLoading, loading, loadingMap } = useLocalStore(getInitialState, actionsImpl);
// loading is set to true is at least one asynchronous action is ongoing.
// loadingMap is action name to boolean or undefined map
// isLoading is a function that accepts 1+ arguments
// It returns true, if at least one of the specified actions is loading.
const isOneLoading = isLoading('action1', 'action2', ['parallelAction', 'promiseId']);
Success / error notifications
A notification function can be called when the asynchronous action succeeded or failed.
- Set notification function on success/error:
- Set notifyError and/or notifySuccess using in global configuration OR
- specify a function to notifySuccess and/or notifyError options of the store.
- In each asynchronous action, set:
- getErrorMessage (message built from the error and action context).
- getSuccessMessage (message built from the action context)
// global notification functions
setGlobalConfig({
notifySuccess: (msg) => console.log(msg),
notifyError: (msg) => console.log(msg),
});
// local notification functions
const store = useLocalStore(getInitialState, actionsImpl, props, {
notifySuccess: (msg) => console.log(msg),
notifyError: (msg) => console.log(msg),
});
// action implementation
const config: StoreConfig<State, Actions, Props> = {
actions: {
loadList: {
getPromise: () => new Promise((_, reject) => setTimeout(reject, 500, new Error('Too bad'))),
getSuccessMessage: () => 'success!',
getErrorMessage: ({ err }) => `Error: ${err.message}`,
},
},
};
Error management
Effects and side effects
When an asynchronous action fails, if the state needs to be updated, set the errorEffects property of the asynchronous action.
If some actions need to be triggered, set errorSideEffects.
const actions: StoreActions<State, Actions> = {
loadList: {
getPromise: () => new Promise((_, reject) => setTimeout(reject, 500, new Error('Too bad'))),
errorEffects: ({ error }) => ({ error: error.message, list: [] }),
getErrorMessage: ({ err }) => err.message,
errorSideEffects: ({ error }) => {
Logger.log(error);
},
},
};
Promise result
If getPromise is failing and the error is managed, the action promise will be returned as resolved to prevent other error managements (console traces…) to be executed, otherwise it will be returned as rejected.
An error is managed if:
- errorEffects is defined and is not returning
null
. - or errorSideEffects is defined,
- or notifyError is defined and getErrorMessage is not returning
null
, - or isErrorManaged is set to
true
.
If the error is managed but you need to have the promise to be rejected, set rejectPromiseOnError to true
.
Global asyncErrorHandler is called for each error with this managed flag for global error management (logging).
Error state
In order to display UI after an error, the last error of each asynchronous action is available on errorMap.
const { errorMap } = useLocalStore(getInitialState, actionsImpl);
if (errorMap.action1)) {
// show retry action
}
if (errorMap.action1 instanceof MyCustomError){
// show retry action with custom error handling
}
Recipe: When only using the errorMap to manage errors (ie. there's no effects nor message and so on), the error is considered not managed resulting in a rejected promise and global asyncErrorHandler called with managed flag set to false. In that case, set isErrorManaged to true
to specify that the error is managed externally.
The error map is updated when the action is called again, or call clearError in a side effect.
Conflicting actions
The StateDecorator is managing the asynchronous action calls one at a time (default) or in parallel.
When a action call occurs while the same action previous call's promise is ongoing, we call such actions calls conflicting actions.
In lots of situations, the UI is disabled using the loading action state, but in other situations one may want to manage such use case (a search bar, autosave feature of an editor etc.).
There are several policies to handles call to an action while there's a previous call to this action is ongoing.
This is controlled by the conflictPolicy property of an asynchronous action.
It can takes the following values (use ConflictPolicy enum), choose the one the more suited to your use case:
- ConflictPolicy.IGNORE: The conflicting action calls are simply ignored.
- ConflictPolicy.REJECT: Conflicting action calls are unwanted, they will be rejected with an error.
- ConflictPolicy.KEEP_ALL (default): All conflicting action calls will be chained and executed one after the other.
- ConflictPolicy.KEEP_LAST: Only the more recent conflicting action call will be executed after the previously ongoing call is resolved. Use case: editor with auto save feature.
- ConflictPolicy.REUSE: If an action is already ongoing, the promise is reused, if the arguments are the same (shallow comparison). Otherwise fallback to ConflictPolicy.KEEP_ALL. Useful for GET requests.
- ConflictPolicy.ABORT: Current call is aborted and new call is executed right after. abortable is set to true automatically if this policy is set.
- ConflictPolicy.PARALLEL: Actions are executed in parallel.
- Use case: several calls with different parameters.
- A getPromiseId function must be provided to assign an identifier to each call from call arguments.
- The isLoading function will return the loading state for each promise identifier.
Retry
If the promise of an asynchronous action fails, the store can retry this promise before failing.
The following properties action are used:
- retryCount: Number of retries in case of "retry error".
- retryDelaySeed: Seed of delay between each retry in milliseconds. The applied delay is retryDelaySeed * retry count. Default is 1000ms.
- isTriggerRetryError: A function that returns if the passed error will trigger a retry or if the action fails directly. Default function is returning true for TypeError instances.
Optimistic actions
⚠ If you are using optimistic effects, make sure to set the optimisticActions middleware to your store.⚠
Using global configuration (for all stores):
import { setGlobalConfig } from 'state-decorator';
import { optimisticActions } from 'state-decorator/middlewares';
setGlobalConfig({
defaultMiddlewares: [optimisticActions()],
});
On a specific store:
import { createStore, StoreConfig } from 'state-decorator';
import { optimisticActions } from 'state-decorator/middlewares';
const config: StoreConfig<State, Actions, Props> = {
getInitialState: () => ({
/* ... */
}),
actions: {
/* ... */
},
middlewares: [optimisticActions()],
};
const store = createStore(config);
An optimistic action assumes that the action will, most of the time, succeed. So it will apply the effects as soon as the asynchronous action is called (as opposite to the regular effects which are applied when the promise is resolved).
If the action succeeds the effects, if any, will be called anyway.
If the action fails, the state will be recomputed to undo this action.
The undo strategy is the following:
- Retrieve the state before the optimist action.
- Replay all the subsequent actions effects.
- If there are other optimistic actions ongoing, update their state before action.
Example:
const config: StoreConfig<State, Actions, Props> = {
actions: {
deleteItem: {
getPromise: ([id]) => new Promise((resolve) => setTimeout(resolve, 500)),
optimisticEffects: ({ args: [id] }) => ({ list: list.filter((item) => item.id === id) }),
errorEffects: ({err} => ({ err: error.message }),
},
},
};
Notes:
- As optimistic actions can be expensive (clone state, save all actions), make sure to use optimistic reducer for a promise that will, most of the time, returns in a short amount of time.
- To update the state before the action, prefer preEffects. optimisticEffects purpose is optimistic actions only. Optimistic actions are not changing the global loading state and are more expensive because the subsequent actions are saved in order to be able to revert the optimistic action in case of failure.
Abort asynchronous action
Ongoing asynchronous actions can be aborted using, under the hood, the AbortController, if you target browsers which support it.
- Action implementation:
- An asynchronous action must be marked as abortable.
- Then, in the promise method, the injected abortSignal have to be used to add a listener on abort of action and reject the promise with a
new DOMException('Aborted', 'AbortError')
. Or pass it to the fetch, i.e.fetch(url, { signal })
.
- Usage:
- In the result of the useLocalStore, use the abortAction with an action name to abort the action
- An abort is a specific failure of an action, ie. errorEffects and errorSideEffects will be called. Use the error type and name to distinguish an aborted action from a regular failed action.
const config: StoreConfig<State, Actions, Props> = {
actions: {
onAction: {
abortable: true,
preEffects: () => ({ isError: false, isSuccess: false, isAborted: false }),
getPromise: ({s, args: [willCrash] abortSignal}) =>
new Promise((resolve, reject) => {
const timeout = window.setTimeout(willCrash ? reject : resolve, 2500, willCrash ? new Error('boom') : 'result');
abortSignal.addEventListener('abort', () => {
window.clearTimeout(timeout);
reject(new DOMException('Aborted', 'AbortError'));
});
}),
errorEffects: ({ err}) => (err.name === 'AbortError' ? { isAborted: true } : { isError: true }),
effects: () => ({ isSuccess: true }),
errorSideEffects: ({ err }) => {
if (e.name === 'AbortError') {
console.log('AbortError side effect');
} else {
console.log('Other error side effect');
}
},
},
},
};
Note: ongoing abortable actions will be automatically aborted on store destruction (during React component unmount for example). The list of aborted actions is available in the onUnmount callback in the store's options.
Helper effect functions
For common / simple use cases, some helpers functions are provided to write basic action effects and improve actions readability.
Example
import { setArgIn } from 'state-decorator/helpers';
type State = {
myProp: string;
};
type Actions = {
setMyProp: (v: string) => void;
};
const actions: StoreActions<State, Actions> = {
setMyProp: setArgIn('myProp'),
// same as
// setMyProp: ({ args: [v] }) => ({ myProp: v }),
setMyProp2: setArgIn('myProp', 'isDirty'),
// same as
// setMyProp2: ({ args: [v] }) => ({ myProp: v, isDirty: true }),
};
| Name | Purpose | Action signatue | | -------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------ | | setArgIn | Effect function to update a state property from action first argument | (v:T) => void | | setTrueIn | Effect function to update a boolean state property to true | () => void | | setFalseIn | Effect function to update a boolean state property to false | () => void | | toggleProp | Effect function to toggle a boolean state property | () => void | | setArgsInMap | Effect function to update a state property of type Record<string, T> using first action argument as key and second as value | (id: string, v:T) => void | | setArgsInArray | Effect function to update a state property of type T[] using first action argument as index and second as value | (index: number, v:T) => void | | setResIn | Effect function of asynchronous action to update a state property with the result of the promise | () => Promise | | setResInArray | Effect function to update a state property of type T[] using index from first action argument as index and as value the result of the promise | (index: number) => Promise | | setResInMap | Effect function to update a state property of type Record<string, T> using key from first action argument as index and as value the result of the promise | (id: string) => Promise |
Call actions on mount
When store is created (for example whe React component in mounted), the onMount option is called.
import { StoreConfig } from 'state-decorator';
const config: StoreConfig<State, Actions, Props> = {
getInitialState,
actions,
onMount: ({ p }) => {
p.loadParentList();
},
};
After React component is mounted using useLocalStore, the onMountDeferred option is called (in a useEffect). The onMount is called during the render of the hooks. If the onMount code is containing a function that is changing parent state, React will not update the parent state and will issue a warning.
import { StoreConfig } from 'state-decorator';
const config: StoreConfig<State, Actions, Props> = {
getInitialState,
actions,
onMountDeferred: ({ p }) => {
p.loadParentList();
},
};
If an initial action is launched each time a property changes, consider using the onMount flag of the onPropsChange entry.
Call actions on store destruction / unmount
When the store is destroyed (for example on React component unmounted), the onUnmount option is called.
import { StoreConfig } from 'state-decorator';
const config: StoreConfig<State, Actions, Props> = {
getInitialState,
actions,
onUnmount: ({ s, abortedActions }) => {
// ex: store s in local storage
// ex: clean cache of aborted actions
},
};
Multiple props change configuration
The store can manage several props change configurations to have a better granularity.
Effects are triggered in order and reuse previously computed state
Side effects are invoked after all effects using computed state
import { StoreConfig } from 'state-decorator';
const config: StoreConfig<State, Actions, Props> = {
getInitialState,
actions,
onPropsChange: [
{
getDeps: (p) => [p.id],
effects: ({ state, props, indices }) => ({ ...state, selectedId: props.id }),
},
{
getDeps: (p) => [p.otherId],
sideEffects: ({ p, a }) => {
a.loadOtherItem(p.id);
},
},
{
getDeps: (p) => [p.otherProp],
effects: ({ props, indices }) => ({ otherProp: props.otherProp }),
sideEffects: ({ p, a }) => {
a.otherSideEffect(p.otherProp);
},
},
],
};
OnMount
If the onMount flag is set on the props change configuration, the effects and side effects will be executed at the store creation when the component is mounted.
It allows to trigger same actions at creation time and when the dependencies props are changed.
import { StoreConfig } from 'state-decorator';
const config: StoreConfig<State, Actions, Props> = {
getInitialState,
actions,
onMount: ({ p }) => {
a.loadUser(p.userId);
},
onPropsChange: [
{
getDeps: (p) => [p.userId],
effects: ({ p }) => ({ userId: p.userId }),
sideEffects: ({ p, a }) => {
a.loadUser(p.userId);
},
},
],
};
// SAME AS
import { StoreConfig } from 'state-decorator';
const config: StoreConfig<State, Actions, Props> = {
getInitialState,
actions,
onPropsChange: [
{
getDeps: (p) => [p.userId],
effects: ({ p, isInit }) => (isInit ? null : { userId: p.userId }),
sideEffects: ({ p, a }) => {
a.loadUser(p.userId);
},
onMount: true,
},
],
};
Caution: the effects are executed too. Use the isInit parameter is passed to the context to apply or not state change.
Recipes
- Use several prop change configurations to separate dependencies between props and have simple definition.
- Use onMount flag to have a more systematic and simpler code.
Global / local Stores and State sharing
There are two ways to share state and actions:
- Using regular props.
- Using state slices.
There are two types of stores:
Global store:
- store is created as an instance (createStore) and it is exported from the JS module.
- it can be imported in any other JS module.
- store lifespan is equal to the application one (ie. it's a singleton).
Local store:
- store is created by a hook (useLocalStore) and is bound to a React component (each instance of this component owns its own store).
- state and actions can be shared using props or sharing the store itself (see example below)
- store is destroyed when its owner React component is unmounted.
Using props
- Bind a React component to make store state / actions available on React component.
- Any change in the store (state, loading actions, ...) will trigger a refresh of the bound React component.
- Pass state / actions using regular React props.
- Use React.memo to prevent unecessary React re-renders if needed.
Global store
// Declare typings & actions as above
// Create a store in a dedicated file, named GlobalStore.ts for example
export const store = createStore(getInitialState, userAppActions);
//----------------------
// In another file
import { memo } from 'react';
import { useStore } from 'state-decorator';
import store from './GlobalStore';
// Bind to react component
// Each time the store state changes, a re-render is done.
export function Container(props: Props) {
const { state, actions } = useStore(store, props);
return (
<div>
<Title text={state.title} />
<Subtitle text={state.subtitle} />
</div>
);
}
// Simple presentation components
// Use memo to prevent re-render if another part of the state has changed than "title"
const Title = memo(function Title(props: { title: string }) {
return <div>{props.text}</div>;
});
const Subtitle = memo(function Subtitle(props: { subtitle: string }) {
return <div>{props.subtitle}</div>;
});
Local Store
import { memo } from 'react';
import { useLocalStore } from 'state-decorator';
// Declare typings & actions as above
// Create and bind to react component
// Each time the store state changes, a re-render is done.
export function Container(props: Props) {
const { state, actions } = useLocalStore(config, props);
return (
<div>
<Title text={state.title} />
<Subtitle text={state.subtitle} />
</div>
);
}
// Simple presentation components
// Use memo to prevent re-render if another part of the state has changed than "title"
const Title = memo(function Title(props: { title: string }) {
return <div>{props.title}</div>;
});
const Subtitle = memo(function Subtitle(props: { subtitle: string }) {
return <div>{props.subtitle}</div>;
});
State slices
Using React props implies useless re-renders and optionally memoization.
To prevent this, React provides context hooks.
The problem is that if the context contains a complex state with lots of props, if any of these props is changing, all components that uses this context will be re-rendered.
To overcome this problem, the StateDecorator provides a useStoreSlice and useStoreContextSlice hook that allow to define and extracts a slice of the state and trigger a re-render of the React component only if the slice has changed.
Global Store
// Declare typings & actions as above
// Create a store in a dedicated file, named GlobalStore.ts for example
export const store = createStore(getInitialState, userAppActions);
//----------------------
import { memo } from 'react';
import { useStoreSlice } from 'state-decorator';
import store from './GlobalStore';
export function Container(props: Props) {
return (
<div>
<Title />
<Subtitle />
</div>
);
}
function Title() {
// extracts the "title" slice
// component is refresh only if title is changed in the store
const { title } = useStoreSlice(store, ['title']);
return <div>{title}</div>;
}
function Subtitle(props: { subtitle: string }) {
const { subtitle } = useStoreSlice(store, ['subtitle']);
return <div>{subtitle}</div>;
}
Local Store
IMPORTANT: the context children MUST be with no prop and memoized to prevent refreshes (React can display warning on the console)
// ------- stores/MyStore.ts --------
import React from 'react';
import { useLocalStore, useStoreContextSlice } from 'state-decorator';
// Declare typings & actions as above
// Create a store context
type StoreContextProps = StoreApi<State, Actions, Props>;
export const StoreContext = createContext<StoreContextProps>(null);
// The StoreContext will allow to access to the store
// But this context is NOT refreshed if the store state is changed
export function StoreContextProvider(p: { children: any; propIn: string }) {
const { children, ...props } = p;
const store = useLocalStore(config, props);
return <StoreContext.Provider value={store}>{children}</StoreContext.Provider>;
}
// In a container file ------------------------
import { StoreContextProvider } from 'stores/MyStore';
// If Container is destroyed store is destroyed
export function Container(props: Props) {
return (
<StoreContextProvider {...props}>
<SubComponent />
<SubComponent2 />
</StoreContextProvider>
);
}
// Component deeper in the component tree...
import { StoreContext } from 'stores/MyStore';
function Title() {
// extracts the "title" slice
// component is refresh only if title is changed in the store
const { title } = useStoreContextSlice(StoreContext, ['title']);
// same as
// const store = useContext(StoreContext)
// const { title } = useStoreSlice(store, ['title']);
return <div>{title}</div>;
}
function Subtitle(props: { subtitle: string }) {
const { subtitle } = useStoreContextSlice(StoreContext, ['subtitle']);
return <div>{subtitle}</div>;
}
Derived state
A derived state is a state that can be deduced from state, props or derived state.
import { createStore } from 'state-decorator';
type State = {
value: number;
};
type Actions = {
add1: () => void;
};
type Props = {
propIn: number;
};
type DerivedState = {
derivedProp: number;
derivedProp2: number;
derivedProp3: number;
};
const store = createStore<State, Actions, Props, DerivedState>({
getInitialState: () => ({ value: 0 }),
actions: {
add1: ({ s }) => ({ ...s, value: s.value + 1 }),
},
derivedState: {
derivedProp: {
// get the list of dependencies to be checked to trigger the computation of the derived state
// if state.value AND/OR props.prop1 changes, derivedProp is recomputed
getDeps: ({ state, props }) => [state.value, props.propIn],
// compute derived state from state & props (use short aliases)
get: ({ s, p }) => s.value * p.propIn,
},
derivedProp2: {
// derived prop from another derived prop
derivedDeps: ['derivedProp'],
// compute derived state from state & props (use short aliases)
get: ({ ds }) => ds.derivedProp * 2,
},
derivedProp2: {
// depends on state, props and another derived state
getDeps: ({ s, p }) => [s.value, p.propIn],
derivedDeps: ['derivedProp'],
get: ({ s, p, ds }) => s.value * p.propIn * ds.derivedProp,
},
},
});
export function App(props: Props) {
const { state } = useStore(store, props);
// state contains the store state and derived state
return (
<div>
{state.derivedProp} {state.derivedProp2} {state.derivedProp3}
</div>
);
}
Recipes
- Derived state is the same as using useMemo in sub component.
- If the derived state is needed in one component only, it may prove better to use a useMemo in this component to save memory when the component is unmounted and store not.
- In general, do not compute derived values directly in state, it's error prone as you can forget some places or implement different logic.
- Use derived state especially if this state is shared accross several sub components
Global configuration
Overrides configuration to set properties that will be used by all stores.
// all parameters are optional
setGlobalConfig({
// Used to clone state and props when managing optimistic conflicting actions.
clone: defaultCloneFunc,
// Compare to objets and returns if they are equal (slices).
comparator: shallow,
// Callback function to handle asychronous actions rejected promise (error reporting).
asyncErrorHandler: () => {},
// Tests if the error will trigger a retry of the action or will fail directly (retry promises)
retryOnErrorFunction: (error: Error) => error instanceof TypeError,
// Notification function on successful asynchronous action (if success message is set on action)
notifySuccess: undefined,
// Notification function on failed asynchronous action (if error message is set on action)
notifyError: undefined,
// Notification function injected in side effects action context (success and error) to notify warning
notifyWarning: undefined,
// Function called to return common error message if error message is not provided or error not managed in action. To override locally in action, provide a function that returns _null_.
getErrorMessage: undefined,
});
Trace actions
To easily trace effects, use logEnabled option, that will automatically add a detailed logger.
To trace effects, add a logger development middleware to the store.
These middlewares are deactivated if the NODE_ENV
during the build is not development
or test
.
Concise logger
import { logEffects } from 'state-decorator/development';
function Container(props: Props) {
const { state, actions } = useLocalStore({ getInitialState, actions, middlewares: [logEffects()] });
return <div />;
}
By default the console.log
function is used but you can pass any function that has the same signature e.g:
function logger(...args: any[]) {
// implementation...
}
const logMiddleware = logEffects(logger);
const store = createStore({ getInitialState, actions, middlewares: [logMiddleware] });
Example:
[StateDecorator sample] loadList effects {list: Array(10), prop1: "", prop2: "", prop3: ""} loading: false
[StateDecorator sample] setProp1 effects {list: Array(10), prop1: "92", prop2: "", prop3: ""}
Detailed Logger
import { logDetailedEffects } from 'state-decorator/development';
function Container(props: Props) {
const { state, actions } = useLocalStore({ getInitialState, actions, middlewares: [logDetailedEffects()] });
return <div />;
}
Like concise logger, by default the console
is used but you can pass any object implementing log, group, groupCollapsed, groupEnd
functions.
Example:
onCalendarTimeRangeChange
▼ Arguments
║ 0 : 2018-08-12T22:00:00.000Z
║ 1 : 2018-08-19T22:00:00.000Z
▶ Before
▶ After
▼ Diff
║ calendarStartDate : null => 2018-08-12T22:00:00.000Z
║ calendarEndDate : null => 2018-08-19T22:00:00.000Z
Note: In v7, it can be activated by adding logEnabled: true
in the configuration.
Redux devtools
You can connect any StateDecorator store to Chrome Redux devtools extension by adding the devtools
middleware to the store.
import { devtools } from 'state-decorator/development';
function Container(props: Props) {
const { state, actions } = useLocalStore({ getInitialState, actions, middlewares: [devtools()] });
return <div />;
}
Unit testing
- The state decorator provides a testing framework for stores.
- The base principle is that a mock store is immutable:
- a new store is created each time state or props are set.
- it allows to share mock store across tests.
- The API is chainable: functions are returning a mock store, excepting test functions.
- On a mock store, we can test the store internals after initialiation, after inbound props have changed, after actions are called. These test function do not change the internal state of the mock store.
Getting started
- Create and setup a mock store (createMockFromStore, createMockStore)
- Test store after initialization (onInit)
- Test store after props changes (onPropsChange)
- Test each action:
- Get mock store action (getAction)
- setup store state/prop (setPartialState, setPartialProps)
- if action is asynchronous, you can mock promise (promiseResolves, promiseRejects)
- call action with params (call)
- await response to test state/actions called...
Example
The store definition is available at TodoStore.ts.
The test is available at Todo.test.ts.
import { createMockFromStore } from 'state-decorator/test';
import todoStore from './fixtures/TodoStore';
describe('Todo', () => {
// create a mock store and setup initial state
// this store can be shared accross tests because it is immutable
const store = createMockFromStore(todoStore)
// or
// const store = createMockStore(storeConfig, props)
// initial state / props can be set in full form or partial form
.setPartialState({
idCount: 3,
todoIds: ['id', 'id2'],
todoMap: {
id: {
id: 'id',
completed: true,
title: 'todo',
},
id2: {
id: 'id2',
completed: false,
title: 'todo 2',
},
},
});
describe('Initialization', () => {
it('if not initial todo => load list', () => {
store
.setPartialState({
todoIds: [],
todoMap: {},
})
.onInit({ initialTodos: null })
.test(({ state, actions }) => {
// test state after initialization, ie:
// - options.onPropsChange with onMount: true
// - options.onMount
expect(state.todoIds).toEqual([]);
expect(state.todoMap).toEqual({});
// test derived state
expect(state.todos).toEqual([]);
// test side effects also
expect(actions.loadRemoteList).toHaveBeenCalled();
});
});
it('initial todos => setup state and do not load list', () => {
store
.setPartialState({
todoIds: [],
todoMap: {},
})
.onInit({
initialTodos: [
{
id: 'item1',
title: 'Item 1',
completed: false,
},
],
})
.test(({ state, actions }) => {
// test state after initialization, ie:
// - options.onPropsChange with onMount: true
// - options.onMount
expect(state.todoIds).toEqual(['item1']);
expect(state.todoMap).toEqual({
item1: {
id: 'item1',
title: 'Item 1',
completed: false,
},
});
// test derived state
expect(state.todos).toEqual([
{
id: 'item1',
title: 'Item 1',
completed: false,
},
]);
// test side effects also
expect(actions.loadRemoteList).not.toHaveBeenCalled();
});
});
});
// Test effects and side effect when props are changing
//
// Here:
// export const todoOptions: StoreOptions<State, Actions, Props, DerivedState> = {
// onPropsChange: [
// {
// getDeps: (p) => [p.initialTodos],
// effects: ({ s, p }) => ({ ...s, ...splitList(p.initialTodos) }),
// onMount: true,
// },
// ],
// };
it('onPropsChange', () => {
store
// onPropsChange allows to test prop change, no side effect on original store
// to create a new store with new props, use setProps or setPartialProps
.onPropsChange({
initialTodos: [
{
id: 'item1',
title: 'Item 1',
completed: false,
},
],
})
.test(({ state, actions }) => {
// test state after prop has changed
expect(state.todoIds).toEqual(['item1']);
expect(state.todoMap).toEqual({
item1: {
id: 'item1',
title: 'Item 1',
completed: false,
},
});
// test derived state after prop has changed
expect(state.todos).toEqual([
{
id: 'item1',
title: 'Item 1',
completed: false,
},
]);
// test side effects also
expect(actions.loadRemoteList).not.toHaveBeenCalled();
});
});
describe('actions', () => {
it('onSetNewTitle', async () => {
// always return a promise whatever if the action is asynchronous or not
const { state, prevState } = await store.getAction('onSetNewTitle').call('new Title');
// previous state is available for comparison
expect(prevState.newTitle).toEqual('');
expect(state.newTitle).toEqual('new Title');
});
it('onCreate', async () => {
const {
state: { todoIds, todoMap, todos },
} = await store
// set partial state and return new store, source store is untouched
.setPartialState({
newTitle: 'new todo',
})
// get a mock action that can be shared too !
.getAction('onCreate')
// Invokes action. It has no side effect on state of mock action
.call();
// assert state
expect(todoIds).toEqual(['id', 'id2', 'id3']);
expect(todoMap).toEqual({
id: {
id: 'id',
completed: true,
title: 'todo',
},
id2: {
id: 'id2',
completed: false,
title: 'todo 2',
},
id3: {
id: 'id3',
completed: false,
title: 'new todo',
},
});
// assert derived state
//
// export const todoOptions: StoreOptions<State, Actions, Props, DerivedState> = {
// derivedState: {
// todos: {
// getDeps: ({ s }) => [s.todoIds, s.todoMap],
// get: ({ s }) => s.todoIds.map((id) => s.todoMap[id]),
// },
// },
// };
expect(todos).toEqual([
{
id: 'id',
completed: true,
title: 'todo',
},
{
id: 'id2',
completed: false,
title: 'todo 2',
},
{
id: 'id3',
completed: false,
title: 'new todo',
},
]);
});
describe('onToggle', () => {
// this mock action is shared in two tests but they can run in parallel
const onToggle = store.getAction('onToggle');
it('toggle false => true', async () => {
// arguments of call() depend on the Store action signature, use auto completion!
const {
state: { todoIds, todoMap },
actions,
} = await onToggle.call('id2');
expect(todoIds).toEqual(['id', 'id2']);
expect(todoMap).toEqual({
id: {
id: 'id',
completed: true,
title: 'todo',
},
id2: {
id: 'id2',
completed: true,
title: 'todo 2',
},
});
// if there are some side effects calling store actions, you can test them
// as mock actions are injected instead of real actions.
expect(actions.updateRemoteList).toHaveBeenCalled();
});
it('toggle true => false', async () => {
const {
state: { todoIds, todoMap },
actions,
} = await onToggle.call('id');
expect(todoIds).toEqual(['id', 'id2']);
expect(todoMap).toEqual({
id: {
id: 'id',
completed: false,
title: 'todo',
},
id2: {
id: 'id2',
completed: false,
title: 'todo 2',
},
});
// if there are some side effects calling store actions, you can test them
// as mock actions are injected instead of real actions.
expect(actions.updateRemoteList).toHaveBeenCalled();
});
});
// Another store / action
describe('loadRemoteList', () => {
// action is shared
const action = store.getAction('loadRemoteList');
it('success', async () => {
const { state } = await action
// override default promise result
.promiseResolves([
{
id: 'item1',
title: 'item title',
completed: true,
},
])
.call();
expect(state.error).toBeFalsy();
expect(state.todoIds).toEqual(['item1']);
expect(state.todoMap).toEqual({
item1: {
id: 'item1',
title: 'item title',
completed: true,
},
});
});
it('error', async () => {
const { state } = await action
// override default promise result with error
.promiseRejects(new Error('boom'))
.call();
expect(state.error).toBeTruthy();
});
});
});
});
Error testing
Managed error
If an error is triggered during the action and this error was managed (in errorEffect, errorSideEffect or getErrorMessage) then the action promise will be resolved.
// asyncManagedError: {
// getPromise: () => Promise.reject(new Error('my error')),
// errorEffects: ({ s }) => ({ ...s, error: true })
// },
it('asyncManagedError works as expected', async () => {
mockStore
.getAction('asyncManagedError')
.call()
.then(({ state }) => {
// check that error was correcly managed
expect(state.error).toEqual(true);
});
});
Managed error and rejectPromiseOnError=true
If an error is triggered during the action and this error was managed (in errorEffect, errorSideEffect or getErrorMessage) but the action has the rejectPromiseOnError flag set then the action promise will be rejected and the sourceError allows to inspect the action.
import { ActionError } from 'state-decorator/test';
// asyncManagedError: {
// rejectPromiseOnError: true,
// getPromise: () => Promise.reject(new MyError()),
// errorEffects: ({ s }) => ({ ...s, error: true })
// },
it('asyncManagedErrorThrow works as expected', async () => {
return mockStore
.getAction('asyncManagedErrorThrow')
.call()
.catch((e: ActionError<State>) => {
// test source error
if (e.sourceError instanceof MyError) {
// check that error was correcly managed
expect(e.state.error).toEqual(true);
} else {
return Promise.reject();
}
});
});
Unexpected error
The action implementation can contain a mistake that triggers an error.
For example:
// asyncThatCrashes: {
// getPromise: () => Promise.resolve(),
// effects: ({ s }) => {
// const nullObj = null;
// nullObj.crash = 'test';
// return s;
// },
// },
it('asyncThatCrashes works as expected', async () => {
return mockStore.getAction('asyncThatCrashes').call();
});
This test will crash but the source of the error is not clear:
Cannot set property 'crash' of null
438 | .catch((e: Error) => {
439 | return Promise.reject(
> 440 | new ActionError(
| ^
441 | e,
442 | getState(newStateRef),
443 | propsRef.current,
at src/test.ts:440:13
Just return the sourceError to get the real stack trace.
it.only('asyncThatCrashes works as expected', async () => {
return mockStore
.getAction('asyncThatCrashes')
.call()
.catch((e) => Promise.reject(e.sourceError));
});
will display
TypeError: Cannot set property 'crash' of null
72 | effects: ({ s }) => {
73 | const nullObj = null;
> 74 | nullObj.crash = 'test';
| ^
75 | return s;
76 | },
77 | },
at Object.effects (tests/Tests.test.ts:74:9)
at processPromiseSuccess (src/impl.ts:687:23)
at src/impl.ts:889:51
API
Create mock store
| Name | Parameters | | | ------------------- | ------------------------------------------------ | --------------------------------- | | createMockStore | (config: StoreConfig, props: Props) => MockStore | Create a mock store from a config | | createMockFromStore | (store: StoreAPI) => MockStore | Create a mock store from a store |
MockStore
| Function | Arguments | Returns | Description | | --------------- | ------------------------------------ | ------------------- | ------------------------------------------------------------------------------------------- | | getAction | actionName: keyof A | MockStoreAction | Get a mock action instance | | onInit | props?: Partial | MockResult | Test store initialization (onPropsChanged marked onMount + onMount) | | onPropsChange | props: P, init?: boolean | MockResult | Test prop changes. Does not change internal props. Init allows to test onMount prop changes | | setPartialProps | props: Partial | new mock store | Create a new store with the merge of current props and partial props | | setPartialState | state: Partial | new mock store | Create a new store with the merge of current state and partial state | | setProps | props: P | new mock store | Create a new store with the specified props | | setState | state: S | new mock store | Create a new store with the specified state | | test | f: (state: S & DS, props: P) => void | same store instance | Call specified fuction to test current state and store |
MockStoreAction
| Function | Parameters | Returns | Description | | --------------- | ------------------------------------ | ------------------- | ----------------------------------------------------------------------------------- | | setState | state: S | new mock action | Create a new store with the specified state | | setPartialState | state: Partial | new mock action | Create a new store with the merge of current state and partial state | | setProps | props: P | new mock action | Create a new store with the specified props | | setPartialProps | props: Partial | new mock action | Create a new store with the merge of current props and partial props | | test | f: (state: S & DS, props: P) => void | same mock action | Call specified fuction to test current state and store | | promiseResolves | res: PromiseResult | new mock action | Returns a new mock action in which promise will return specified result | | promiseRejects | err: Error | new mock action | Returns a new mock action in which promise will return specified error | | call | ...args: Parameters | Promise | Call action and returns the results. Has no side effect on internal state and props |
MockResult
| Attribute | Description | | --------- | --------------------------------- | | state | State after action | | props | Props after action | | actions | Mock actions to test side effects |
Immutability
- Immutability: each effects must return a new state instance (or
null
if there's no change). - I recommend using Immer to manage complex cases (deep nesting).
Build
This library provides CJS and ES files.
Webpack users should add this alias rule to prevent importing both (especially when importing another file than index.js
):
resolve: {
alias: {
'state-decorator': 'state-decorator/es',
},
},
Limitations
Due to bundle size constraints and used in specific use cases only, a basic function is used to clone state / props / arguments in conflicting actions and optimistic effects use cases. In some edge cases (like clone moment objects, class instances), you must provide a clone implementation like Lodash cloneDeep implementation.
"Index signature is missing in type" or "'XxxXxx' does not satisfy the constraint 'DecoratedActions'" error during TS compilation:
- https://github.com/Microsoft/TypeScript/issues/15300
- Solution: the Actions interface must either extends DecoratedActions interface or be a type.
Migration
V6 to V7
Step 1: V6 compatibility layer
Only useLocalStore is supported by compatiblity layer.
-import useLocalStore from 'state-decorator';
+import useLocalStore from 'state-decorator/v6';
Step 2: Migrate code
- Merge getInitialState, actions, options in a single object.
- Remove state spread in effects
- const getInitialState: (p:Props) => ({myProp: ''});
-
- const actions: StoreActions<State, Actions, Props> = {
- myAction: ({s, args: [v]}) => ({ ...s, myProp: v })
-}
-
- const options: StoreOptions<State, Action, Props> = {
- onMount: () => {},
- onPropsChange: {
- getDeps: () => [],
- effects: ({s, p}) => ({ ...s, myProp: p.prop }),
- }
- };
-
- function MyComponent(p:Props) {
- const {state, actions} = useLocalStore(getInitialState, actions, props, options, [logEffects()]);
- return <div>{myProp}</div>;
- }
+ const config:StoreConfig<State, Actions, Props> = {
+ getInitialState: () => ({ myProp: '' }),
+ actions: {
+ myAction: ({args: [v]}) => ({ myProp: v})
+ // or
+ myAction: setArgIn('myProp'),
+ },
+ onMount: () => {},
+ onPropsChange: {
+ getDeps: () => [],
+ effects: ({ p }) => ({ myProp: p.prop })
+ },
+ middlewares: [logEffects()],
+ }
+
+ function MyComponent(p:Props) {
+ const {state, actions} = useLocalStore(config);
+ return <div>{myProp}</div>;
+ }
V5 to V6
Step 1: v5 compatibility layer
To use the new engine while still using v5 API, juste import the v5 compatibility layer. You can new store in new code.
-import useStateDecorator, { StateDecoratorActions } from 'state-decorator';
+import useStateDecorator, { StateDecoratorActions } from 'state-decorator/v5';
If you are using optimistic effects, make sure to set the optimisticActions middleware as a default middleware, see Optimistic actions.
import { setGlobalConfig } from 'state-decorator';
import { optimisticActions } from 'state-decorator/middlewares';
setGlobalConfig({
defaultMiddlewares: [optimisticActions()],
});
Step 2: Migrate code but keep v5 tests
Code
Types
| Before | After | | --------------------- | ------------ | | StateDecoratorActions | StoreActions | | StateDecoratorOptions | StoreOptions |
Optimistic actions
If you are using optimistic effects, make sure to set the optimisticActions middleware to your stores, see Optimistic actions.
Hooks
Choose one of useLocalStore, useStore, useStoreSlice.
Actions
- rename action properties
| Before | After | | ----------------- | ----------------- | | action | effects | | onActionDone | sideEffects | | preReducer | preEffects | | optimisticReducer | optimisticEffects | | reducer | effects | | errorReducer | errorEffects | | onDone | sideEffects | | onFail | errorSideEffects | | promise | getPromise | | promiseGet | getGetPromise | | successMessage | getSuccessMessage | | errorMessage | getErrorMessage |
- replace argumtents by action context
- const actions: StateDecoratorActions<S, A, P> = {
+ const actions: StoreActions<S, A, P> = {
loadList: {
- promise: ([id], s, p, a) => {
+ getPromise: ({ s, args: [id], p, a }) => {
/* ... */
},
- reducer: (s, res) => ({
+ effects: ({ s, res }) => ({
/* ... */
}),
- errorReducer: (s, err) => ({
+ errorEffects: ({ s, err }) => ({
/* ... */
}),
},
};
Options
onMount
- const options: StateDecoratorOptions<S, A, P> = {
- onMount: (actions, props, state) => { /* initial side effects */ }
- };
+ const options: StoreOptions<S, A, P> = {
+ onMount: ({state, props, actions}) => { /* initial side effects */ }
+} ;
logEnabled
- function Container(props: Props) {
- const { state, actions } = useStateDecorator(getInitialState, actionsImpl, props, { logEnabled: true });
- return <div />;
- }
+ import { logDetailedEffects } from 'state-decorator/middlewares';
+
+ function Container(props: Props) {
+ const { state, actions } = u