2021-07-30 14:17:40 +10:00
|
|
|
package engine
|
|
|
|
|
|
|
|
import (
|
2021-07-30 17:26:23 +10:00
|
|
|
"encoding/gob"
|
2021-07-30 16:31:08 +10:00
|
|
|
"math"
|
2021-07-30 14:17:40 +10:00
|
|
|
"sort"
|
|
|
|
|
|
|
|
"github.com/hajimehoshi/ebiten/v2"
|
|
|
|
)
|
|
|
|
|
2021-07-30 17:26:23 +10:00
|
|
|
func init() {
|
|
|
|
gob.Register(Scene{})
|
|
|
|
}
|
|
|
|
|
2021-07-30 14:25:32 +10:00
|
|
|
// Scene manages drawing and updating a bunch of components.
|
|
|
|
type Scene struct {
|
2021-07-30 14:17:40 +10:00
|
|
|
Components []interface{}
|
2021-08-02 14:38:48 +10:00
|
|
|
Disabled bool
|
2021-08-01 17:08:26 +10:00
|
|
|
Hidden bool
|
|
|
|
ID
|
|
|
|
ZPos
|
2021-07-30 14:17:40 +10:00
|
|
|
}
|
|
|
|
|
2021-07-30 14:25:32 +10:00
|
|
|
// Draw draws all components in order.
|
2021-08-12 14:06:01 +10:00
|
|
|
func (s *Scene) Draw(screen *ebiten.Image, opts ebiten.DrawImageOptions) {
|
2021-08-01 17:08:26 +10:00
|
|
|
if s.Hidden {
|
|
|
|
return
|
|
|
|
}
|
2021-07-30 14:36:11 +10:00
|
|
|
for _, i := range s.Components {
|
2021-07-30 14:17:40 +10:00
|
|
|
if d, ok := i.(Drawer); ok {
|
2021-08-12 14:06:01 +10:00
|
|
|
d.Draw(screen, opts)
|
2021-07-30 14:17:40 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-05 15:14:56 +10:00
|
|
|
// Prepare does an initial Z-order sort.
|
|
|
|
func (s *Scene) Prepare(*Game) { s.sortByZ() }
|
|
|
|
|
2021-07-30 14:17:40 +10:00
|
|
|
// sortByZ sorts the components by Z position.
|
2021-08-05 15:14:56 +10:00
|
|
|
// Everything without a Z sorts first. Stable sort is used to avoid Z-fighting
|
|
|
|
// (among layers without a Z, or those with equal Z).
|
2021-07-30 14:36:11 +10:00
|
|
|
func (s *Scene) sortByZ() {
|
|
|
|
sort.SliceStable(s.Components, func(i, j int) bool {
|
|
|
|
a, aok := s.Components[i].(ZPositioner)
|
|
|
|
b, bok := s.Components[j].(ZPositioner)
|
2021-07-30 14:17:40 +10:00
|
|
|
if aok && bok {
|
|
|
|
return a.Z() < b.Z()
|
|
|
|
}
|
|
|
|
return !aok && bok
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-08-02 12:21:24 +10:00
|
|
|
// Scan returns all immediate subcomponents.
|
2021-08-02 12:16:10 +10:00
|
|
|
func (s *Scene) Scan() []interface{} { return s.Components }
|
2021-08-01 16:10:30 +10:00
|
|
|
|
2021-07-30 14:17:40 +10:00
|
|
|
// Update calls Update on all Updater components.
|
2021-07-30 14:36:11 +10:00
|
|
|
func (s *Scene) Update() error {
|
2021-08-02 14:38:48 +10:00
|
|
|
if s.Disabled {
|
|
|
|
return nil
|
|
|
|
}
|
2021-08-05 15:14:56 +10:00
|
|
|
|
2021-07-30 14:36:11 +10:00
|
|
|
for _, c := range s.Components {
|
2021-07-30 16:31:08 +10:00
|
|
|
// Update each updater in turn
|
2021-07-30 14:17:40 +10:00
|
|
|
if u, ok := c.(Updater); ok {
|
|
|
|
if err := u.Update(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-08-05 15:14:56 +10:00
|
|
|
// Check if the updates put the components out of order; if so, sort
|
|
|
|
curZ := -math.MaxFloat64 // fun fact: this is min float64
|
|
|
|
for _, c := range s.Components {
|
|
|
|
z, ok := c.(ZPositioner)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
t := z.Z()
|
|
|
|
if t < curZ {
|
|
|
|
s.sortByZ()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
curZ = t
|
2021-07-30 14:17:40 +10:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|