openquell/game/game.go

127 lines
3.2 KiB
Go
Raw Normal View History

2024-02-06 15:26:20 +01:00
package game
import (
"fmt"
2024-02-06 15:26:20 +01:00
"image"
"log/slog"
"openquell/config"
"openquell/observers"
2024-02-06 15:26:20 +01:00
"github.com/hajimehoshi/ebiten/v2"
"github.com/mlange-42/arche/ecs"
)
type Game struct {
World *ecs.World
Bounds image.Rectangle
ScreenWidth, ScreenHeight, Cellsize int
Scenes map[SceneName]Scene
CurrentScene SceneName
Observer *observers.GameObserver
2024-02-28 13:15:45 +01:00
//Levels []*Level // fed in LevelScene.GenerateLevels()
Config *config.Config
2024-02-06 15:26:20 +01:00
}
func NewGame(width, height, cellsize int, cfg *config.Config, startscene SceneName) *Game {
2024-02-06 15:26:20 +01:00
world := ecs.NewWorld()
game := &Game{
Bounds: image.Rectangle{},
World: &world,
ScreenWidth: width,
ScreenHeight: height,
Scenes: map[SceneName]Scene{},
Cellsize: cellsize,
Config: cfg,
2024-02-06 15:26:20 +01:00
}
game.Scenes[Welcome] = NewWelcomeScene(game)
2024-02-18 18:19:34 +01:00
game.Scenes[Menu] = NewMenuScene(game)
game.Scenes[About] = NewAboutScene(game)
2024-02-18 18:19:34 +01:00
game.Scenes[Popup] = NewPopupScene(game)
game.Scenes[Play] = NewLevelScene(game, cfg.Startlevel)
2024-02-18 18:19:34 +01:00
game.Scenes[Select] = NewSelectScene(game)
game.CurrentScene = startscene
2024-02-28 13:15:45 +01:00
game.Observer = observers.NewGameObserver(
&world, cfg.Startlevel, len(game.Levels), width, height, cellsize)
fmt.Println(game.World.Stats().String())
2024-02-06 15:26:20 +01:00
return game
}
func (game *Game) GetCurrentScene() Scene {
return game.Scenes[game.CurrentScene]
}
2024-02-06 15:26:20 +01:00
func (game *Game) Update() error {
gameobserver := observers.GetGameObserver(game.World)
2024-02-18 18:19:34 +01:00
// handle level ends
timer := gameobserver.StopTimer
if timer.IsReady() {
// a level is either lost or won, we display a small popup
// asking the user how to continue from here
timer.Reset()
slog.Debug("timer ready", "lost", gameobserver.Lost, "retry", gameobserver.Retry)
if !gameobserver.Lost {
2024-02-28 13:15:45 +01:00
level := // current level?
score := ((level.MinMoves * 100) / gameobserver.Moves) / 30
gameobserver.AddScore()
}
game.Scenes[Nextlevel] = NewNextlevelScene(game, gameobserver.Lost)
game.CurrentScene = Nextlevel
}
scene := game.GetCurrentScene()
scene.Update()
2024-02-18 18:19:34 +01:00
if scene.Clearscreen() {
ebiten.SetScreenClearedEveryFrame(true)
} else {
ebiten.SetScreenClearedEveryFrame(false)
}
next := scene.GetNext()
if next != game.CurrentScene {
if next == Play && game.CurrentScene == Nextlevel {
// switched from nextlevel (lost or won) popup to play (either retry or next level)
if !gameobserver.Retry {
gameobserver.CurrentLevel++
}
gameobserver.Retry = false
}
if next == Play {
// fresh setup of actual level every time we enter the play scene
game.Scenes[Play].SetLevel(gameobserver.CurrentLevel)
}
// make sure we stay on the selected scene
scene.ResetNext()
// finally switch
game.CurrentScene = next
// FIXME: add some reset function to gameobserver for these kinds of things
gameobserver.Lost = false
2024-02-06 15:26:20 +01:00
}
timer.Update()
2024-02-06 15:26:20 +01:00
return nil
}
func (game *Game) Draw(screen *ebiten.Image) {
game.GetCurrentScene().Draw(screen)
2024-02-06 15:26:20 +01:00
}
func (g *Game) Layout(newWidth, newHeight int) (int, int) {
return g.ScreenWidth, g.ScreenHeight
}