Immutability Helpers
Redux is a tiny pattern that represents states as immutable objects. Redux was originally designed for React. Most Redux concepts, such as pure functions, are centered around the React ecosystem. Nowadays Redux is not directly related to React.
The cornerstone of Redux is immutability. Immutability is an amazing pattern to minimise unpredictable behaviour in our code. We're not going to cover functional programming in this article. However we're going to look at very useful packages that are called "immutability helpers".
Most developers have to deal with, so called, "deep objects" and most important follow the immutability concept, when it comes to changing the value of some deeply nested property. Given the following code:
export interface Task {
title: string;
dates: {
startDate: string;
dueDate: string;
};
}
export interface TrelloStateModel {
tasks: {
[taskId: string]: Task;
};
}
@State<TrelloStateModel>({
name: 'trello',
defaults: {
tasks: {}
}
})
@Injectable()
export class TrelloState {}
Let's imagine that we're faced with the task of changing the
dueDate
property:export class UpdateDueDate {
static readonly type = '[Trello] Update due date';
constructor(public taskId: string, public dueDate: string) {}
}
Let's see how we would implement the
updateDueDate
action handler:export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
ctx.setState(state => ({
tasks: {
...state.tasks,
[action.taskId]: {
...state.tasks[action.taskId],
dates: {
...state.tasks[action.taskId].dates,
dueDate: action.dueDate
}
}
}
}));
}
}
This code will work but unfortunately it is complicated to maintain and understand. It's not self-descriptive and will be daunting for new developers.
There are different ways to improve this code. Let us look at a few different packages that can help in this regard.
State operators are first-class immutability helpers that NGXS provides out of the box. The
patch
operator will become your best friend in case of choosing state operators as your immutability helpers. Let's see how we could re-write the above code with the help of the patch
state operator:import { patch } from '@ngxs/store/operators';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
ctx.setState(
patch({
tasks: patch({
[action.taskId]: patch({
dates: patch({
dueDate: action.dueDate
})
})
})
})
);
}
}
immer
is a very popular library that allows you to make changes to immutable objects as if they were mutable. The below code shows how to write the same code with the help of Immer:import { produce } from 'immer';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
const state = produce(ctx.getState(), draft => {
draft.tasks[action.taskId].dates.dueDate = action.dueDate;
});
ctx.setState(state);
}
}
Immer's
produce
function can be also used as a state operator:import { produce } from 'immer';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
ctx.setState(
produce(draft => {
draft.tasks[action.taskId].dates.dueDate = action.dueDate;
})
);
}
}
You may notice how much less code this is and how much better it looks. From the
immer
repository:Using Immer is like having a personal assistant; he takes a letter (the current state) and gives you a copy (draft) to jot changes onto. Once you are done, the assistant will take your draft and produce the real immutable, final letter for you (the next state).
immutability-helper
is a small package that lets you mutate a copy of data without changing the original source:import update from 'immutability-helper';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
const state = update(ctx.getState(), {
tasks: {
[action.taskId]: {
dates: {
dueDate: {
$set: action.dueDate
}
}
}
}
});
ctx.setState(state);
}
}
object-path-immutable
is a small library that allows you to modify deep object properties without modifying the original object. Let's look at how we could write the same code using this library:import immutable from 'object-path-immutable';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
const state = immutable.set(
ctx.getState(),
`tasks.${action.taskId}.dates.dueDate`,
action.dueDate
);
ctx.setState(state);
}
}
immutable-assign
is a lightweight library that pursues the same goal. Its syntax is similar to immer
's:import * as iassign from 'immutable-assign';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
const state = iassign(ctx.getState(), state => {
state.tasks[action.taskId].dates.dueDate = action.dueDate;
return state;
});
ctx.setState(state);
}
}
Ramda is a great library for functional programming and it is used in a large number of projects. This example might be useful for people who use both Ramda and NGXS in their projects:
import * as R from 'ramda';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
const property = R.lensPath(['tasks', action.taskId, 'dates', 'dueDate']);
const state = R.set(property, action.dueDate, ctx.getState());
ctx.setState(state);
}
}
icepick
is a zero-dependency library for working with immutable collections. Given the following re-written code:import * as icepick from 'icepick';
export class TrelloState {
@Action(UpdateDueDate)
updateDueDate(ctx: StateContext<TrelloStateModel>, action: UpdateDueDate) {
const state = icepick.setIn(
ctx.getState(),
['tasks', action.taskId, 'dates', 'dueDate'],
action.dueDate
);
ctx.setState(state);
}
}
We have looked at several different libraries that might be helpful in accompanying the concept of immutability. Choose the right one for your needs.
Last modified 1yr ago