ichigo/engine/scene.go

76 lines
1.6 KiB
Go
Raw Normal View History

package engine
import (
"sort"
"github.com/hajimehoshi/ebiten/v2"
)
// Drawer is a component that can draw itself. Draw is called often.
type Drawer interface {
Draw(screen *ebiten.Image, geom ebiten.GeoM)
}
// Updater is a component that can update. Update is called repeatedly.
type Updater interface {
Update() error
}
// ZPositioner is used to reorder layers.
type ZPositioner interface {
Z() float64
}
2021-07-30 14:25:32 +10:00
// Scene manages drawing and updating a bunch of components.
type Scene struct {
Components []interface{}
2021-07-30 14:36:11 +10:00
Transform ebiten.GeoM
needsSort bool
}
2021-07-30 14:25:32 +10:00
// Draw draws all components in order.
2021-07-30 14:36:11 +10:00
func (s *Scene) Draw(screen *ebiten.Image, geom ebiten.GeoM) {
geom.Concat(s.Transform)
for _, i := range s.Components {
if d, ok := i.(Drawer); ok {
d.Draw(screen, geom)
}
}
}
// SetNeedsSort informs l that its layers may be out of order.
2021-07-30 14:36:11 +10:00
func (s *Scene) SetNeedsSort() {
s.needsSort = true
}
// sortByZ sorts the components by Z position.
2021-07-30 14:25:32 +10:00
// Stable sort is used to avoid Z-fighting among layers without a Z, or
// among those with equal Z. All non-ZPositioners are sorted first.
2021-07-30 14:36:11 +10:00
func (s *Scene) sortByZ() {
s.needsSort = false
sort.SliceStable(s.Components, func(i, j int) bool {
a, aok := s.Components[i].(ZPositioner)
b, bok := s.Components[j].(ZPositioner)
if aok && bok {
return a.Z() < b.Z()
}
return !aok && bok
})
}
// Update calls Update on all Updater components.
2021-07-30 14:36:11 +10:00
func (s *Scene) Update() error {
for _, c := range s.Components {
if u, ok := c.(Updater); ok {
if err := u.Update(); err != nil {
return err
}
}
}
2021-07-30 14:36:11 +10:00
if s.needsSort {
s.sortByZ()
}
return nil
}