mirror of
https://github.com/OpenDiablo2/OpenDiablo2
synced 2024-10-31 16:27:18 -04:00
bdfdeda67b
* WIP: Add support for missiles Break AnimatedEntity into two parts to support single and composite animations. Summon misssiles on right click. * Break animated entity down further Move npc only logic to npc struct, reduce duplication in map entities * Change a bunch of int32s to int
191 lines
5.2 KiB
Go
191 lines
5.2 KiB
Go
package d2map
|
|
|
|
import (
|
|
"github.com/beefsack/go-astar"
|
|
"math"
|
|
"strings"
|
|
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2common/d2enum"
|
|
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2core/d2audio"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2core/d2gamestate"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2core/d2render"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2core/d2term"
|
|
)
|
|
|
|
type MapEntity interface {
|
|
Render(target d2render.Surface)
|
|
Advance(tickTime float64)
|
|
GetPosition() (float64, float64)
|
|
}
|
|
|
|
type MapEngine struct {
|
|
gameState *d2gamestate.GameState
|
|
|
|
debugVisLevel int
|
|
|
|
regions []*MapRegion
|
|
entities []MapEntity
|
|
viewport *Viewport
|
|
camera Camera
|
|
}
|
|
|
|
func CreateMapEngine(gameState *d2gamestate.GameState) *MapEngine {
|
|
engine := &MapEngine{
|
|
gameState: gameState,
|
|
viewport: NewViewport(0, 0, 800, 600),
|
|
}
|
|
|
|
d2term.BindAction("mapdebugvis", "set map debug visualization level", func(level int) {
|
|
engine.debugVisLevel = level
|
|
})
|
|
|
|
engine.viewport.SetCamera(&engine.camera)
|
|
return engine
|
|
}
|
|
|
|
func (m *MapEngine) GetStartPosition() (float64, float64) {
|
|
var startX, startY float64
|
|
if len(m.regions) > 0 {
|
|
region := m.regions[0]
|
|
startX, startY = region.getStartTilePosition()
|
|
}
|
|
|
|
return startX, startY
|
|
}
|
|
|
|
func (m *MapEngine) GetCenterPosition() (float64, float64) {
|
|
var centerX, centerY float64
|
|
if len(m.regions) > 0 {
|
|
region := m.regions[0]
|
|
centerX = float64(region.tileRect.Left) + float64(region.tileRect.Width)/2
|
|
centerY = float64(region.tileRect.Top) + float64(region.tileRect.Height)/2
|
|
}
|
|
|
|
return centerX, centerY
|
|
}
|
|
|
|
func (m *MapEngine) MoveCameraTo(x, y float64) {
|
|
m.camera.MoveTo(x, y)
|
|
}
|
|
|
|
func (m *MapEngine) MoveCameraBy(x, y float64) {
|
|
m.camera.MoveBy(x, y)
|
|
}
|
|
|
|
func (m *MapEngine) ScreenToWorld(x, y int) (float64, float64) {
|
|
return m.viewport.ScreenToWorld(x, y)
|
|
}
|
|
|
|
func (m *MapEngine) ScreenToOrtho(x, y int) (float64, float64) {
|
|
return m.viewport.ScreenToOrtho(x, y)
|
|
}
|
|
|
|
func (m *MapEngine) WorldToOrtho(x, y float64) (float64, float64) {
|
|
return m.viewport.WorldToOrtho(x, y)
|
|
}
|
|
|
|
func (m *MapEngine) GenerateMap(regionType d2enum.RegionIdType, levelPreset int, fileIndex int) {
|
|
region, entities := loadRegion(m.gameState.Seed, 0, 0, regionType, levelPreset, fileIndex)
|
|
m.regions = append(m.regions, region)
|
|
m.entities = append(m.entities, entities...)
|
|
}
|
|
|
|
func (m *MapEngine) GenerateAct1Overworld() {
|
|
d2audio.PlayBGM("/data/global/music/Act1/town1.wav") // TODO: Temp stuff here
|
|
|
|
region, entities := loadRegion(m.gameState.Seed, 0, 0, d2enum.RegionAct1Town, 1, -1)
|
|
m.regions = append(m.regions, region)
|
|
m.entities = append(m.entities, entities...)
|
|
|
|
if strings.Contains(region.regionPath, "E1") {
|
|
region, entities := loadRegion(m.gameState.Seed, region.tileRect.Width-1, 0, d2enum.RegionAct1Town, 2, -1)
|
|
m.AppendRegion(region)
|
|
m.entities = append(m.entities, entities...)
|
|
} else if strings.Contains(region.regionPath, "S1") {
|
|
region, entities := loadRegion(m.gameState.Seed, 0, region.tileRect.Height-1, d2enum.RegionAct1Town, 3, -1)
|
|
m.AppendRegion(region)
|
|
m.entities = append(m.entities, entities...)
|
|
}
|
|
}
|
|
|
|
func (m *MapEngine) AppendRegion(region *MapRegion) {
|
|
// TODO: Stitch together region.walkableArea
|
|
m.regions = append(m.regions, region)
|
|
}
|
|
|
|
func (m *MapEngine) GetRegionAtTile(x, y int) *MapRegion {
|
|
for _, region := range m.regions {
|
|
if region.tileRect.IsInRect(x, y) {
|
|
return region
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (m *MapEngine) AddEntity(entity MapEntity) {
|
|
m.entities = append(m.entities, entity)
|
|
}
|
|
|
|
func (m *MapEngine) RemoveEntity(entity MapEntity) {
|
|
if entity == nil {
|
|
return
|
|
}
|
|
|
|
// In-place filter to remove the given entity.
|
|
n := 0
|
|
for _, check := range m.entities {
|
|
if check != entity {
|
|
m.entities[n] = check
|
|
n++
|
|
}
|
|
}
|
|
m.entities = m.entities[:n]
|
|
}
|
|
|
|
func (m *MapEngine) Advance(tickTime float64) {
|
|
for _, region := range m.regions {
|
|
if region.isVisbile(m.viewport) {
|
|
region.advance(tickTime)
|
|
}
|
|
}
|
|
|
|
for _, entity := range m.entities {
|
|
entity.Advance(tickTime)
|
|
}
|
|
}
|
|
|
|
func (m *MapEngine) Render(target d2render.Surface) {
|
|
for _, region := range m.regions {
|
|
if region.isVisbile(m.viewport) {
|
|
region.renderPass1(m.viewport, target)
|
|
region.renderDebug(m.debugVisLevel, m.viewport, target)
|
|
region.renderPass2(m.entities, m.viewport, target)
|
|
region.renderPass3(m.viewport, target)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *MapEngine) PathFind(startX, startY, endX, endY float64) (path []astar.Pather, distance float64, found bool) {
|
|
startTileX := int(math.Floor(startX))
|
|
startTileY := int(math.Floor(startY))
|
|
startSubtileX := int((startX - float64(int(startX))) * 5)
|
|
startSubtileY := int((startY - float64(int(startY))) * 5)
|
|
startRegion := m.GetRegionAtTile(startTileX, startTileY)
|
|
startNode := &startRegion.walkableArea[startSubtileY+((startTileY-startRegion.tileRect.Top)*5)][startSubtileX+((startTileX-startRegion.tileRect.Left)*5)]
|
|
|
|
endTileX := int(math.Floor(endX))
|
|
endTileY := int(math.Floor(endY))
|
|
endSubtileX := int((endX - float64(int(endX))) * 5)
|
|
endSubtileY := int((endY - float64(int(endY))) * 5)
|
|
endRegion := m.GetRegionAtTile(endTileX, endTileY)
|
|
endNode := &endRegion.walkableArea[endSubtileY+((endTileY-endRegion.tileRect.Top)*5)][endSubtileX+((endTileX-endRegion.tileRect.Left)*5)]
|
|
|
|
path, distance, found = astar.Path(endNode, startNode)
|
|
if path != nil {
|
|
path = path[1:]
|
|
}
|
|
return
|
|
}
|