mirror of
https://github.com/make-42/hayai.git
synced 2025-01-18 18:47:10 +01:00
174 lines
3.4 KiB
Go
174 lines
3.4 KiB
Go
package beep
|
|
|
|
// Take returns a Streamer which streams at most num samples from s.
|
|
//
|
|
// The returned Streamer propagates s's errors through Err.
|
|
func Take(num int, s Streamer) Streamer {
|
|
return &take{
|
|
s: s,
|
|
remains: num,
|
|
}
|
|
}
|
|
|
|
type take struct {
|
|
s Streamer
|
|
remains int
|
|
}
|
|
|
|
func (t *take) Stream(samples [][2]float64) (n int, ok bool) {
|
|
if t.remains <= 0 {
|
|
return 0, false
|
|
}
|
|
toStream := t.remains
|
|
if len(samples) < toStream {
|
|
toStream = len(samples)
|
|
}
|
|
n, ok = t.s.Stream(samples[:toStream])
|
|
t.remains -= n
|
|
return n, ok
|
|
}
|
|
|
|
func (t *take) Err() error {
|
|
return t.s.Err()
|
|
}
|
|
|
|
// Loop takes a StreamSeeker and plays it count times. If count is negative, s is looped infinitely.
|
|
//
|
|
// The returned Streamer propagates s's errors.
|
|
func Loop(count int, s StreamSeeker) Streamer {
|
|
return &loop{
|
|
s: s,
|
|
remains: count,
|
|
}
|
|
}
|
|
|
|
type loop struct {
|
|
s StreamSeeker
|
|
remains int
|
|
}
|
|
|
|
func (l *loop) Stream(samples [][2]float64) (n int, ok bool) {
|
|
if l.remains == 0 || l.s.Err() != nil {
|
|
return 0, false
|
|
}
|
|
for len(samples) > 0 {
|
|
sn, sok := l.s.Stream(samples)
|
|
if !sok {
|
|
if l.remains > 0 {
|
|
l.remains--
|
|
}
|
|
if l.remains == 0 {
|
|
break
|
|
}
|
|
err := l.s.Seek(0)
|
|
if err != nil {
|
|
return n, true
|
|
}
|
|
continue
|
|
}
|
|
samples = samples[sn:]
|
|
n += sn
|
|
}
|
|
return n, true
|
|
}
|
|
|
|
func (l *loop) Err() error {
|
|
return l.s.Err()
|
|
}
|
|
|
|
// Seq takes zero or more Streamers and returns a Streamer which streams them one by one without pauses.
|
|
//
|
|
// Seq does not propagate errors from the Streamers.
|
|
func Seq(s ...Streamer) Streamer {
|
|
i := 0
|
|
return StreamerFunc(func(samples [][2]float64) (n int, ok bool) {
|
|
for i < len(s) && len(samples) > 0 {
|
|
sn, sok := s[i].Stream(samples)
|
|
samples = samples[sn:]
|
|
n, ok = n+sn, ok || sok
|
|
if !sok {
|
|
i++
|
|
}
|
|
}
|
|
return n, ok
|
|
})
|
|
}
|
|
|
|
// Mix takes zero or more Streamers and returns a Streamer which streams them mixed together.
|
|
//
|
|
// Mix does not propagate errors from the Streamers.
|
|
func Mix(s ...Streamer) Streamer {
|
|
return StreamerFunc(func(samples [][2]float64) (n int, ok bool) {
|
|
var tmp [512][2]float64
|
|
|
|
for len(samples) > 0 {
|
|
toStream := len(tmp)
|
|
if toStream > len(samples) {
|
|
toStream = len(samples)
|
|
}
|
|
|
|
// clear the samples
|
|
for i := range samples[:toStream] {
|
|
samples[i] = [2]float64{}
|
|
}
|
|
|
|
snMax := 0 // max number of streamed samples in this iteration
|
|
for _, st := range s {
|
|
// mix the stream
|
|
sn, sok := st.Stream(tmp[:toStream])
|
|
if sn > snMax {
|
|
snMax = sn
|
|
}
|
|
ok = ok || sok
|
|
|
|
for i := range tmp[:sn] {
|
|
samples[i][0] += tmp[i][0]
|
|
samples[i][1] += tmp[i][1]
|
|
}
|
|
}
|
|
|
|
n += snMax
|
|
if snMax < len(tmp) {
|
|
break
|
|
}
|
|
samples = samples[snMax:]
|
|
}
|
|
|
|
return n, ok
|
|
})
|
|
}
|
|
|
|
// Dup returns two Streamers which both stream the same data as the original s. The two Streamers
|
|
// can't be used concurrently without synchronization.
|
|
func Dup(s Streamer) (t, u Streamer) {
|
|
var tBuf, uBuf [][2]float64
|
|
return &dup{&tBuf, &uBuf, s}, &dup{&uBuf, &tBuf, s}
|
|
}
|
|
|
|
type dup struct {
|
|
myBuf, itsBuf *[][2]float64
|
|
s Streamer
|
|
}
|
|
|
|
func (d *dup) Stream(samples [][2]float64) (n int, ok bool) {
|
|
buf := *d.myBuf
|
|
n = copy(samples, buf)
|
|
ok = len(buf) > 0
|
|
buf = buf[n:]
|
|
samples = samples[n:]
|
|
*d.myBuf = buf
|
|
|
|
if len(samples) > 0 {
|
|
sn, sok := d.s.Stream(samples)
|
|
n += sn
|
|
ok = ok || sok
|
|
*d.itsBuf = append(*d.itsBuf, samples[:sn]...)
|
|
}
|
|
|
|
return n, ok
|
|
}
|
|
|
|
func (d *dup) Err() error {
|
|
return d.s.Err()
|
|
}
|