mild-map 1.0.0 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.ts CHANGED
@@ -1,6 +1,7 @@
1
1
  declare class MildMap<K, V> {
2
2
  #private;
3
3
  constructor(entries?: readonly (readonly [K, V])[] | null);
4
+ get size(): number;
4
5
  delete(key: K): boolean;
5
6
  get(key: K): V | undefined;
6
7
  has(key: K): boolean;
package/dist/index.js CHANGED
@@ -5,20 +5,39 @@ class MildMap {
5
5
  /* VARIABLES */
6
6
  #strong;
7
7
  #weak;
8
+ #size;
9
+ #finalizationRegistry;
10
+ #finalizationTokens;
8
11
  /* CONSTRUCTOR */
9
12
  constructor(entries) {
10
13
  /* VARIABLES */
11
14
  this.#strong = new Map();
12
15
  this.#weak = new WeakMap();
16
+ this.#size = 0;
17
+ this.#finalizationRegistry = new FinalizationRegistry(() => this.#size -= 1);
18
+ this.#finalizationTokens = new WeakMap();
13
19
  if (entries) {
14
20
  for (const [key, value] of entries) {
15
21
  this.set(key, value);
16
22
  }
17
23
  }
18
24
  }
25
+ /* GETTER API */
26
+ get size() {
27
+ return this.#size;
28
+ }
19
29
  /* API */
20
30
  delete(key) {
31
+ const hasKey = this.has(key);
32
+ if (!hasKey)
33
+ return false;
34
+ this.#size -= 1;
21
35
  if (isWeakReferrable(key)) {
36
+ const token = this.#finalizationTokens.get(key);
37
+ if (token) {
38
+ this.#finalizationRegistry.unregister(token);
39
+ this.#finalizationTokens.delete(key);
40
+ }
22
41
  return this.#weak.delete(key);
23
42
  }
24
43
  else {
@@ -42,8 +61,17 @@ class MildMap {
42
61
  }
43
62
  }
44
63
  set(key, value) {
64
+ const hasKey = this.has(key);
65
+ if (!hasKey) {
66
+ this.#size += 1;
67
+ }
45
68
  if (isWeakReferrable(key)) {
46
69
  this.#weak.set(key, value);
70
+ if (!hasKey) {
71
+ const token = {};
72
+ this.#finalizationRegistry.register(key, token, token);
73
+ this.#finalizationTokens.set(key, token);
74
+ }
47
75
  }
48
76
  else {
49
77
  this.#strong.set(key, value);
package/dist/utils.d.ts CHANGED
@@ -1,2 +1,2 @@
1
- declare const isWeakReferrable: (value: unknown) => value is symbol | object;
1
+ declare const isWeakReferrable: (value: unknown) => value is object;
2
2
  export { isWeakReferrable };
package/package.json CHANGED
@@ -2,7 +2,7 @@
2
2
  "name": "mild-map",
3
3
  "repository": "github:fabiospampinato/mild-map",
4
4
  "description": "A WeakMap that supports any value, it holds strong references to primitives, and weak references to objects.",
5
- "version": "1.0.0",
5
+ "version": "1.1.0",
6
6
  "type": "module",
7
7
  "main": "dist/index.js",
8
8
  "exports": "./dist/index.js",
package/readme.md CHANGED
@@ -12,6 +12,8 @@ npm install --save mild-map
12
12
 
13
13
  It has the same API of a WeakMap, except that it just works with primitives too.
14
14
 
15
+ It also implements the `size` property, like a Map.
16
+
15
17
  ## License
16
18
 
17
19
  MIT © Fabio Spampinato
package/src/index.ts CHANGED
@@ -11,6 +11,10 @@ class MildMap<K, V> {
11
11
 
12
12
  #strong = new Map<K, V> ();
13
13
  #weak = new WeakMap<any, V> ();
14
+ #size = 0;
15
+
16
+ #finalizationRegistry = new FinalizationRegistry ( () => this.#size -= 1 );
17
+ #finalizationTokens = new WeakMap<object, object> ();
14
18
 
15
19
  /* CONSTRUCTOR */
16
20
 
@@ -28,12 +32,35 @@ class MildMap<K, V> {
28
32
 
29
33
  }
30
34
 
35
+ /* GETTER API */
36
+
37
+ get size () {
38
+
39
+ return this.#size;
40
+
41
+ }
42
+
31
43
  /* API */
32
44
 
33
45
  delete ( key: K ): boolean {
34
46
 
47
+ const hasKey = this.has ( key );
48
+
49
+ if ( !hasKey ) return false;
50
+
51
+ this.#size -= 1;
52
+
35
53
  if ( isWeakReferrable ( key ) ) {
36
54
 
55
+ const token = this.#finalizationTokens.get ( key );
56
+
57
+ if ( token ) {
58
+
59
+ this.#finalizationRegistry.unregister ( token );
60
+ this.#finalizationTokens.delete ( key );
61
+
62
+ }
63
+
37
64
  return this.#weak.delete ( key );
38
65
 
39
66
  } else {
@@ -74,10 +101,27 @@ class MildMap<K, V> {
74
101
 
75
102
  set ( key: K, value: V ): this {
76
103
 
104
+ const hasKey = this.has ( key );
105
+
106
+ if ( !hasKey ) {
107
+
108
+ this.#size += 1;
109
+
110
+ }
111
+
77
112
  if ( isWeakReferrable ( key ) ) {
78
113
 
79
114
  this.#weak.set ( key, value );
80
115
 
116
+ if ( !hasKey ) {
117
+
118
+ const token = {};
119
+
120
+ this.#finalizationRegistry.register ( key, token, token );
121
+ this.#finalizationTokens.set ( key, token );
122
+
123
+ }
124
+
81
125
  } else {
82
126
 
83
127
  this.#strong.set ( key, value );
package/src/utils.ts CHANGED
@@ -5,7 +5,7 @@ import {SUPPORTS_SYMBOLS_AS_WEAKMAP_KEYS} from './constants';
5
5
 
6
6
  /* IMPORT */
7
7
 
8
- const isWeakReferrable = ( value: unknown ): value is object | symbol => {
8
+ const isWeakReferrable = ( value: unknown ): value is object => {
9
9
 
10
10
  if ( value === null ) return false;
11
11
 
package/test/index.js CHANGED
@@ -16,6 +16,8 @@ describe ( 'MildMap', it => {
16
16
  let primitive = 0;
17
17
  let object = {};
18
18
 
19
+ t.is ( map.size, 0 );
20
+
19
21
  t.is ( map.get ( primitive ), undefined );
20
22
  t.is ( map.get ( object ), undefined );
21
23
 
@@ -28,6 +30,8 @@ describe ( 'MildMap', it => {
28
30
  map.set ( primitive, 'primitive' );
29
31
  map.set ( object, 'object' );
30
32
 
33
+ t.is ( map.size, 2 );
34
+
31
35
  t.is ( map.get ( primitive ), 'primitive' );
32
36
  t.is ( map.get ( object ), 'object' );
33
37
 
@@ -37,9 +41,13 @@ describe ( 'MildMap', it => {
37
41
  t.true ( map.delete ( primitive ) );
38
42
  t.true ( map.delete ( object ) );
39
43
 
44
+ t.is ( map.size, 0 );
45
+
40
46
  map.set ( primitive, 'primitive' );
41
47
  map.set ( object, 'object' );
42
48
 
49
+ t.is ( map.size, 2 );
50
+
43
51
  /* CLEANUP */
44
52
 
45
53
  let deleted = 0;
@@ -55,6 +63,7 @@ describe ( 'MildMap', it => {
55
63
  await delay ( 500 );
56
64
 
57
65
  t.is ( deleted, 1 );
66
+ t.is ( map.size, 1 );
58
67
 
59
68
  });
60
69