ts-loupe

Strongly typed and lightweight lens library.

Usage no npm install needed!

<script type="module">
  import tsLoupe from 'https://cdn.skypack.dev/ts-loupe';
</script>

README

TS Loupe

Build Status License Library minified size Library minified + gzipped size

About

Lenses is a pattern used to read and update properties within an object.

Installation

This library is published in the NPM registry and can be installed using any compatible package manager.

npm install ts-loupe --save

# For Yarn, use the command below.
yarn add ts-loupe

Installation from CDN

This module has an UMD bundle available through JSDelivr and Unpkg CDNs.

<!-- For UNPKG use the code below. -->
<script src="https://unpkg.com/ts-loupe"></script>

<!-- For JSDelivr use the code below. -->
<script src="https://cdn.jsdelivr.net/npm/ts-loupe"></script>

API

Lens

Is a pair of two functions that abstracts the way of we access the requested field. In other words is a focus into some data structure.

Signature:

type Getter<A, B> = (data: A) => B;

type Setter<A, B> = (value: B) => (data: A) => A;

interface Lens<A, B> {
  get: Getter<A, B>;
  set: Setter<A, B>;
}

Example:

type User = { name: string; age: number };

const user: User = { name: 'Jerry Lee', age: 18 };

const getName = (user: User) => user.name;
const setName = (name: User['name']) => (data: User) => ({ ...data, name });

const nameLens = lens(getName, setName);

nameLens.get(user); // returns: "Jerry Lee".
nameLens.set('Leon Lan')(user); // returns: A new user with Leon Lan as the name.

View

Returns the data structure pointed to by the lens getter function.

Signature:

type LensView = <O, V>(lens: Pick<Lens<O, V>, 'get'>) => (data: O) => V;

Example:

type User = { name: string };

const user: User = { name: 'Len Lon' };

const getName = (user: User) => user.name;
const setName = (name: User['name']) => (data: User) => ({ ...data, name });

const nameLens = lens(getName, setName);

view(nameLens)(user); //returns: "Len Lon".

Set

Set is used to 'set' a value into the data structure pointed to by the lens setter function.

Signature:

type LensSet = <O, V>(
  lens: Pick<Lens<O, V>, 'set'>
) => (value: V) => (data: O) => O;

Example:

type User = { name: string };

const user: User = { name: 'Jackie Chan' };

const getName = (user: User) => user.name;
const setName = (name: User['name']) => (data: User) => ({ ...data, name });
const nameLens = lens(getName, setName);

lensSet(nameLens)('John Wick')(user); // returns: A new user with "John Wick" as the name.

Over

Applies the function to the given lens property and returns the result.

Signature:

type LensOver = <O, V>(
  lens: Lens<O, V>
) => (fn: (value: V) => V) => (data: O) => O;

Example:

type User = { name: string };

const user: User = { name: 'Santino' };
const nameLens = prop<User>('name');

over(nameLens)(name => `${name} D'Antonio`)(user); // returns: A new user with "Santino D'Antonio" as the name.

Prop

Creates a lens focused on a given property.

Signature:

type LensProp = <O, K extends keyof O = keyof O>(key: K) => Lens<O, O[K]>;

Example:

type User = { age: string; name: string };
lensProp<User>('age'); // returns: A lens instance focused on the age propertie.

Compose

Compose two lenses Lens<A, B> and Lens<B, C>, to produce a new lens Lens<A, C>.

Signature:

interface LensCompose {
  <A, B, C>(...lenses: [Lens<A, B>, Lens<B, C>]): Lens<A, C>;
}

Example:

type User = {
  pocket: {
    money: number;
  };
};

const user: User = {
  pocket: {
    money: 3213
  }
};

const pocketLens = lensProp<User>('pocket');
const moneyLens = lensProp<User['pocket']>('money');

compose(
  pocketLens,
  moneyLens
); // returns: A new lens focused on money propertie.

License

Released under MIT License.