gg/engine.go
2023-12-23 00:09:07 +03:00

251 lines
4.4 KiB
Go

package gg
import (
"github.com/hajimehoshi/ebiten/v2"
"github.com/hajimehoshi/ebiten/v2/inpututil"
"github.com/di4f/gods/maps"
//"fmt"
"time"
)
// The type represents order of drawing.
// Higher values are drawn later.
type Layer float64
func (l Layer) GetLayer() Layer {
return l
}
// Window configuration type.
type WindowConfig struct {
// The title of the window.
Title string
// Width and height of the window
// in pixels.
Width,
Height int
// Optional settings with
// self describing names.
FixedSize,
Fullscreen,
VSync bool
}
// The main structure that represents current state of [game] engine.
type Engine struct {
wcfg *WindowConfig
// The main holder for objects.
// Uses the map structure to quickly
// delete and create new objects.
Objects maps.Map[Object, struct{}]
// The main camera to display in window.
// If is set to nil then the engine will panic.
Camera *Camera
// The same delta time for all frames
// and all objects.
lastTime time.Time
dt Float
// Temporary stuff
keys, prevKeys []Key
cursorPos, prevCursorPos Vector
mouseButtons, prevMouseButtons []MouseButton
outerEvents, handleEvents EventChan
}
type engine Engine
// Get currently pressed keys.
func (e *Engine) Keys() []Key {
return e.keys
}
// Returns new empty Engine.
func NewEngine(
cfg *WindowConfig,
) *Engine {
/*w := Float(cfg.Width)
h := Float(cfg.Height)*/
ret := &Engine{}
ret.wcfg = cfg
ret.Camera = ret.NewCamera()
ret.outerEvents = make(EventChan)
ret.handleEvents = make(EventChan)
ret.Objects = maps.NewOrdered[Object, struct{}]()
return ret
}
// Get the real window size in the current context.
func (c *Engine) RealWinSize() Vector {
return V(
Float(c.wcfg.Width),
Float(c.wcfg.Height),
)
}
func (c *Engine) AbsWinSize() Vector {
return c.RealWinSize().Div(c.Camera.Scale)
}
func (e *Engine) EventInput() EventChan {
return e.outerEvents
}
// Add new object considering what
// interfaces it implements.
func (e *Engine) Add(b any) error {
object, _ := b.(Object)
if e.Objects.Has(object) {
return ObjectExistErr
}
/*o, ok := e.makeObject(b)
if !ok {
return ObjectNotImplementedErr
}*/
starter, ok := b.(Starter)
if ok {
starter.Start(&Context{
Engine: e,
})
}
e.Objects.Set(object, struct{}{})
return nil
}
// Delete object from Engine.
func (e *Engine) Del(b any) error {
object, _ := b.(Object)
if !e.Objects.Has(object) {
return ObjectNotExistErr
}
deleter, ok := b.(Deleter)
if ok {
deleter.Delete(&Context{
Engine: e,
})
}
e.Objects.Del(b)
return nil
}
func (e *engine) Update() error {
eng := (*Engine)(e)
e.prevKeys = e.keys
e.keys = inpututil.
AppendPressedKeys(e.keys[:0])
events := []any{}
diff := keyDiff(e.prevKeys, e.keys)
for _, key := range diff {
var event any
if eng.IsPressed(key) {
event = &KeyDown{
Key: key,
}
} else {
event = &KeyUp{
Key: key,
}
}
events = append(events, event)
}
e.dt = time.Since(e.lastTime).Seconds()
for object := range e.Objects.KeyChan() {
eventer, ok := object.(Eventer)
if ok {
for _, event := range events {
eventer.Event(&Context{
Engine: eng,
Event: event,
})
}
}
updater, ok := object.(Updater)
if !ok {
continue
}
updater.Update(&Context{
Engine: eng,
})
}
e.lastTime = time.Now()
return nil
}
func (e *engine) Draw(i *ebiten.Image) {
eng := (*Engine)(e)
m := map[Layer][]Drawer{}
for object := range eng.Objects.KeyChan() {
drawer, ok := object.(Drawer)
if !ok {
continue
}
l := drawer.GetLayer()
layer, ok := m[l]
// Create new if has no the layer
if !ok {
m[l] = []Drawer{drawer}
continue
}
m[l] = append(layer, drawer)
}
// Drawing layers.
layers := maps.NewSparse[Layer, []Drawer](nil, m)
for layer := range layers.Chan() {
for _, drawer := range layer {
drawer.Draw(&Context{
Engine: eng,
Image: i,
})
}
}
// Empty the buff to generate it again.
eng.Camera.buf = nil
}
func (e *engine) Layout(ow, oh int) (int, int) {
if e.wcfg.FixedSize {
return e.wcfg.Width, e.wcfg.Height
}
return ow, oh
}
// Return the delta time duration value.
func (e *Engine) DT() Float {
return e.dt
}
func (e *Engine) Run() error {
ebiten.SetWindowTitle(e.wcfg.Title)
ebiten.SetWindowSize(e.wcfg.Width, e.wcfg.Height)
ebiten.SetWindowSizeLimits(1, 1, e.wcfg.Width, e.wcfg.Height)
ebiten.SetVsyncEnabled(e.wcfg.VSync)
e.lastTime = time.Now()
//fmt.Println(e.Objects)
return ebiten.RunGame((*engine)(e))
}