145 lines
3.2 KiB
Go
145 lines
3.2 KiB
Go
package engine
|
|
|
|
import (
|
|
"encoding/gob"
|
|
"io/fs"
|
|
"log"
|
|
"sync"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2"
|
|
)
|
|
|
|
func init() {
|
|
gob.Register(&Game{})
|
|
}
|
|
|
|
// Game implements the ebiten methods using a collection of components.
|
|
// One component must be the designated root component - usually a
|
|
// scene of some kind.
|
|
type Game struct {
|
|
Disabled
|
|
Hidden
|
|
ScreenWidth int
|
|
ScreenHeight int
|
|
Root DrawUpdater // typically a *Scene or SceneRef though
|
|
|
|
dbmu sync.RWMutex
|
|
db map[string]interface{}
|
|
}
|
|
|
|
// Draw draws the entire thing, with default draw options.
|
|
func (g *Game) Draw(screen *ebiten.Image) {
|
|
if g.Hidden {
|
|
return
|
|
}
|
|
g.Root.Draw(screen, ebiten.DrawImageOptions{})
|
|
}
|
|
|
|
// Layout returns the configured screen width/height.
|
|
func (g *Game) Layout(outsideWidth, outsideHeight int) (w, h int) {
|
|
return g.ScreenWidth, g.ScreenHeight
|
|
}
|
|
|
|
// Update updates the scene.
|
|
func (g *Game) Update() error {
|
|
if g.Disabled {
|
|
return nil
|
|
}
|
|
return g.Root.Update()
|
|
}
|
|
|
|
// RegisterComponent tells the game there is a new component. Currently this is
|
|
// only necessary for components with IDs.
|
|
func (g *Game) RegisterComponent(c interface{}) {
|
|
i, ok := c.(Identifier)
|
|
if !ok {
|
|
return
|
|
}
|
|
id := i.Ident()
|
|
if id == "" {
|
|
return
|
|
}
|
|
g.dbmu.Lock()
|
|
if _, exists := g.db[id]; exists {
|
|
log.Printf("duplicate id %q", id)
|
|
}
|
|
g.db[id] = c
|
|
g.dbmu.Unlock()
|
|
}
|
|
|
|
// UnregisterComponent tells the game the component is no more.
|
|
// Note this does not remove any references held by other components.
|
|
func (g *Game) UnregisterComponent(c interface{}) {
|
|
i, ok := c.(Identifier)
|
|
if !ok {
|
|
return
|
|
}
|
|
id := i.Ident()
|
|
if id == "" {
|
|
return
|
|
}
|
|
g.dbmu.Lock()
|
|
delete(g.db, id)
|
|
g.dbmu.Unlock()
|
|
}
|
|
|
|
// Component returns the component with a given ID, or nil if there is none.
|
|
func (g *Game) Component(id string) interface{} {
|
|
g.dbmu.RLock()
|
|
defer g.dbmu.RUnlock()
|
|
return g.db[id]
|
|
}
|
|
|
|
// Scan implements Scanner.
|
|
func (g *Game) Scan() []interface{} { return []interface{}{g.Root} }
|
|
|
|
// Walk calls v with every component reachable from c via Scan, recursively,
|
|
// for as long as visit returns nil.
|
|
func Walk(c interface{}, v func(interface{}) error) error {
|
|
if err := v(c); err != nil {
|
|
return err
|
|
}
|
|
sc, ok := c.(Scanner)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
for _, c := range sc.Scan() {
|
|
if err := Walk(c, v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Load calls Load on all Loaders reachable via Scan (using Walk).
|
|
// It stops on the first error.
|
|
func (g *Game) Load(assets fs.FS) error {
|
|
return Walk(g.Root, func(c interface{}) error {
|
|
l, ok := c.(Loader)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return l.Load(assets)
|
|
})
|
|
}
|
|
|
|
// Prepare builds the component database (using Walk) and then calls
|
|
// Prepare on every Preparer. You must call Prepare before any calls
|
|
// to Component. You may call Prepare again (e.g. as an alternative to
|
|
// fastidiously calling RegisterComponent/UnregisterComponent).
|
|
func (g *Game) Prepare() {
|
|
g.dbmu.Lock()
|
|
g.db = make(map[string]interface{})
|
|
g.dbmu.Unlock()
|
|
// Moment in time where db is empty... whatev.
|
|
Walk(g.Root, func(c interface{}) error {
|
|
g.RegisterComponent(c)
|
|
return nil
|
|
})
|
|
Walk(g.Root, func(c interface{}) error {
|
|
if p, ok := c.(Prepper); ok {
|
|
p.Prepare(g)
|
|
}
|
|
return nil
|
|
})
|
|
}
|