mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-12 12:08:58 +01:00
audio: Add doc
This commit is contained in:
parent
65cbbdd5d9
commit
e93982ffb5
@ -12,6 +12,17 @@
|
|||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
|
// Package audio provides audio players. This can be used with or without ebiten package.
|
||||||
|
//
|
||||||
|
// The stream format must be 16-bit little endian and 2 channels.
|
||||||
|
//
|
||||||
|
// An audio context has a sample rate you can set and all streams you want to play must have the same
|
||||||
|
// sample rate.
|
||||||
|
//
|
||||||
|
// An audio context can generate 'players' (instances of audio.Player),
|
||||||
|
// and you can play sound by calling Play function of players.
|
||||||
|
// When multiple players play, mixing is automatically done.
|
||||||
|
// Note that too many players may cause distortion.
|
||||||
package audio
|
package audio
|
||||||
|
|
||||||
import (
|
import (
|
||||||
@ -182,11 +193,37 @@ func (s *mixingStream) playerCurrent(player *Player) time.Duration {
|
|||||||
|
|
||||||
// TODO: Enable to specify the format like Mono8?
|
// TODO: Enable to specify the format like Mono8?
|
||||||
|
|
||||||
|
// A Context is a current state of audio.
|
||||||
|
//
|
||||||
|
// The typical usage with ebiten package is:
|
||||||
|
//
|
||||||
|
// var audioContext *audio.Context
|
||||||
|
//
|
||||||
|
// func update(screen *ebiten.Image) error {
|
||||||
|
// // Update updates the audio stream by 1/60 [sec].
|
||||||
|
// if err := audioContext.Update(); err != nil {
|
||||||
|
// return err
|
||||||
|
// }
|
||||||
|
// // ...
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// func main() {
|
||||||
|
// audioContext, err = audio.NewContext(sampleRate)
|
||||||
|
// if err != nil {
|
||||||
|
// panic(err)
|
||||||
|
// }
|
||||||
|
// ebiten.Run(run, update, 320, 240, 2, "Audio test")
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// This is 'sync mode' in that game's (logical) time and audio time are synchronized.
|
||||||
|
// You can also call Update independently from the game loop as 'async mode'.
|
||||||
|
// In this case, audio goes on even when the game stops e.g. by diactivating the screen.
|
||||||
type Context struct {
|
type Context struct {
|
||||||
stream *mixingStream
|
stream *mixingStream
|
||||||
errorCh chan error
|
errorCh chan error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NewContext creates a new audio context with the given sample rate (e.g. 44100).
|
||||||
func NewContext(sampleRate int) (*Context, error) {
|
func NewContext(sampleRate int) (*Context, error) {
|
||||||
// TODO: Panic if one context exists.
|
// TODO: Panic if one context exists.
|
||||||
c := &Context{
|
c := &Context{
|
||||||
@ -220,11 +257,12 @@ func NewContext(sampleRate int) (*Context, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update proceeds the inner (logical) time of the context by 1/60 second.
|
// Update proceeds the inner (logical) time of the context by 1/60 second.
|
||||||
|
//
|
||||||
// This is expected to be called in the game's updating function (sync mode)
|
// This is expected to be called in the game's updating function (sync mode)
|
||||||
// or an independent goroutine with timers (unsync mode).
|
// or an independent goroutine with timers (async mode).
|
||||||
// In sync mode, the game logical time syncs the audio logical time and
|
// In sync mode, the game logical time syncs the audio logical time and
|
||||||
// you will find audio stops when the game stops e.g. when the window is deactivated.
|
// you will find audio stops when the game stops e.g. when the window is deactivated.
|
||||||
// In unsync mode, the audio never stops even when the game stops.
|
// In async mode, the audio never stops even when the game stops.
|
||||||
func (c *Context) Update() error {
|
func (c *Context) Update() error {
|
||||||
select {
|
select {
|
||||||
case err := <-c.errorCh:
|
case err := <-c.errorCh:
|
||||||
@ -255,16 +293,16 @@ type Player struct {
|
|||||||
volume float64
|
volume float64
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewPlayer creates a new player with the given data to the given channel.
|
// NewPlayer creates a new player with the given stream.
|
||||||
// The given data is queued to the end of the buffer.
|
|
||||||
// This may not be played immediately when data already exists in the buffer.
|
|
||||||
//
|
//
|
||||||
// src's format must be linear PCM (16bits, 2 channel stereo, little endian)
|
// src's format must be linear PCM (16bits little endian, 2 channel stereo)
|
||||||
// without a header (e.g. RIFF header).
|
// without a header (e.g. RIFF header).
|
||||||
|
// The sample rate must be same as that of the audio context.
|
||||||
func (c *Context) NewPlayer(src ReadSeekCloser) (*Player, error) {
|
func (c *Context) NewPlayer(src ReadSeekCloser) (*Player, error) {
|
||||||
return c.stream.newPlayer(src)
|
return c.stream.newPlayer(src)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Close closes the stream. Ths source stream passed by NewPlayer will also be closed.
|
||||||
func (p *Player) Close() error {
|
func (p *Player) Close() error {
|
||||||
return p.stream.closePlayer(p)
|
return p.stream.closePlayer(p)
|
||||||
}
|
}
|
||||||
@ -296,19 +334,23 @@ func (p *Player) bufferLength() int {
|
|||||||
return len(p.buf)
|
return len(p.buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Play plays the stream.
|
||||||
func (p *Player) Play() error {
|
func (p *Player) Play() error {
|
||||||
p.stream.addPlayer(p)
|
p.stream.addPlayer(p)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsPlaying returns boolean indicating whether the player is playing.
|
||||||
func (p *Player) IsPlaying() bool {
|
func (p *Player) IsPlaying() bool {
|
||||||
return p.stream.hasPlayer(p)
|
return p.stream.hasPlayer(p)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Rewind rewinds the current position to the start.
|
||||||
func (p *Player) Rewind() error {
|
func (p *Player) Rewind() error {
|
||||||
return p.Seek(0)
|
return p.Seek(0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Seek seeks the position with the given offset.
|
||||||
func (p *Player) Seek(offset time.Duration) error {
|
func (p *Player) Seek(offset time.Duration) error {
|
||||||
return p.stream.seekPlayer(p, offset)
|
return p.stream.seekPlayer(p, offset)
|
||||||
}
|
}
|
||||||
@ -323,20 +365,23 @@ func (p *Player) seek(offset int64) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Pause pauses the playing.
|
||||||
func (p *Player) Pause() error {
|
func (p *Player) Pause() error {
|
||||||
p.stream.removePlayer(p)
|
p.stream.removePlayer(p)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Current returns the current position.
|
||||||
func (p *Player) Current() time.Duration {
|
func (p *Player) Current() time.Duration {
|
||||||
return p.stream.playerCurrent(p)
|
return p.stream.playerCurrent(p)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Volume returns the current volume of this player [0-1].
|
||||||
func (p *Player) Volume() float64 {
|
func (p *Player) Volume() float64 {
|
||||||
return p.volume
|
return p.volume
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetVolume sets the volume.
|
// SetVolume sets the volume of this player.
|
||||||
// volume must be in between 0 and 1. This function panics otherwise.
|
// volume must be in between 0 and 1. This function panics otherwise.
|
||||||
func (p *Player) SetVolume(volume float64) {
|
func (p *Player) SetVolume(volume float64) {
|
||||||
// The condition must be true when volume is NaN.
|
// The condition must be true when volume is NaN.
|
||||||
|
Loading…
Reference in New Issue
Block a user