2022-10-14 18:02:03 +08:00
package main
import (
2022-10-15 16:51:38 +08:00
. "dnmshared"
2022-10-14 18:02:03 +08:00
"fmt"
"github.com/hajimehoshi/ebiten/v2"
"github.com/solarlune/resolv"
2022-10-15 16:51:38 +08:00
"go.uber.org/zap"
"image/color"
2022-10-14 18:02:03 +08:00
)
type WorldColliderDisplay struct {
2022-10-15 16:51:38 +08:00
Game * Game
Space * resolv . Space
2022-10-14 18:02:03 +08:00
}
func ( world * WorldColliderDisplay ) Init ( ) {
}
func NewWorldColliderDisplay ( game * Game , stageDiscreteW , stageDiscreteH , stageTileW , stageTileH int32 , playerPosMap StrToVec2DListMap , barrierMap StrToPolygon2DListMap ) * WorldColliderDisplay {
2022-10-14 18:47:09 +08:00
2022-10-15 16:51:38 +08:00
playerList := * ( playerPosMap [ "PlayerStartingPos" ] )
2022-10-14 18:47:09 +08:00
barrierList := * ( barrierMap [ "Barrier" ] )
world := & WorldColliderDisplay { Game : game }
Logger . Info ( "Parsed variables" , zap . Any ( "stageDiscreteW" , stageDiscreteW ) , zap . Any ( "stageDiscreteH" , stageDiscreteH ) , zap . Any ( "stageTileW" , stageTileW ) , zap . Any ( "stageTileH" , stageTileH ) )
2022-10-14 18:02:03 +08:00
spaceW := stageDiscreteW * stageTileW
spaceH := stageDiscreteH * stageTileH
spaceOffsetX := float64 ( spaceW ) * 0.5
spaceOffsetY := float64 ( spaceH ) * 0.5
2022-10-19 08:32:20 +08:00
playerColliderRadius := float64 ( 32 )
2022-10-22 13:38:10 +08:00
playerColliders := make ( [ ] * resolv . Object , len ( playerList ) )
2022-10-15 16:51:38 +08:00
space := resolv . NewSpace ( int ( spaceW ) , int ( spaceH ) , 16 , 16 )
2022-10-19 08:32:20 +08:00
for i , player := range playerList {
2022-10-19 15:10:11 +08:00
playerCollider := GenerateRectCollider ( player . X , player . Y , playerColliderRadius * 2 , playerColliderRadius * 2 , spaceOffsetX , spaceOffsetY , "Player" ) // [WARNING] Deliberately not using a circle because "resolv v0.5.1" doesn't yet align circle center with space cell center, regardless of the "specified within-object offset"
2022-10-22 13:38:10 +08:00
Logger . Info ( fmt . Sprintf ( "Player Collider#%d: player.X=%v, player.Y=%v, radius=%v, spaceOffsetX=%v, spaceOffsetY=%v, shape=%v; calibrationCheckX=player.X-radius+spaceOffsetX=%v" , i , player . X , player . Y , playerColliderRadius , spaceOffsetX , spaceOffsetY , playerCollider . Shape , player . X - playerColliderRadius + spaceOffsetX ) )
playerColliders [ i ] = playerCollider
2022-10-15 16:51:38 +08:00
space . Add ( playerCollider )
}
barrierLocalId := 0
2022-10-15 21:39:22 +08:00
for _ , barrierUnaligned := range barrierList {
2022-10-19 15:10:11 +08:00
barrierCollider := GenerateConvexPolygonCollider ( barrierUnaligned , spaceOffsetX , spaceOffsetY , "Barrier" )
2022-10-22 13:38:10 +08:00
Logger . Info ( fmt . Sprintf ( "Added barrier: shape=%v" , barrierCollider . Shape ) )
2022-10-19 15:10:11 +08:00
space . Add ( barrierCollider )
2022-10-15 16:51:38 +08:00
barrierLocalId ++
}
2022-10-14 18:02:03 +08:00
2022-10-14 18:47:09 +08:00
world . Space = space
2022-10-19 08:32:20 +08:00
2022-10-22 13:38:10 +08:00
moveToCollide := true
if moveToCollide {
toTestPlayerCollider := playerColliders [ 0 ]
oldDx , oldDy := - 2.98 , - 50.0
dx , dy := oldDx , oldDy
if collision := toTestPlayerCollider . Check ( oldDx , oldDy , "Barrier" ) ; collision != nil {
playerShape := toTestPlayerCollider . Shape . ( * resolv . ConvexPolygon )
barrierShape := collision . Objects [ 0 ] . Shape . ( * resolv . ConvexPolygon )
if overlapped , pushbackX , pushbackY := CalcPushbacks ( oldDx , oldDy , playerShape , barrierShape ) ; overlapped {
Logger . Info ( fmt . Sprintf ( "Collided & overlapped: player.X=%v, player.Y=%v, oldDx=%v, oldDy=%v, playerShape=%v, toCheckBarrier=%v, pushbackX=%v, pushbackY=%v" , toTestPlayerCollider . X , toTestPlayerCollider . Y , oldDx , oldDy , ConvexPolygonStr ( playerShape ) , ConvexPolygonStr ( barrierShape ) , pushbackX , pushbackY ) )
dx -= pushbackX
dy -= pushbackY
} else {
Logger . Info ( fmt . Sprintf ( "Collider BUT not overlapped: player.X=%v, player.Y=%v, oldDx=%v, oldDy=%v, playerShape=%v, toCheckBarrier=%v" , toTestPlayerCollider . X , toTestPlayerCollider . Y , oldDx , oldDy , ConvexPolygonStr ( playerShape ) , ConvexPolygonStr ( barrierShape ) ) )
}
}
toTestPlayerCollider . X += dx
toTestPlayerCollider . Y += dy
toTestPlayerCollider . Update ( )
}
2022-10-19 08:32:20 +08:00
2022-10-15 16:51:38 +08:00
return world
2022-10-14 18:02:03 +08:00
}
func ( world * WorldColliderDisplay ) Update ( ) {
}
func ( world * WorldColliderDisplay ) Draw ( screen * ebiten . Image ) {
for _ , o := range world . Space . Objects ( ) {
if o . HasTags ( "Player" ) {
2022-10-15 16:51:38 +08:00
drawColor := color . RGBA { 0 , 255 , 0 , 255 }
2022-10-19 08:32:20 +08:00
DrawPolygon ( screen , o . Shape . ( * resolv . ConvexPolygon ) , drawColor )
2022-10-15 16:51:38 +08:00
} else {
drawColor := color . RGBA { 60 , 60 , 60 , 255 }
DrawPolygon ( screen , o . Shape . ( * resolv . ConvexPolygon ) , drawColor )
2022-10-14 18:02:03 +08:00
}
}
2022-10-15 21:39:22 +08:00
world . Game . DebugDraw ( screen , world . Space )
2022-10-14 18:02:03 +08:00
if world . Game . ShowHelpText {
world . Game . DrawText ( screen , 16 , 16 ,
"~ Collider Display test ~" ,
"F1: Toggle Debug View" ,
"F2: Show / Hide help text" ,
"R: Restart world" ,
fmt . Sprintf ( "%d FPS (frames per second)" , int ( ebiten . CurrentFPS ( ) ) ) ,
fmt . Sprintf ( "%d TPS (ticks per second)" , int ( ebiten . CurrentTPS ( ) ) ) ,
)
}
}