Skip to content

Lenses for TypeScript / JavaScript (inspired by Scala Lenses)

License

Notifications You must be signed in to change notification settings

wajda/ts-lenses

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

12 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

A simple utility library that implements functional Lenses - composable setter and getter pairs that focuses on particular nested properties of an object.


Usage

// Import lenses library
import * as lens from 'lenses'

// Define your lens
const myLens = lens.of("a.b.c")

// Use the lens to get or set a nested value
myLens.get( {a: {b: {c: 42}}} )     // returns 42
myLens.set( {a: {b: {c: 42}}}, 777) // returns {a: {b: {c: 777}}}

// missing properties are created
myLens.set({}, 777) // still returns {a: {b: {c: 777}}}

set() method is a pure function. It creates shallow copies of objects along the path.

Lens compositions

const fooBarBazLens = lens.composition(lens.of("foo"), lens.of("bar"), lens.of("baz"))  // or...
const fooBarBazLens = lens.of("foo.bar").compose(lens.of("baz"))

// lens compositions are associative:
const fooBarBazLens = lens.of("foo").compose(lens.of("bar.baz"))                        // is the same as...
const fooBarBazLens = lens.of("foo").compose(lens.of("bar").compose(lens.of("baz")))    // is the same as...
const fooBarBazLens = (lens.of("foo").compose(lens.of("bar"))).compose(lens.of("baz"))  // is the same as...
const fooBarBazLens = lens.of("foo").focus("bar").focus("baz")                          // is the same as...
const fooBarBazLens = lens.of("foo").focus("bar.baz")                                   // is the same as...
const fooBarBazLens = lens.of("foo.bar.baz")

Lens projections

Projections are lenses with arity > 1, defined as complex lenses. Projections point to multiple properties of an object at any levels of nesting. Projections can be nested.

const shortContactInfoProjection = 
    lens.projection(
        lens.of("name"),
        lens.of("surname"),
        lens.of("contacts").project(
            lens.of("tel"),
            lens.of("email")))

const personInfo = {
    name: "John",
    surname: "Smith",
    age: 30,
    sex: "male",
    otherPrivateInfo: {...},
    contacts: {
        tel: 111222333444,
        email: "[email protected]",
        address: "Earth"
    },
    otherInfo: {...}
}

shortContactInfoProjection.get(personInfo) // returns ["John", "Smith", 111222333444, "[email protected]"]
shortContactInfoProjection.set(personInfo, "John", "Smith", 111222333444, "[email protected]") // returns updated copy

Todo

  1. Associative projection composition. Currently a Projection can only be a terminal element in a Lens composition chain.

  2. [Optimization] Do not copy an object if the current value is equal to the updating value.

  3. Performance testing

  4. Typing

About

Lenses for TypeScript / JavaScript (inspired by Scala Lenses)

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published