1
1
mirror of https://github.com/OpenDiablo2/OpenDiablo2 synced 2024-07-01 03:15:23 +00:00
OpenDiablo2/d2core/engine.go

226 lines
7.3 KiB
Go
Raw Normal View History

2019-11-10 08:36:53 +00:00
package d2core
2019-10-24 13:31:59 +00:00
import (
"log"
2019-10-26 00:28:14 +00:00
"math"
2019-11-09 06:13:49 +00:00
"runtime"
"strconv"
"time"
2019-10-24 13:31:59 +00:00
"github.com/OpenDiablo2/D2Shared/d2data/d2dc6"
"github.com/OpenDiablo2/D2Shared/d2common/d2resource"
2019-11-10 13:51:02 +00:00
"github.com/OpenDiablo2/D2Shared/d2helper"
"github.com/OpenDiablo2/OpenDiablo2/d2corecommon/d2coreinterface"
"github.com/OpenDiablo2/D2Shared/d2common/d2enum"
2019-10-26 00:28:14 +00:00
2019-11-10 08:36:53 +00:00
"github.com/OpenDiablo2/OpenDiablo2/d2render"
"github.com/OpenDiablo2/D2Shared/d2data"
2019-11-10 08:36:53 +00:00
"github.com/OpenDiablo2/D2Shared/d2data/d2datadict"
2019-11-10 08:36:53 +00:00
"github.com/OpenDiablo2/OpenDiablo2/d2audio"
"github.com/OpenDiablo2/D2Shared/d2common"
"github.com/OpenDiablo2/OpenDiablo2/d2corecommon"
2019-11-10 13:51:02 +00:00
"github.com/OpenDiablo2/OpenDiablo2/d2render/d2ui"
2019-10-24 13:31:59 +00:00
"github.com/hajimehoshi/ebiten"
"github.com/hajimehoshi/ebiten/ebitenutil"
"github.com/hajimehoshi/ebiten/inpututil"
2019-10-24 13:31:59 +00:00
)
// Engine is the core OpenDiablo2 engine
2019-10-24 13:31:59 +00:00
type Engine struct {
Settings *d2corecommon.Configuration // Engine configuration settings from json file
CheckedPatch map[string]bool // First time we check a file, we'll check if it's in the patch. This notes that we've already checked that.
LoadingSprite d2render.Sprite // The sprite shown when loading stuff
loadingProgress float64 // LoadingProcess is a range between 0.0 and 1.0. If set, loading screen displays.
loadingIndex int // Determines which load function is currently being called
thingsToLoad []func() // The load functions for the next scene
stepLoadingSize float64 // The size for each loading step
CurrentScene d2coreinterface.Scene // The current scene being rendered
UIManager *d2ui.Manager // The UI manager
SoundManager *d2audio.Manager // The sound manager
nextScene d2coreinterface.Scene // The next scene to be loaded at the end of the game loop
fullscreenKey bool // When true, the fullscreen toggle is still being pressed
lastTime float64 // Last time we updated the scene
showFPS bool
assetManager *assetManager
2019-10-24 13:31:59 +00:00
}
// CreateEngine creates and instance of the OpenDiablo2 engine
2019-11-10 15:44:13 +00:00
func CreateEngine() Engine {
var result Engine
2019-10-24 13:31:59 +00:00
result.loadConfigurationFile()
result.assetManager = createAssetManager(result.Settings)
2019-11-10 13:51:02 +00:00
d2resource.LanguageCode = result.Settings.Language
d2datadict.LoadPalettes(nil, &result)
2019-11-10 15:44:13 +00:00
d2common.LoadTextDictionary(&result)
d2datadict.LoadLevelTypes(&result)
d2datadict.LoadLevelPresets(&result)
d2datadict.LoadLevelWarps(&result)
d2datadict.LoadObjectTypes(&result)
d2datadict.LoadObjects(&result)
d2datadict.LoadWeapons(&result)
d2datadict.LoadArmors(&result)
d2datadict.LoadMiscItems(&result)
2019-11-10 15:44:13 +00:00
d2datadict.LoadUniqueItems(&result)
d2datadict.LoadMissiles(&result)
d2datadict.LoadSounds(&result)
d2data.LoadAnimationData(&result)
d2datadict.LoadMonStats(&result)
2019-11-15 03:20:01 +00:00
LoadHeroObjects()
2019-11-10 15:44:13 +00:00
result.SoundManager = d2audio.CreateManager(&result)
2019-10-27 00:09:33 +00:00
result.SoundManager.SetVolumes(result.Settings.BgmVolume, result.Settings.SfxVolume)
2019-11-10 15:44:13 +00:00
result.UIManager = d2ui.CreateManager(&result, *result.SoundManager)
2019-11-10 13:51:02 +00:00
result.LoadingSprite = result.LoadSprite(d2resource.LoadingScreen, d2enum.Loading)
2019-10-24 13:31:59 +00:00
loadingSpriteSizeX, loadingSpriteSizeY := result.LoadingSprite.GetSize()
result.LoadingSprite.MoveTo(int(400-(loadingSpriteSizeX/2)), int(300+(loadingSpriteSizeY/2)))
return result
}
func (v *Engine) loadConfigurationFile() {
log.Println("Loading configuration file")
v.Settings = d2corecommon.LoadConfiguration()
2019-10-24 13:31:59 +00:00
}
func (v *Engine) LoadFile(fileName string) []byte {
return v.assetManager.LoadFile(fileName)
}
2019-10-24 13:31:59 +00:00
// IsLoading returns true if the engine is currently in a loading state
2019-11-10 15:44:13 +00:00
func (v Engine) IsLoading() bool {
2019-10-26 00:28:14 +00:00
return v.loadingProgress < 1.0
2019-10-24 13:31:59 +00:00
}
// LoadSprite loads a sprite from the game's data files
2019-11-10 15:44:13 +00:00
func (v Engine) LoadSprite(fileName string, palette d2enum.PaletteType) d2render.Sprite {
dc6, _ := d2dc6.LoadDC6(v.LoadFile(fileName), d2datadict.Palettes[palette])
sprite := d2render.CreateSpriteFromDC6(dc6)
2019-10-24 13:31:59 +00:00
return sprite
}
// updateScene handles the scene maintenance for the engine
func (v *Engine) updateScene() {
if v.nextScene == nil {
if v.thingsToLoad != nil {
if v.loadingIndex < len(v.thingsToLoad) {
v.thingsToLoad[v.loadingIndex]()
v.loadingIndex++
if v.loadingIndex < len(v.thingsToLoad) {
v.StepLoading()
} else {
v.FinishLoading()
v.thingsToLoad = nil
}
return
}
}
2019-10-24 13:31:59 +00:00
return
}
if v.CurrentScene != nil {
v.CurrentScene.Unload()
2019-11-09 06:13:49 +00:00
runtime.GC()
2019-10-24 13:31:59 +00:00
}
v.CurrentScene = v.nextScene
v.nextScene = nil
2019-10-25 23:12:42 +00:00
v.UIManager.Reset()
v.thingsToLoad = v.CurrentScene.Load()
v.loadingIndex = 0
v.SetLoadingStepSize(1.0 / float64(len(v.thingsToLoad)))
2019-10-26 00:28:14 +00:00
v.ResetLoading()
2019-10-25 19:06:47 +00:00
}
2019-10-24 13:31:59 +00:00
// Update updates the internal state of the engine
func (v *Engine) Update() {
2019-10-25 21:15:44 +00:00
if ebiten.IsKeyPressed(ebiten.KeyAlt) && ebiten.IsKeyPressed(ebiten.KeyEnter) {
if !v.fullscreenKey {
ebiten.SetFullscreen(!ebiten.IsFullscreen())
}
v.fullscreenKey = true
} else {
v.fullscreenKey = false
}
if inpututil.IsKeyJustPressed(ebiten.KeyF6) {
v.showFPS = !v.showFPS
}
if inpututil.IsKeyJustPressed(ebiten.KeyF8) {
ebiten.SetVsyncEnabled(!ebiten.IsVsyncEnabled())
}
2019-10-24 13:31:59 +00:00
v.updateScene()
if v.CurrentScene == nil {
2019-10-25 19:06:47 +00:00
log.Fatal("no scene loaded")
}
2019-10-25 23:12:42 +00:00
if v.IsLoading() {
return
}
2019-10-26 00:28:14 +00:00
currentTime := float64(time.Now().UnixNano()) / float64(time.Second)
deltaTime := math.Min((currentTime - v.lastTime), 0.1)
v.lastTime = currentTime
v.CurrentScene.Update(deltaTime)
2019-10-25 23:12:42 +00:00
v.UIManager.Update()
2019-10-24 13:31:59 +00:00
}
// Draw draws the game
2019-11-10 15:44:13 +00:00
func (v Engine) Draw(screen *ebiten.Image) {
2019-10-26 00:28:14 +00:00
if v.loadingProgress < 1.0 {
2019-11-13 05:31:52 +00:00
v.LoadingSprite.Frame = int16(d2helper.Max(0, d2helper.Min(uint32(len(v.LoadingSprite.Frames)-1), uint32(float64(len(v.LoadingSprite.Frames)-1)*v.loadingProgress))))
2019-10-24 13:31:59 +00:00
v.LoadingSprite.Draw(screen)
} else {
if v.CurrentScene == nil {
2019-10-25 19:06:47 +00:00
log.Fatal("no scene loaded")
2019-10-24 13:31:59 +00:00
}
v.CurrentScene.Render(screen)
2019-10-25 23:12:42 +00:00
v.UIManager.Draw(screen)
2019-10-24 13:31:59 +00:00
}
if v.showFPS {
ebitenutil.DebugPrintAt(screen, "vsync:"+strconv.FormatBool(ebiten.IsVsyncEnabled())+"\nFPS:"+strconv.Itoa(int(ebiten.CurrentFPS())), 5, 565)
var m runtime.MemStats
runtime.ReadMemStats(&m)
ebitenutil.DebugPrintAt(screen, "Alloc "+strconv.FormatInt(int64(m.Alloc)/1024/1024, 10), 680, 0)
ebitenutil.DebugPrintAt(screen, "Pause "+strconv.FormatInt(int64(m.PauseTotalNs/1024/1024), 10), 680, 10)
ebitenutil.DebugPrintAt(screen, "HeapSys "+strconv.FormatInt(int64(m.HeapSys/1024/1024), 10), 680, 20)
ebitenutil.DebugPrintAt(screen, "NumGC "+strconv.FormatInt(int64(m.NumGC), 10), 680, 30)
cx, cy := ebiten.CursorPosition()
ebitenutil.DebugPrintAt(screen, "Coords "+strconv.FormatInt(int64(cx), 10) + ","+strconv.FormatInt(int64(cy), 10), 680, 40)
}
2019-10-24 13:31:59 +00:00
}
// SetNextScene tells the engine what scene to load on the next update cycle
func (v *Engine) SetNextScene(nextScene d2coreinterface.Scene) {
2019-10-24 13:31:59 +00:00
v.nextScene = nextScene
}
2019-10-26 00:28:14 +00:00
// SetLoadingStepSize sets the size of the loading step
func (v *Engine) SetLoadingStepSize(size float64) {
v.stepLoadingSize = size
}
2019-10-25 19:06:47 +00:00
2019-10-26 00:28:14 +00:00
// ResetLoading resets the loading progress
func (v *Engine) ResetLoading() {
v.loadingProgress = 0.0
}
// StepLoading increments the loading progress
func (v *Engine) StepLoading() {
v.loadingProgress = math.Min(1.0, v.loadingProgress+v.stepLoadingSize)
}
// FinishLoading terminates the loading phase
func (v *Engine) FinishLoading() {
v.loadingProgress = 1.0
2019-10-25 19:06:47 +00:00
}