enable time-travel in your apps. undo/redo middleware for zustand. built with zustand. <1 kB
Try a live demo
npm i zustand zundo
zustand v4.3.0 or higher is required for TS usage. v4.0.0 or higher is required for JS usage. Node 16 or higher is required.
- Solves the issue of managing state in complex user applications
- "It Just Works" mentality
- Small and fast
- Provides simple middleware to add undo/redo capabilities
- Leverages zustand for state management
- Works with multiple stores in the same app
- Has an unopinionated and extensible API
This returns the familiar store accessible by a hook! But now your store tracks past actions.
import { temporal } from 'zundo';
import { create } from 'zustand';
// define the store (typescript)
interface StoreState {
bears: number;
increasePopulation: () => void;
removeAllBears: () => void;
}
// creates a store with undo/redo capability
const useStoreWithUndo = create<StoreState>()(
temporal((set) => ({
bears: 0,
increasePopulation: () => set((state) => ({ bears: state.bears + 1 })),
removeAllBears: () => set({ bears: 0 }),
})),
);
If you're using React, you can convert the store to a React hook using create from zustand
.
import { useStore } from 'zustand';
import type { TemporalState } from 'zundo';
const useTemporalStore = <T,>(
selector: (state: TemporalState<StoreState>) => T,
equality?: (a: T, b: T) => boolean,
) => useStore(originalStore.temporal, selector, equality);
Use your store anywhere, including undo
, redo
, and clear
!
const App = () => {
const { bears, increasePopulation, removeAllBears } = useStoreWithUndo();
const { undo, redo, clear } = useTemporalStore((state) => state);
// or if you don't use create from zustand, you can use the store directly.
// } = useStoreWithUndo.temporal.getState();
// if you want reactivity, you'll need to subscribe to the temporal store.
return (
<>
bears: {bears}
<button onClick={() => increasePopulation}>increase</button>
<button onClick={() => removeAllBears}>remove</button>
<button onClick={() => undo()}>undo</button>
<button onClick={() => redo()}>redo</button>
<button onClick={() => clear()}>clear</button>
</>
);
};
(config: StateCreator, options?: ZundoOptions) => StateCreator
zundo has one export: temporal
. It is used to as middleware for create
from zustand. The config
parameter is your store created by zustand. The second options
param is optional and has the following API.
type onSave<TState> =
| ((pastState: TState, currentState: TState) => void)
| undefined;
export interface ZundoOptions<TState, PartialTState = TState> {
partialize?: (state: TState) => PartialTState;
limit?: number;
equality?: (currentState: TState, pastState: TState) => boolean;
onSave?: onSave<TState>;
handleSet?: (
handleSet: StoreApi<TState>['setState'],
) => StoreApi<TState>['setState'];
pastStates?: Partial<PartialTState>[];
futureStates?: Partial<PartialTState>[];
wrapTemporal?: (
storeInitializer: StateCreator<
_TemporalState<TState>,
[StoreMutatorIdentifier, unknown][],
[]
>,
) => StateCreator<
_TemporalState<TState>,
[StoreMutatorIdentifier, unknown][],
[StoreMutatorIdentifier, unknown][]
>;
}
partialize?: (state: TState) => PartialTState
Use the partialize
option to omit or include specific fields. Pass a callback that returns the desired fields. This can also be used to exclude fields. By default, the entire state object is tracked.
// Only field1 and field2 will be tracked
const useStoreA = create<StoreState>(
temporal(
set => ({ ... }),
{ partialize: (state) => {
const { field1, field2, ...rest } = state
return { field1, field2 }
}}
)
)
// Everything besides field1 and field2 will be tracked
const useStoreB = create<StoreState>(
temporal(
set => ({ ... }),
{ partialize: (state) => {
const { field1, field2, ...rest } = state
return rest;
}}
)
)
limit?: number
For performance reasons, you may want to limit the number of previous and future states stored in history. Setting limit
will limit the number of previous and future states stored in the temporal
store. When the limit is reached, the oldest state is dropped. By default, no limit is set.
const useStore = create<StoreState>(
temporal(
set => ({ ... }),
{ limit: 100 }
)
);
equality?: (currentState: TState, pastState: TState) => boolean
For performance reasons, you may want to use a custom equality
function to determine when a state change should be tracked. You can write your own or use something like lodash/deepEqual
or zustand/shallow
. By default, all state changes to your store are tracked.
import { shallow } from 'zustand/shallow'
// Use an existing equality function
const useStoreA = create<StoreState>(
temporal(
set => ({ ... }),
{ equality: shallow }
)
);
// Write your own equality function
const useStoreB = create<StoreState>(
temporal(
set => ({ ... }),
{ equality: (a, b) => a.field1 !== b.field1 }
)
);
onSave?: (pastState: TState, currentState: TState) => void
Sometimes, you may need to call a function when the temporal store is updated. This can be configured using onSave
in the options, or by programmatically setting the callback if you need lexical context (see the TemporalState
API below for more information).
import { shallow } from 'zustand/shallow'
const useStoreA = create<StoreState>(
temporal(
set => ({ ... }),
{ onSave: (state) => console.log('saved', state) }
)
);
handleSet?: (handleSet: StoreApi<TState>['setState']) => StoreApi<TState>['setState']
Sometimes multiple state changes might happen in a short amount of time and you only want to store one change in history. To do so, we can utilize the handleSet
callback to set a timeout to prevent new changes from being stored in history. This can be used with something like lodash.throttle
or debounce
. This a way to provide middleware to the temporal store's setter function.
const withTemporal = temporal<MyState>(
(set) => ({ ... }),
{
handleSet: (handleSet) =>
throttle<typeof handleSet>((state) => {
console.info('handleSet called');
handleSet(state);
}, 1000),
},
);
pastStates?: Partial<PartialTState>[]
futureStates?: Partial<PartialTState>[]
You can initialize the temporal store with past and future states. This is useful when you want to load a previous state from a database or initialize the store with a default state. By default, the temporal store is initialized with an empty array of past and future states.
Note: The
pastStates
andfutureStates
do not respect the limit set in the options. If you want to limit the number of past and future states, you must do so manually prior to initializing the store.
const withTemporal = temporal<MyState>(
(set) => ({ ... }),
{
pastStates: [{ field1: 'value1' }, { field1: 'value2' }],
futureStates: [{ field1: 'value3' }, { field1: 'value4' }],
},
);
wrapTemporal?: (storeInitializer: StateCreator<_TemporalState<TState>, [StoreMutatorIdentifier, unknown][], []>) => StateCreator<_TemporalState<TState>, [StoreMutatorIdentifier, unknown][], [StoreMutatorIdentifier, unknown][]>
You can wrap the temporal store with your own middleware. This is useful if you want to add additional functionality to the temporal store. For example, you can add persist
middleware to the temporal store to persist the past and future states to local storage.
Note: The
temporal
middleware can be added to thetemporal
store. This way, you could track the history of the history. π€―
import { persist } from 'zustand/middleware'
const withTemporal = temporal<MyState>(
(set) => ({ ... }),
{
wrapTemporal: (storeInitializer) => persist(storeInitializer, { name: 'temporal-persist' }),
},
);
When using zustand with the temporal
middleware, a temporal
object is attached to your vanilla or React-based store. temporal
is a vanilla zustand store: see StoreApi from zustand for more details.
Use temporal.getState()
to access to temporal store!
While
setState
,subscribe
, anddestory
exist ontemporal
, you should not need to use them.
To use within React hooks, we need to convert the vanilla store to a React-based store using create
from zustand
. This is done by passing the vanilla store to create
from zustand
.
import { create } from 'zustand';
import { temporal } from 'zundo';
const useStore = create(
temporal(
set => ({ ... }),
{ ... }
)
);
const useTemporalStore = create(useStore.temporal);
temporal.getState()
returns the TemporalState
which contains undo
, redo
, and other helpful functions and fields.
interface TemporalState<TState> {
pastStates: TState[];
futureStates: TState[];
undo: (steps?: number) => void;
redo: (steps?: number) => void;
clear: () => void;
isTracking: boolean;
pause: () => void;
resume: () => void;
setOnSave: (onSave: onSave<TState>) => void;
}
pastStates: TState[]
pastStates
is an array of previous states. The most recent previous state is at the end of the array. This is the state that will be applied when undo
is called.
futureStates: TState[]
futureStates
is an array of future states. States are added when undo
is called. The most recent future state is at the end of the array. This is the state that will be applied when redo
is called. The future states are the "past past states."
undo: (steps?: number) => void
undo
: call function to apply previous state (if there are previous states). Optionally pass a number of steps to undo to go back multiple state at once.
redo: (steps?: number) => void
redo
: call function to apply future state (if there are future states). Future states are "previous previous states." Optionally pass a number of steps to redo go forward multiple states at once.
clear: () => void
clear
: call function to remove all stored states from your undo store. Sets pastStates
and futureStates
to arrays with length of 0. Warning: clearing cannot be undone.
Dispatching a new state will clear all of the future states.
isTracking: boolean
isTracking
: a stateful flag in the temporal
store that indicates whether the temporal
store is tracking state changes or not. Possible values are true
or false
. To programmatically pause and resume tracking, use pause()
and resume()
explained below.
pause: () => void
pause
: call function to pause tracking state changes. This will prevent new states from being stored in history within the temporal store. Sets isTracking
to false
.
resume: () => void
resume
: call function to resume tracking state changes. This will allow new states to be stored in history within the temporal store. Sets isTracking
to true
.
setOnSave: (onSave: (pastState: State, currentState: State) => void) => void
setOnSave
: call function to set a callback that will be called when the temporal store is updated. This can be used to call the temporal store setter using values from the lexical context. This is useful when needing to throttle or debounce updates to the temporal store.
Click to expand
This is a work in progress. Submit a PR!
- create nicer API, or a helper hook in react land (useTemporal). or vanilla version of the it
- support history branches rather than clearing the future states
- store state delta rather than full object
- track state for multiple stores at once
PRs are welcome! pnpm is used as a package manager. Run pnpm install
to install local dependencies.
Charles Kornoelje (@_charkour)
View the releases for the change log.
Ivo IliΔ (@theivoson)