-
Notifications
You must be signed in to change notification settings - Fork 4.3k
/
Copy pathobservable-map.ts
79 lines (73 loc) · 1.72 KB
/
observable-map.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/**
* WordPress dependencies
*/
import { useMemo, useSyncExternalStore } from '@wordpress/element';
export type ObservableMap< K, V > = {
get( name: K ): V | undefined;
set( name: K, value: V ): void;
delete( name: K ): void;
subscribe( name: K, listener: () => void ): () => void;
};
/**
* A key/value map where the individual entries are observable by subscribing to them
* with the `subscribe` methods.
*/
export function observableMap< K, V >(): ObservableMap< K, V > {
const map = new Map< K, V >();
const listeners = new Map< K, Set< () => void > >();
function callListeners( name: K ) {
const list = listeners.get( name );
if ( ! list ) {
return;
}
for ( const listener of list ) {
listener();
}
}
return {
get( name ) {
return map.get( name );
},
set( name, value ) {
map.set( name, value );
callListeners( name );
},
delete( name ) {
map.delete( name );
callListeners( name );
},
subscribe( name, listener ) {
let list = listeners.get( name );
if ( ! list ) {
list = new Set();
listeners.set( name, list );
}
list.add( listener );
return () => {
list.delete( listener );
if ( list.size === 0 ) {
listeners.delete( name );
}
};
},
};
}
/**
* React hook that lets you observe an individual entry in an `ObservableMap`.
*
* @param map The `ObservableMap` to observe.
* @param name The map key to observe.
*/
export function useObservableValue< K, V >(
map: ObservableMap< K, V >,
name: K
): V | undefined {
const [ subscribe, getValue ] = useMemo(
() => [
( listener: () => void ) => map.subscribe( name, listener ),
() => map.get( name ),
],
[ map, name ]
);
return useSyncExternalStore( subscribe, getValue, getValue );
}