-
Notifications
You must be signed in to change notification settings - Fork 16
/
mixer.go
91 lines (76 loc) · 2.41 KB
/
mixer.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
85
86
87
88
89
90
91
package beep
// Mixer allows for dynamic mixing of arbitrary number of Streamers. Mixer automatically removes
// drained Streamers. Depending on the KeepAlive() setting, Stream will either play silence or
// drain when all Streamers have been drained. By default, Mixer keeps playing silence.
type Mixer struct {
streamers []Streamer
stopWhenEmpty bool
}
// KeepAlive configures the Mixer to either keep playing silence when all its Streamers have
// drained (keepAlive == true) or stop playing (keepAlive == false).
func (m *Mixer) KeepAlive(keepAlive bool) {
m.stopWhenEmpty = !keepAlive
}
// Len returns the number of Streamers currently playing in the Mixer.
func (m *Mixer) Len() int {
return len(m.streamers)
}
// Add adds Streamers to the Mixer.
func (m *Mixer) Add(s ...Streamer) {
m.streamers = append(m.streamers, s...)
}
// Clear removes all Streamers from the mixer.
func (m *Mixer) Clear() {
for i := range m.streamers {
m.streamers[i] = nil
}
m.streamers = m.streamers[:0]
}
// Stream the samples of all Streamers currently in the Mixer mixed together. Depending on the
// KeepAlive() setting, Stream will either play silence or drain when all Streamers have been
// drained.
func (m *Mixer) Stream(samples [][2]float64) (n int, ok bool) {
if m.stopWhenEmpty && len(m.streamers) == 0 {
return 0, false
}
var tmp [512][2]float64
for len(samples) > 0 {
toStream := min(len(tmp), len(samples))
// clear the samples
clear(samples[:toStream])
snMax := 0
for si := 0; si < len(m.streamers); si++ {
// mix the stream
sn, sok := m.streamers[si].Stream(tmp[:toStream])
for i := range tmp[:sn] {
samples[i][0] += tmp[i][0]
samples[i][1] += tmp[i][1]
}
if sn > snMax {
snMax = sn
}
if sn < toStream || !sok {
// remove drained streamer
last := len(m.streamers) - 1
m.streamers[si] = m.streamers[last]
m.streamers[last] = nil
m.streamers = m.streamers[:last]
si--
if m.stopWhenEmpty && len(m.streamers) == 0 {
return n + snMax, true
}
}
}
samples = samples[toStream:]
n += toStream
}
return n, true
}
// Err always returns nil for Mixer.
//
// There are two reasons. The first one is that erroring Streamers are immediately drained and
// removed from the Mixer. The second one is that one Streamer shouldn't break the whole Mixer and
// you should handle the errors right where they can happen.
func (m *Mixer) Err() error {
return nil
}