Skip to content

A collection of helper functions to perform immutable operations on plain JavaScript objects and arrays

License

Notifications You must be signed in to change notification settings

nitishkr88/redux-immutable-ops

Repository files navigation

redux-immutable-ops


Build Status codecov.io styled with prettier

A collection of helper functions to perform immutable operations on plain JavaScript objects and arrays. Works on the immutable update patters as defined by Redux.

Article

Installation

npm install --save redux-immutable-ops

yarn add redux-immutable-ops

Features

  • Small footprint
  • Lodash inspired API
  • Conforms to Redux standards
  • Interoperable with JavaScript
  • Structural Sharing
  • Typed with Flow

Example Usage

Updating Nested Objects in a reducer function.

Set state.first.second[someId].fourth from someValue key in action.

Note: state should not be mutated

  • Using spread operator

    function updateVeryNestedField(state, action) {
       return {
           ...state,
           first : {
               ...state.first,
               second : {
                   ...state.first.second,
                   [action.someId] : {
                       ...state.first.second[action.someId],
                       fourth : action.someValue
                   }
               }
           }
       }
    }
  • Using redux-immutable-ops

    import { setIn } from "redux-immutable-ops";
    
    function updateVeryNestedField(state, action) {
      return setIn(state, `first.second[${action.someId}].fourth`, action.someValue)
    }

API

setIn(state: Object | Array<*>, path: string, value: any): Object | Array<*>

Returns an object, with the value at path set to value. path can be a dot-separated list of attribute values to traverse. Index of array value which need to be updated can also be specified in the path. Paths not already existing would be created.

import { setIn } from "redux-immutable-ops"

const state = {
  foo: {
    bar: ['baz', { cat: 42 }]
  }
}

const result1 = setIn(state, 'marvel.best.hero[0]', 'punisher')
expect(result1).not.toBe(state)
expect(result1).toEqual({
  foo: {
    bar: [
      'baz',
      {
        cat: 42
      }
    ]
  },

  marvel: {
    best: {
      hero: ['punisher']
    }
  }
})

const result2 = setIn(state, 'foo.bar[0]', 'cat')
expect(result2).not.toBe(state)
expect(result2).toEqual({
  foo: {
    bar: [
      'cat',
      {
        cat: 42
      }
    ]
  }
})

getIn(state: Object | Array<*>, path: string): any

Returns the value specified by path. path can be a dot-separated list of attribute values to traverse. Index of array value which need to be fetched can also be specified in the path.

import { getIn } from "redux-immutable-ops"

const state = {
  foo: {
    bar: ['baz', { cat: 42 }]
  }
}

expect(getIn(state, 'foo.bar[0]')).toBe('baz')
expect(getIn(state, 'foo.bar[1].cat')).toBe(42)

deleteIn(state: Object | Array<*>, path: string): ?(Object | Array<*>)

Returns a new object, without the key specified in path. path can be a dot-separated list of attribute values to traverse. Index of array value which need to be deleted can also be specified in the path.

import { deleteIn } from "redux-immutable-ops"

const state = ['the', 'quick', 'brown', 'fox']

expect(deleteIn(state, 0)).toEqual(['quick', 'brown', 'fox'])
expect(deleteIn(state, 0)).not.toBe(state)
expect(deleteIn(state, 2)).toEqual(['the', 'quick', 'fox'])
expect(deleteIn(state, 2)).not.toBe(state)

const state = {
  foo: {
    bar: ['baz', { cat: 42 }]
  }
}

const result1 = deleteIn(state, 'foo.bar[0]')
expect(result1).not.toBe(state)

expect(result1).toEqual({
  foo: {
    bar: [
      {
        cat: 42
      }
    ]
  }
})

deleteInRecursive(state: Object | Array<*>, path: string): ?(Object | Array<*>)

Recursively deletes and cleans up the node and the parent(s) as specified by path, and returns a new object. path can be a dot-separated list of attribute values to traverse. Index of array value which need to be deleted can also be specified in the path.

It makes sure that only parents with no remaining children are removed. It also sets the value to undefined if leaf structure is an array.

import { deleteInRecursive } from "redux-immutable-ops"

// should not delete parent if it has other children
expect(
  deleteInRecursive(
    {
      a: {
        b: 1,
        c: 2
      },
      d: {
        e: 3
      }
    },
    'a.b'
  )
).toEqualMap({ // extending expect. implementation in source
  a: {
    c: 2
  },
  d: {
    e: 3
  }
})

// should just set to undefined if leaf structure is an array
expect(
  deleteInRecursive(
    {
      a: [11, 12, 13]
    },
    'a[1]'
  )
).toEqualMap({
  a: [11, undefined, 13]
})

// should delete parent if no other children is present
expect(
  deleteInRecursive(
    {
      a: {
        b: 1,
        c: 2
      },
      d: {
        e: 3
      }
    },
    'd.e'
  )
).toEqualMap({
  a: {
    b: 1,
    c: 2
  }
})

Array specific operations


pop(array: Array<*>): Array<*>

Returns a shallow copy of the array after removing the last element

import { pop } from "redux-immutable-ops"

const arr = ['a', 'b', 'c', 'd']
expect(pop(arr)).toEqual(['a', 'b', 'c'])
expect(arr).toEqual(['a', 'b', 'c', 'd'])

// for empty or undefined array
expect(pop([])).toEqual([])
expect(pop(undefined)).toEqual([])

push(array: Array<*>, value: any): Array<*>

Returns a shallow copy of the array after pushing the new value

import { push } from "redux-immutable-ops"

const arr1 = [{foo: 'bar'}, 2]
const arr2 = push(arr1, 3)
expect(arr2).toEqual([{foo: 'bar'}, 2, 3])
expect(arr2).not.toBe(arr1)
 
// pushing when initial array is undefined
expect(push(undefined, {foo: 'bar'})).toEqual([{'foo': 'bar'}])

shift(array: Array<*>): Array<*>

Returns a shallow copy of the array with the first element removed

import { shift } from "redux-immutable-ops"

const arr1 = ['a', 'b', 'c']
const arr2 = shift(arr1)
expect(arr2).toEqual(['b', 'c'])
expect(arr1).toEqual(['a', 'b', 'c'])

// for empty or undefined array
expect(shift([])).toEqual([])
expect(shift(undefined)).toEqual([])

unshift(array: Array<*>, ...values: any[]): Array<*>

Returns a shallow copy of the array with one or more elements added to the beginning

import { unshift } from "redux-immutable-ops"

expect(unshift(['b', 'c'], 'a')).toEqual(['a', 'b', 'c'])

const arr = ['c', 'd']
expect(unshift(arr, 'a', 'b')).toEqual(['a', 'b', 'c', 'd'])
expect(unshift(arr, 'e', 'f')).toEqual(['e', 'f', 'c', 'd'])
expect(arr).toEqual(['c', 'd'])

// for emtpty or underfined arrays
expect(unshift([], 'a')).toEqual(['a'])
expect(unshift([], 'a', 'b', 'c')).toEqual(['a', 'b', 'c'])
expect(unshift(undefined, 'a')).toEqual(['a'])
expect(unshift(undefined, 'a', 'b', 'c')).toEqual(['a', 'b', 'c'])

splice(array: Array<*>, index: number, removeNum: number, value: any): Array<*>

Similar to Array.splice, but returns a shallow copy.

import { splice } from "redux-immutable-ops"

// insert value at start
expect(splice(['b', 'c', 'd'], 0, 0, 'a')).toEqual(['a', 'b', 'c', 'd'])

// insert value at end
expect(splice(['a', 'b', 'c'], 3, 0, 'd')).toEqual(['a', 'b', 'c', 'd'])

// insert value at an index
expect(splice(['a', 'b', 'd'], 2, 0, 'c')).toEqual(['a', 'b', 'c', 'd'])

// replace value at an index
expect(splice(['a', 'b', 'c', 'd'], 1, 1, 'e')).toEqual(['a', 'e', 'c', 'd'])

License

MIT. See LICENSE

About

A collection of helper functions to perform immutable operations on plain JavaScript objects and arrays

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published