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".
The Problem
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:
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
immutability-helper is a small package that lets you mutate a copy of data without changing the original source:
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
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
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
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);
}
}
Summary
We have looked at several different libraries that might be helpful in accompanying the concept of immutability. Choose the right one for your needs.
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: