2016-04-13 18:26:31 +02:00
<!DOCTYPE html>
2016-08-27 17:58:27 +02:00
< script src = "../scripts/force-https.js" > < / script >
2016-04-13 18:26:31 +02:00
< link rel = "shortcut icon" href = "../favicon.png" type = "image/png" >
< link rel = "icon" href = "../favicon.png" type = "image/png" >
< title > Ebiten example - audio< / title >
2016-08-27 17:48:15 +02:00
< link rel = "stylesheet" href = "../stylesheets/bootstrap.min.css" >
< link rel = "stylesheet" href = "../stylesheets/highlight-github.css" >
< link rel = "stylesheet" href = "../stylesheets/ebiten.css" >
2016-09-02 18:55:07 +02:00
< script src = "../scripts/googleanalytics.js" > < / script >
2016-08-26 19:28:57 +02:00
< header class = "navbar" > < div class = "container" >
< div class = "navbar-header" >
2016-08-27 17:48:15 +02:00
< a class = "navbar-brand" href = ".." > Ebiten< / a >
2016-08-26 19:28:57 +02:00
< / div >
2016-08-26 19:37:08 +02:00
< nav class = "collapse navbar-collapse" >
< ul class = "nav navbar-nav navbar-right" >
< li > < a href = "https://github.com/hajimehoshi/ebiten" > GitHub< / a > < / li >
2016-08-27 17:58:27 +02:00
< li > < a href = "https://godoc.org/github.com/hajimehoshi/ebiten" > GoDoc< / a > < / li >
2016-08-26 19:37:08 +02:00
< li > < a href = "https://github.com/hajimehoshi/ebiten/wiki" > Wiki< / a >
< / ul >
< / nav >
2016-08-26 19:28:57 +02:00
< / header >
2016-04-13 18:26:31 +02:00
2016-08-26 19:28:57 +02:00
< main > < div class = "container" >
2016-08-26 20:18:16 +02:00
< h2 > Ebiten example - audio< / h2 >
2016-08-26 19:28:57 +02:00
< iframe src = "audio.content.html" width = "640" height = "480" > < / iframe >
2016-08-27 07:00:06 +02:00
< pre > < code class = "language-go" > // + build example
2016-08-26 16:33:36 +02:00
2017-08-17 16:13:15 +02:00
// This is an example to implement an audio player.
// See examples/wav for a simpler example to play a sound file.
2016-08-26 16:33:36 +02:00
package main
2016-04-13 18:26:31 +02:00
import (
2016-08-26 22:36:52 +02:00
" fmt"
" image/color"
" io/ioutil"
" log"
" time"
2016-04-13 18:26:31 +02:00
2016-08-26 22:36:52 +02:00
" github.com/hajimehoshi/ebiten"
" github.com/hajimehoshi/ebiten/audio"
2017-07-01 10:10:55 +02:00
" github.com/hajimehoshi/ebiten/audio/mp3"
2016-08-26 22:36:52 +02:00
" github.com/hajimehoshi/ebiten/audio/wav"
" github.com/hajimehoshi/ebiten/ebitenutil"
2016-04-13 18:26:31 +02:00
)
const (
2016-08-26 22:36:52 +02:00
screenWidth = 320
screenHeight = 240
2017-06-09 04:43:50 +02:00
2017-07-01 10:10:55 +02:00
// This sample rate doesn' t match with wav/mp3' s sample rate,
2017-06-09 04:43:50 +02:00
// but decoders adjust them.
sampleRate = 48000
2016-04-13 18:26:31 +02:00
)
var (
2016-08-26 22:36:52 +02:00
playerBarImage *ebiten.Image
playerCurrentImage *ebiten.Image
2016-04-13 18:26:31 +02:00
)
func init() {
2017-06-09 04:43:50 +02:00
playerBarImage, _ = ebiten.NewImage(300, 4, ebiten.FilterNearest)
playerBarImage.Fill(& color.RGBA{0x80, 0x80, 0x80, 0xff})
2016-04-13 18:26:31 +02:00
2017-06-09 04:43:50 +02:00
playerCurrentImage, _ = ebiten.NewImage(4, 10, ebiten.FilterNearest)
playerCurrentImage.Fill(& color.RGBA{0xff, 0xff, 0xff, 0xff})
}
type Input struct {
mouseButtonStates map[ebiten.MouseButton]int
keyStates map[ebiten.Key]int
}
func (i *Input) update() {
2017-08-17 16:13:15 +02:00
for _, key := range []ebiten.Key{ebiten.KeyP, ebiten.KeyS, ebiten.KeyX, ebiten.KeyZ, ebiten.KeyB} {
2017-06-09 04:43:50 +02:00
if !ebiten.IsKeyPressed(key) {
i.keyStates[key] = 0
} else {
i.keyStates[key]+ +
}
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
if !ebiten.IsMouseButtonPressed(ebiten.MouseButtonLeft) {
i.mouseButtonStates[ebiten.MouseButtonLeft] = 0
} else {
i.mouseButtonStates[ebiten.MouseButtonLeft]+ +
2016-08-26 22:36:52 +02:00
}
2016-04-13 18:26:31 +02:00
}
2017-06-09 04:43:50 +02:00
func (i *Input) isKeyTriggered(key ebiten.Key) bool {
return i.keyStates[key] == 1
}
func (i *Input) isKeyPressed(key ebiten.Key) bool {
return i.keyStates[key] > 0
}
func (i *Input) isMouseButtonTriggered(mouseButton ebiten.MouseButton) bool {
return i.mouseButtonStates[mouseButton] == 1
}
2016-04-13 18:26:31 +02:00
type Player struct {
2017-06-09 04:43:50 +02:00
input *Input
audioContext *audio.Context
audioPlayer *audio.Player
2017-07-01 10:10:55 +02:00
current time.Duration
2017-06-09 04:43:50 +02:00
total time.Duration
seBytes []uint8
seCh chan []uint8
volume128 int
2016-04-13 18:26:31 +02:00
}
var (
2017-06-09 04:43:50 +02:00
musicPlayer *Player
2016-04-13 18:26:31 +02:00
)
func playerBarRect() (x, y, w, h int) {
2016-08-26 22:36:52 +02:00
w, h = playerBarImage.Size()
x = (screenWidth - w) / 2
y = screenHeight - h - 16
return
2016-04-13 18:26:31 +02:00
}
2017-06-09 04:43:50 +02:00
func NewPlayer(audioContext *audio.Context) (*Player, error) {
const bytesPerSample = 4 // TODO: This should be defined in audio package
wavF, err := ebitenutil.OpenFile(" _resources/audio/jab.wav" )
if err != nil {
return nil, err
2016-08-26 22:36:52 +02:00
}
2017-07-01 10:10:55 +02:00
mp3F, err := ebitenutil.OpenFile(" _resources/audio/classic.mp3" )
2017-06-09 04:43:50 +02:00
if err != nil {
return nil, err
2016-08-26 22:36:52 +02:00
}
2017-07-01 10:10:55 +02:00
s, err := mp3.Decode(audioContext, mp3F)
2017-06-09 04:43:50 +02:00
if err != nil {
return nil, err
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
p, err := audio.NewPlayer(audioContext, s)
2016-08-26 22:36:52 +02:00
if err != nil {
2017-06-09 04:43:50 +02:00
return nil, err
}
player := & Player{
input: & Input{
mouseButtonStates: map[ebiten.MouseButton]int{},
keyStates: map[ebiten.Key]int{},
},
audioContext: audioContext,
audioPlayer: p,
total: time.Second * time.Duration(s.Size()) / bytesPerSample / sampleRate,
volume128: 128,
seCh: make(chan []uint8),
}
2017-07-01 10:10:55 +02:00
if player.total == 0 {
player.total = 1
}
2017-06-09 04:43:50 +02:00
player.audioPlayer.Play()
go func() {
s, err := wav.Decode(audioContext, wavF)
if err != nil {
log.Fatal(err)
return
}
b, err := ioutil.ReadAll(s)
if err != nil {
log.Fatal(err)
return
}
player.seCh < - b
}()
return player, nil
}
func (p *Player) update() error {
p.input.update()
select {
case p.seBytes = < -p.seCh:
close(p.seCh)
p.seCh = nil
default:
}
2017-07-01 10:10:55 +02:00
if p.audioPlayer.IsPlaying() {
p.current = p.audioPlayer.Current()
}
2017-06-09 04:43:50 +02:00
p.updateBar()
p.updatePlayPause()
p.updateSE()
p.updateVolume()
2017-08-17 16:13:15 +02:00
if p.input.isKeyTriggered(ebiten.KeyB) {
b := ebiten.IsRunnableInBackground()
ebiten.SetRunnableInBackground(!b)
}
2017-06-09 04:43:50 +02:00
if err := p.audioContext.Update(); err != nil {
2016-08-26 22:36:52 +02:00
return err
}
2017-06-09 04:43:50 +02:00
return nil
2016-04-13 18:26:31 +02:00
}
2017-06-09 04:43:50 +02:00
func (p *Player) updateSE() {
if p.seBytes == nil {
2017-03-04 15:27:25 +01:00
return
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
if !p.input.isKeyTriggered(ebiten.KeyP) {
return
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
sePlayer, _ := audio.NewPlayerFromBytes(p.audioContext, p.seBytes)
sePlayer.Play()
2016-04-13 18:26:31 +02:00
}
2017-06-09 04:43:50 +02:00
func (p *Player) updateVolume() {
if p.input.isKeyPressed(ebiten.KeyZ) {
p.volume128--
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
if p.input.isKeyPressed(ebiten.KeyX) {
p.volume128+ +
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
if p.volume128 < 0 {
p.volume128 = 0
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
if 128 < p.volume128 {
p.volume128 = 128
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
p.audioPlayer.SetVolume(float64(p.volume128) / 128)
2016-04-13 18:26:31 +02:00
}
2017-06-09 04:43:50 +02:00
func (p *Player) updatePlayPause() {
if !p.input.isKeyTriggered(ebiten.KeyS) {
2017-03-04 15:27:25 +01:00
return
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
if p.audioPlayer.IsPlaying() {
p.audioPlayer.Pause()
2017-03-04 15:27:25 +01:00
return
2016-11-28 19:42:16 +01:00
}
2017-06-09 04:43:50 +02:00
p.audioPlayer.Play()
}
func (p *Player) updateBar() {
if !p.input.isMouseButtonTriggered(ebiten.MouseButtonLeft) {
2017-03-04 15:27:25 +01:00
return
2016-08-26 22:36:52 +02:00
}
2017-06-09 04:43:50 +02:00
// Start seeking.
2016-08-26 22:36:52 +02:00
x, y := ebiten.CursorPosition()
bx, by, bw, bh := playerBarRect()
const padding = 4
if y < by-padding || by+ bh+ padding < = y {
2017-03-04 15:27:25 +01:00
return
2016-08-26 22:36:52 +02:00
}
if x < bx || bx+ bw < = x {
2017-03-04 15:27:25 +01:00
return
2016-08-26 22:36:52 +02:00
}
pos := time.Duration(x-bx) * p.total / time.Duration(bw)
2017-07-01 10:10:55 +02:00
p.current = pos
p.audioPlayer.Seek(pos)
2016-04-13 18:26:31 +02:00
}
func (p *Player) close() error {
2016-08-26 22:36:52 +02:00
return p.audioPlayer.Close()
2016-04-13 18:26:31 +02:00
}
2017-06-09 04:43:50 +02:00
func (p *Player) draw(screen *ebiten.Image) {
2016-08-26 22:36:52 +02:00
op := & ebiten.DrawImageOptions{}
x, y, w, h := playerBarRect()
op.GeoM.Translate(float64(x), float64(y))
2017-03-04 15:27:25 +01:00
screen.DrawImage(playerBarImage, op)
2016-08-26 22:36:52 +02:00
currentTimeStr := " 00:00"
2017-06-09 04:43:50 +02:00
// Current Time
2017-07-01 10:10:55 +02:00
c := p.current
2017-06-09 04:43:50 +02:00
m := (c / time.Minute) % 100
s := (c / time.Second) % 60
currentTimeStr = fmt.Sprintf(" %02d:%02d" , m, s)
// Bar
cw, ch := playerCurrentImage.Size()
cx := int(time.Duration(w)*c/p.total) + x - cw/2
cy := y - (ch-h)/2
op = & ebiten.DrawImageOptions{}
op.GeoM.Translate(float64(cx), float64(cy))
screen.DrawImage(playerCurrentImage, op)
2016-04-13 18:26:31 +02:00
2016-08-26 22:36:52 +02:00
msg := fmt.Sprintf(`FPS: %0.2f
2016-04-13 18:26:31 +02:00
Press S to toggle Play/Pause
Press P to play SE
Press Z or X to change volume of the music
2017-08-17 16:13:15 +02:00
Press B to switch the run-in-background state
2016-04-13 18:26:31 +02:00
%s`, ebiten.CurrentFPS(), currentTimeStr)
2017-03-04 15:27:25 +01:00
ebitenutil.DebugPrint(screen, msg)
2017-06-09 04:43:50 +02:00
}
func update(screen *ebiten.Image) error {
if err := musicPlayer.update(); err != nil {
2016-08-26 22:36:52 +02:00
return err
}
2017-06-09 04:43:50 +02:00
if ebiten.IsRunningSlowly() {
return nil
}
musicPlayer.draw(screen)
2016-08-26 22:36:52 +02:00
return nil
2016-04-13 18:26:31 +02:00
}
func main() {
2017-06-09 04:43:50 +02:00
audioContext, err := audio.NewContext(sampleRate)
2016-04-13 18:26:31 +02:00
if err != nil {
2016-08-26 22:36:52 +02:00
log.Fatal(err)
2016-04-13 18:26:31 +02:00
}
2017-07-13 19:26:41 +02:00
2017-06-09 04:43:50 +02:00
musicPlayer, err = NewPlayer(audioContext)
2016-04-13 18:26:31 +02:00
if err != nil {
2016-08-26 22:36:52 +02:00
log.Fatal(err)
}
if err := ebiten.Run(update, screenWidth, screenHeight, 2, " Audio (Ebiten Demo)" ); err != nil {
log.Fatal(err)
}
if musicPlayer != nil {
if err := musicPlayer.close(); err != nil {
log.Fatal(err)
}
}
2016-04-13 18:26:31 +02:00
}
< / code > < / pre >
2016-08-26 19:28:57 +02:00
< / div > < / main >
< footer > < div class = "container" >
< p > © 2013 Hajime Hoshi< / p >
2016-08-28 00:10:41 +02:00
< p > Code is licensed under < a href = "https://github.com/hajimehoshi/ebiten/blob/master/LICENSE" > the Apache License 2.0< / a > .< / p >
2016-08-26 19:28:57 +02:00
< p > The content of this page is licensed under < a href = "https://creativecommons.org/licenses/by/4.0/" > the Creative Commons Attribution 4.0 License< / a > .< / p >
< / div > < / footer >
2016-04-13 18:26:31 +02:00
2016-08-27 17:48:15 +02:00
< script src = "../scripts/highlight.pack.js" > < / script >
2016-08-27 07:00:06 +02:00
< script > hljs . initHighlightingOnLoad ( ) ; < / script >