Map

Overview

The @supercharge/map package provides an extended JavaScript Map class. It comes with additional methods like .isEmpty(), .map(callback), .filter(callback), and many more.

You already know methods like .map() from arrays and having them available on maps improves your development experience and makes your code more readable.

Installation

The @supercharge/map package lives independently from the Supercharge framework. Using it in your application requires you to install it as a project dependency:

npm i @supercharge/map

You can use this package with every project even if it’s not build on Supercharge. Enjoy!

Working With Maps

Import the @supercharge/map package and use it the same way you would use JavaScript’s Map class:

Please notice, the @supercharge/map package is not a drop-in replacement for the native Map class shipped by JavaScript.

JavaScript’s Map class provides a “wrong” parameter order in callback. JS gives you value, key, map which is aligning arrays, sets, and maps. And this feels wrong to me. It should be key, value, map. And this package does it “right”. That’s why this package exists.

This is the Map class JavaScript should have shipped.

const Map = require('@supercharge/map')

const cache = new Map()

cache.isEmpty()
// true

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.isNotEmpty()
// true

const users = cache.map((key, value, map) => {
  return { [key]: value }
})

// [{ 'user:1': 'Marcus' }, { 'user:2': 'Supercharge' }]

Available Methods

Here’s a list of available methods in the package:

clear

The clear method removes all entries from the map:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.size
// 2

cache.clear()

cache.size
// 0

delete

The delete method removes the entry identified by the given key.

Calling map.delete(key) returns true if the given key is present in the map and has been removed. Returns false if the key isn’t present in the map:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.has('user:1')
// true

const removed = cache.delete('user:1')
// true

cache.has('user:1')
// false

entries

The entries method returns an iterator containing the [key, value] pairs present in the map (in insertion order):

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

const iterator = cache.entries()

iterator.next().value
// ['user:1', 'Marcus']

iterator.next().value
// ['user:2', 'Supercharge']

You may also iterate through the [key, value] pairs using a for..of or loop:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

for (const [key, value] of cache.entries()) {
  console.log(`"${key}" mapping to --> ${value}`)
}

// "user:1" mapping to --> Marcus
// "user:2" mapping to --> Supercharge

filter

The filter method returns a map containing only items matching the given predicate.

The predicate function will be called once for each entry in the map (in insertion order). The predicate function receives the key, value, map arguments:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')
  .set('user:3', 'Norman')
  .set('user:4', 'Christian')


const filtered = cache.filter((key, value, map) => {
  return key === 'user:1' || value === 'Supercharge'
})

filtered.forEach((key, value, map) => {
  console.log(`"${key}" --> ${value}`)
})

// "user:1" --> Marcus
// "user:2" --> Supercharge

Please notice: the argument order key, value, map is different from the argument order of native a JavaScript Map value, key, map.

forEach

The forEach method processes a given action callback function once for each entry in the map in insertion order. The action callback function receives the key, value, map arguments:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.forEach((key, value, map) => {
  console.log(`"${key}" mapping to --> ${value}`)
})

// "user:1" mapping to --> Marcus
// "user:2" mapping to --> Supercharge

Please notice: the argument order key, value, map is different from the argument order of native a JavaScript Map value, key, map.

get

The get method returns the value identified by the given key. Returns undefined if the given key is not present in the map:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.get('user:1')
// 'Marcus'

cache.get('user:5000')
// undefined

getOrDefault

The getOrDefault method returns the value identified by the given key or the defaultValue if the given key is not present in the map:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.getOrDefault('user:1', 'Default-Username')
// 'Marcus'

cache.getOrDefault('user:5000', 'Default-Username')
// 'Default-Username'

has

The has method returns true if the given key is present in the map, otherwise false:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.has('user:1')
// true

cache.has('user:5000')
// false

isEmpty

The isEmpty method returns true if the map has no entries. Returns false if key/value pairs are present in the map:

const cache = new Map()

cache.isEmpty()
// true

cache.set('user:1', 'Marcus')

cache.isEmpty()
// false

isNotEmpty

The isNotEmpty method returns true if the map is empty. Returns false if key-value-pairs are present the map.

const cache = new Map()

cache.isNotEmpty()
// false

cache.set('user:1', 'Marcus')

cache.isNotEmpty()
// true

keys

The keys method returns an iterator containing the keys present in the map (in insertion order):

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

const keyIterator = cache.keys()

keyIterator.next().value
// 'user:1'

keyIterator.next().value
// 'user:2'

You may also iterate through the keys using a for..of or loop:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

for (const key of cache.keys()) {
  console.log(`"${key}" mapping to --> ${cache.get(key)}`)
}

// "user:1" mapping to --> Marcus
// "user:2" mapping to --> Supercharge

map

The map method returns an array containing the results of the given transform function. The transform function will be called once for each entry in the map in insertion order. The transform function receives the key, value, map arguments:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

const users = cache.map((key, value, map) => {
  return { [key]: value }
})

// [{ 'user:1': 'Marcus' }, { 'user:2': 'Supercharge' }]

Please notice: the argument order key, value, map is different from the argument order of native a JavaScript Map value, key, map.

of

The static of method creates a new map instance of the given key-value-pairs. It’s basically a shortcut for new Map(entries):

const cache = Map.of([
    ['user:1', 'Marcus'],
    ['user:2', 'Supercharge']
])

cache.get('user:1')
// 'Marcus'

cache.has('user:2')
// true

Notice the list of lists as the argument to create a new map instance. The lists with two items represent a key-value-pair.

set

The set method adds or updates an entry with the given key and value in the map:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

cache.get('user:1')
// 'Marcus'

cache.has('user:5000')
// false

size

The size method returns the number of key-value-pairs in the map:

const map = new Map([
  ['key', 'value'] // adding the first pair
])

map.set('name', 'Marcus') // adding the second pair

const size = map.size()
// 2

values

The values method returns an iterator object containing the values present in the map (in insertion order):

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

const valueIterator = cache.values()

valueIterator.next().value
// 'user:1'

valueIterator.next().value
// 'user:2'

You may also iterate through the values using a for..of or loop:

const cache = new Map()

cache
  .set('user:1', 'Marcus')
  .set('user:2', 'Supercharge')

for (const value of cache.values()) {
  console.log(value)
}

// 'Marcus'
// 'Supercharge'