mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-23 17:32:02 +01:00
audio: Add NewInfiniteLoopWithIntro (#652)
This commit is contained in:
parent
b9265d5120
commit
59bcbca379
@ -19,56 +19,107 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
)
|
)
|
||||||
|
|
||||||
// InfiniteLoop represents a loop which never ends.
|
// InfiniteLoop represents a looped stream which never ends.
|
||||||
type InfiniteLoop struct {
|
type InfiniteLoop struct {
|
||||||
src ReadSeekCloser
|
src ReadSeekCloser
|
||||||
size int64
|
lstart int64
|
||||||
|
llength int64
|
||||||
|
pos int64
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewInfiniteLoop creates a new infinite loop stream with a source stream and size in bytes.
|
// NewInfiniteLoop creates a new infinite loop stream with a source stream and length in bytes.
|
||||||
func NewInfiniteLoop(src ReadSeekCloser, size int64) *InfiniteLoop {
|
func NewInfiniteLoop(src ReadSeekCloser, length int64) *InfiniteLoop {
|
||||||
|
return NewInfiniteLoopWithIntro(src, 0, length)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewInfiniteLoopWithIntro creates a new infinite loop stream with an intro part.
|
||||||
|
// NewInfiniteLoopWithIntro accepts a source stream src, introLength in bytes and loopLength in bytes.
|
||||||
|
func NewInfiniteLoopWithIntro(src ReadSeekCloser, introLength int64, loopLength int64) *InfiniteLoop {
|
||||||
return &InfiniteLoop{
|
return &InfiniteLoop{
|
||||||
src: src,
|
src: src,
|
||||||
size: size,
|
lstart: introLength,
|
||||||
|
llength: loopLength,
|
||||||
|
pos: -1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (i *InfiniteLoop) length() int64 {
|
||||||
|
return i.lstart + i.llength
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InfiniteLoop) ensurePos() error {
|
||||||
|
if i.pos >= 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
pos, err := i.src.Seek(0, io.SeekCurrent)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if pos >= i.length() {
|
||||||
|
return fmt.Errorf("audio: stream position must be less than the specified length")
|
||||||
|
}
|
||||||
|
i.pos = pos
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// Read is implementation of ReadSeekCloser's Read.
|
// Read is implementation of ReadSeekCloser's Read.
|
||||||
func (i *InfiniteLoop) Read(b []byte) (int, error) {
|
func (i *InfiniteLoop) Read(b []byte) (int, error) {
|
||||||
|
if err := i.ensurePos(); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if i.pos+int64(len(b)) > i.length() {
|
||||||
|
b = b[:i.length()-i.pos]
|
||||||
|
}
|
||||||
|
|
||||||
n, err := i.src.Read(b)
|
n, err := i.src.Read(b)
|
||||||
if err == io.EOF {
|
i.pos += int64(n)
|
||||||
if _, err := i.Seek(0, io.SeekStart); err != nil {
|
if i.pos > i.length() {
|
||||||
|
panic("not reached")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil && err != io.EOF {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err == io.EOF || i.pos == i.length() {
|
||||||
|
pos, err := i.Seek(i.lstart, io.SeekStart)
|
||||||
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
err = nil
|
i.pos = pos
|
||||||
}
|
}
|
||||||
return n, err
|
return n, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Seek is implementation of ReadSeekCloser's Seek.
|
// Seek is implementation of ReadSeekCloser's Seek.
|
||||||
func (i *InfiniteLoop) Seek(offset int64, whence int) (int64, error) {
|
func (i *InfiniteLoop) Seek(offset int64, whence int) (int64, error) {
|
||||||
|
if err := i.ensurePos(); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
next := int64(0)
|
next := int64(0)
|
||||||
switch whence {
|
switch whence {
|
||||||
case io.SeekStart:
|
case io.SeekStart:
|
||||||
next = offset
|
next = offset
|
||||||
case io.SeekCurrent:
|
case io.SeekCurrent:
|
||||||
current, err := i.src.Seek(0, io.SeekCurrent)
|
next = i.pos + offset
|
||||||
if err != nil {
|
|
||||||
return 0, err
|
|
||||||
}
|
|
||||||
next = current + offset
|
|
||||||
case io.SeekEnd:
|
case io.SeekEnd:
|
||||||
return 0, fmt.Errorf("audio: whence must be io.SeekStart or io.SeekCurrent for InfiniteLoop")
|
return 0, fmt.Errorf("audio: whence must be io.SeekStart or io.SeekCurrent for InfiniteLoop")
|
||||||
}
|
}
|
||||||
if next < 0 {
|
if next < 0 {
|
||||||
return 0, fmt.Errorf("audio: position must >= 0")
|
return 0, fmt.Errorf("audio: position must >= 0")
|
||||||
}
|
}
|
||||||
next %= i.size
|
if next >= i.lstart {
|
||||||
pos, err := i.src.Seek(next, io.SeekStart)
|
next = ((next - i.lstart) % i.llength) + i.lstart
|
||||||
if err != nil {
|
}
|
||||||
|
// Ignore the new position returned by Seek since the source position might not be match with the position
|
||||||
|
// managed by this.
|
||||||
|
if _, err := i.src.Seek(next, io.SeekStart); err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
return pos, nil
|
i.pos = next
|
||||||
|
return i.pos, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close is implementation of ReadSeekCloser's Close.
|
// Close is implementation of ReadSeekCloser's Close.
|
||||||
|
@ -78,3 +78,72 @@ func TestInfiniteLoop(t *testing.T) {
|
|||||||
t.Errorf("got: %v, want: %v", err, nil)
|
t.Errorf("got: %v, want: %v", err, nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestInfiniteLoopWithIntro(t *testing.T) {
|
||||||
|
const (
|
||||||
|
srcLength = 17 * 4
|
||||||
|
introLength = 19 * 4
|
||||||
|
loopLength = 23 * 4
|
||||||
|
)
|
||||||
|
|
||||||
|
indexToByte := func(index int) byte {
|
||||||
|
return byte(math.Sin(float64(index)) * 256)
|
||||||
|
}
|
||||||
|
src := make([]byte, srcLength)
|
||||||
|
for i := range src {
|
||||||
|
src[i] = indexToByte(i)
|
||||||
|
}
|
||||||
|
srcInf := NewInfiniteLoop(BytesReadSeekCloser(src), srcLength)
|
||||||
|
l := NewInfiniteLoopWithIntro(srcInf, introLength, loopLength)
|
||||||
|
|
||||||
|
buf := make([]byte, srcLength*4)
|
||||||
|
if _, err := io.ReadFull(l, buf); err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
for i, b := range buf {
|
||||||
|
got := b
|
||||||
|
want := byte(0)
|
||||||
|
if i < introLength {
|
||||||
|
want = indexToByte(i % srcLength)
|
||||||
|
} else {
|
||||||
|
want = indexToByte(((i-introLength)%loopLength + introLength) % srcLength)
|
||||||
|
}
|
||||||
|
if got != want {
|
||||||
|
t.Errorf("index: %d, got: %v, want: %v", i, got, want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
n, err := l.Seek(srcLength*5+128, io.SeekStart)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
if want := int64((srcLength*5+128-introLength)%loopLength + introLength); n != want {
|
||||||
|
t.Errorf("got: %v, want: %v", n, want)
|
||||||
|
}
|
||||||
|
|
||||||
|
n2, err := l.Seek(srcLength*6+64, io.SeekCurrent)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
if want := int64(((srcLength*11+192)-introLength)%loopLength + introLength); n2 != want {
|
||||||
|
t.Errorf("got: %v, want: %v", n, want)
|
||||||
|
}
|
||||||
|
|
||||||
|
buf2 := make([]byte, srcLength*7)
|
||||||
|
if _, err := io.ReadFull(l, buf2); err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
for i, b := range buf2 {
|
||||||
|
got := b
|
||||||
|
idx := ((int(n2+int64(i))-introLength)%loopLength + introLength) % srcLength
|
||||||
|
want := indexToByte(idx)
|
||||||
|
if got != want {
|
||||||
|
t.Errorf("index: %d, got: %v, want: %v", i, got, want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Seek to negative position is an error.
|
||||||
|
if _, err := l.Seek(-1, io.SeekStart); err == nil {
|
||||||
|
t.Errorf("got: %v, want: %v", err, nil)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user