This repository has been archived by the owner on Feb 16, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 344
/
dispatcher.go
84 lines (75 loc) · 2.07 KB
/
dispatcher.go
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
80
81
82
83
84
package astilectron
import "sync"
// Listener represents a listener executed when an event is dispatched
type Listener func(e Event) (deleteListener bool)
// listenable represents an object that can listen
type listenable interface {
On(eventName string, l Listener)
}
// dispatcher represents an object capable of dispatching events
type dispatcher struct {
id int
// Indexed by target ID then by event name then be listener id
// We use a map[int]Listener so that deletion is as smooth as possible
// It means it doesn't store listeners in order
l map[string]map[string]map[int]Listener
m sync.Mutex
}
// newDispatcher creates a new dispatcher
func newDispatcher() *dispatcher {
return &dispatcher{
l: make(map[string]map[string]map[int]Listener),
}
}
// addListener adds a listener
func (d *dispatcher) addListener(targetID, eventName string, l Listener) {
d.m.Lock()
defer d.m.Unlock()
if _, ok := d.l[targetID]; !ok {
d.l[targetID] = make(map[string]map[int]Listener)
}
if _, ok := d.l[targetID][eventName]; !ok {
d.l[targetID][eventName] = make(map[int]Listener)
}
d.id++
d.l[targetID][eventName][d.id] = l
}
// delListener delete a specific listener
func (d *dispatcher) delListener(targetID, eventName string, id int) {
d.m.Lock()
defer d.m.Unlock()
if _, ok := d.l[targetID]; !ok {
return
}
if _, ok := d.l[targetID][eventName]; !ok {
return
}
delete(d.l[targetID][eventName], id)
}
// Dispatch dispatches an event
func (d *dispatcher) dispatch(e Event) {
// needed so dispatches of events triggered in the listeners can be received without blocking
go func() {
for id, l := range d.listeners(e.TargetID, e.Name) {
if l(e) {
d.delListener(e.TargetID, e.Name, id)
}
}
}()
}
// listeners returns the listeners for a target ID and an event name
func (d *dispatcher) listeners(targetID, eventName string) (l map[int]Listener) {
d.m.Lock()
defer d.m.Unlock()
l = map[int]Listener{}
if _, ok := d.l[targetID]; !ok {
return
}
if _, ok := d.l[targetID][eventName]; !ok {
return
}
for k, v := range d.l[targetID][eventName] {
l[k] = v
}
return
}