2022-10-14 18:02:03 +08:00
package main
import (
2022-11-25 08:21:03 +08:00
. "battle_srv/protos"
2022-10-15 16:51:38 +08:00
. "dnmshared"
2022-11-12 11:20:16 +08:00
. "dnmshared/sharedprotos"
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-11-09 14:20:26 +08:00
playerPosList := * ( 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-12-14 21:30:01 +08:00
worldToVirtualGridRatio := float64 ( 1000 )
2022-12-15 12:28:39 +08:00
virtualGridToWorldRatio := float64 ( 1 ) / worldToVirtualGridRatio
2022-12-14 21:30:01 +08:00
playerDefaultSpeed := 1 * worldToVirtualGridRatio
2022-12-16 13:55:43 +08:00
minStep := ( int ( float64 ( playerDefaultSpeed ) * virtualGridToWorldRatio ) << 3 )
2022-12-14 21:30:01 +08:00
playerColliderRadius := float64 ( 12 )
2022-11-09 14:20:26 +08:00
playerColliders := make ( [ ] * resolv . Object , len ( playerPosList . Eles ) )
2022-12-15 12:28:39 +08:00
snapIntoPlatformOverlap := float64 ( 0.1 )
2022-11-12 12:18:00 +08:00
space := resolv . NewSpace ( int ( spaceW ) , int ( spaceH ) , minStep , minStep )
2022-12-16 13:55:43 +08:00
topPadding , bottomPadding , leftPadding , rightPadding := snapIntoPlatformOverlap , snapIntoPlatformOverlap , snapIntoPlatformOverlap , snapIntoPlatformOverlap
2022-11-09 14:20:26 +08:00
for i , playerPos := range playerPosList . Eles {
2022-12-15 12:28:39 +08:00
colliderWidth , colliderHeight := playerColliderRadius * 2 , playerColliderRadius * 4
2022-12-16 13:55:43 +08:00
playerCollider := GenerateRectCollider ( playerPos . X , playerPos . Y , colliderWidth , colliderHeight , topPadding , bottomPadding , leftPadding , rightPadding , 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-12-15 10:03:48 +08:00
Logger . Info ( fmt . Sprintf ( "Player Collider#%d: player world pos=(%.2f, %.2f), shape=%v" , i , playerPos . X , playerPos . Y , ConvexPolygonStr ( playerCollider . Shape . ( * resolv . ConvexPolygon ) ) ) )
2022-10-22 13:38:10 +08:00
playerColliders [ i ] = playerCollider
2022-10-15 16:51:38 +08:00
space . Add ( playerCollider )
}
barrierLocalId := 0
2022-11-09 14:20:26 +08:00
for _ , barrierUnaligned := range barrierList . Eles {
2022-10-19 15:10:11 +08:00
barrierCollider := GenerateConvexPolygonCollider ( barrierUnaligned , spaceOffsetX , spaceOffsetY , "Barrier" )
2022-12-14 21:30:01 +08:00
Logger . Debug ( fmt . Sprintf ( "Added barrier: shape=%v" , ConvexPolygonStr ( barrierCollider . Shape . ( * resolv . ConvexPolygon ) ) ) )
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-12-16 13:55:43 +08:00
moveToCollide := true
2022-10-22 13:38:10 +08:00
if moveToCollide {
2022-11-12 11:20:16 +08:00
effPushback := Vec2D { X : float64 ( 0 ) , Y : float64 ( 0 ) }
2022-10-22 13:38:10 +08:00
toTestPlayerCollider := playerColliders [ 0 ]
2022-12-16 13:55:43 +08:00
//colliderWidth, colliderHeight := playerColliderRadius*2, playerColliderRadius*4
//newVx, newVy := int32(27999), int32(-420270)
//toTestPlayerCollider.X, toTestPlayerCollider.Y = VirtualGridToPolygonColliderBLPos(newVx, newVy, colliderWidth, colliderHeight, topPadding, bottomPadding, leftPadding, rightPadding, spaceOffsetX, spaceOffsetY, virtualGridToWorldRatio)
2022-11-12 22:53:35 +08:00
2022-12-14 21:30:01 +08:00
Logger . Info ( fmt . Sprintf ( "Checking collision for playerShape=%v" , ConvexPolygonStr ( toTestPlayerCollider . Shape . ( * resolv . ConvexPolygon ) ) ) )
2022-11-12 22:53:35 +08:00
2022-11-12 11:20:16 +08:00
toTestPlayerCollider . Update ( )
2022-11-12 20:34:38 +08:00
if collision := toTestPlayerCollider . Check ( 0 , 0 ) ; collision != nil {
2022-10-22 13:38:10 +08:00
playerShape := toTestPlayerCollider . Shape . ( * resolv . ConvexPolygon )
2022-11-12 11:20:16 +08:00
for _ , obj := range collision . Objects {
2022-12-14 21:30:01 +08:00
bShape := obj . Shape . ( * resolv . ConvexPolygon )
2022-12-15 12:28:39 +08:00
Logger . Warn ( fmt . Sprintf ( "Checking potential: a=%v, b=%v" , ConvexPolygonStr ( playerShape ) , ConvexPolygonStr ( bShape ) ) )
2022-12-14 21:30:01 +08:00
if overlapped , pushbackX , pushbackY , overlapResult := CalcPushbacks ( 0 , 0 , playerShape , bShape ) ; overlapped {
Logger . Warn ( fmt . Sprintf ( "Overlapped: a=%v, b=%v, pushbackX=%v, pushbackY=%v" , ConvexPolygonStr ( playerShape ) , ConvexPolygonStr ( bShape ) , pushbackX , pushbackY ) )
2022-11-12 11:20:16 +08:00
effPushback . X += pushbackX
effPushback . Y += pushbackY
} else {
2022-12-14 21:30:01 +08:00
Logger . Warn ( fmt . Sprintf ( "Collided BUT not overlapped: a=%v, b=%v, overlapResult=%v" , ConvexPolygonStr ( playerShape ) , ConvexPolygonStr ( bShape ) , overlapResult ) )
2022-11-12 11:20:16 +08:00
}
2022-10-22 13:38:10 +08:00
}
2022-11-12 11:20:16 +08:00
toTestPlayerCollider . X -= effPushback . X
toTestPlayerCollider . Y -= effPushback . Y
toTestPlayerCollider . Update ( )
Logger . Info ( fmt . Sprintf ( "effPushback={%v, %v}" , effPushback . X , effPushback . Y ) )
2022-10-22 13:38:10 +08:00
}
}
2022-11-25 08:21:03 +08:00
meleeBullet := & MeleeBullet {
// for offender
StartupFrames : int32 ( 18 ) ,
ActiveFrames : int32 ( 1 ) ,
RecoveryFrames : int32 ( 61 ) ,
RecoveryFramesOnBlock : int32 ( 61 ) ,
RecoveryFramesOnHit : int32 ( 61 ) ,
Moveforward : & Vec2D {
X : 0 ,
Y : 0 ,
} ,
HitboxOffset : float64 ( 24.0 ) ,
HitboxSize : & Vec2D {
X : float64 ( 45.0 ) ,
Y : float64 ( 32.0 ) ,
} ,
// for defender
HitStunFrames : int32 ( 18 ) ,
BlockStunFrames : int32 ( 9 ) ,
Pushback : float64 ( 22.0 ) ,
ReleaseTriggerType : int32 ( 1 ) , // 1: rising-edge, 2: falling-edge
Damage : int32 ( 5 ) ,
}
2022-12-16 13:55:43 +08:00
bulletLeftToRight := false
2022-11-25 08:21:03 +08:00
if bulletLeftToRight {
xfac := float64 ( 1.0 )
offenderWx , offenderWy := playerPosList . Eles [ 0 ] . X , playerPosList . Eles [ 0 ] . Y
bulletWx , bulletWy := offenderWx + xfac * meleeBullet . HitboxOffset , offenderWy
2022-12-16 13:55:43 +08:00
newBulletCollider := GenerateRectCollider ( bulletWx , bulletWy , meleeBullet . HitboxSize . X , meleeBullet . HitboxSize . Y , topPadding , bottomPadding , leftPadding , rightPadding , spaceOffsetX , spaceOffsetY , "MeleeBullet" )
2022-11-25 08:21:03 +08:00
space . Add ( newBulletCollider )
bulletShape := newBulletCollider . Shape . ( * resolv . ConvexPolygon )
Logger . Warn ( fmt . Sprintf ( "bullet ->: Added bullet collider to space: a=%v" , ConvexPolygonStr ( bulletShape ) ) )
if collision := newBulletCollider . Check ( 0 , 0 ) ; collision != nil {
for _ , obj := range collision . Objects {
objShape := obj . Shape . ( * resolv . ConvexPolygon )
if overlapped , pushbackX , pushbackY , overlapResult := CalcPushbacks ( 0 , 0 , bulletShape , objShape ) ; overlapped {
Logger . Warn ( fmt . Sprintf ( "bullet ->: Overlapped: a=%v, b=%v, pushbackX=%v, pushbackY=%v" , ConvexPolygonStr ( bulletShape ) , ConvexPolygonStr ( objShape ) , pushbackX , pushbackY ) )
} else {
Logger . Warn ( fmt . Sprintf ( "bullet ->: Collided BUT not overlapped: a=%v, b=%v, overlapResult=%v" , ConvexPolygonStr ( bulletShape ) , ConvexPolygonStr ( objShape ) , overlapResult ) )
}
}
}
}
2022-12-16 13:55:43 +08:00
bulletRightToLeft := false
2022-11-25 08:21:03 +08:00
if bulletRightToLeft {
xfac := float64 ( - 1.0 )
offenderWx , offenderWy := playerPosList . Eles [ 1 ] . X , playerPosList . Eles [ 1 ] . Y
bulletWx , bulletWy := offenderWx + xfac * meleeBullet . HitboxOffset , offenderWy
2022-12-16 13:55:43 +08:00
newBulletCollider := GenerateRectCollider ( bulletWx , bulletWy , meleeBullet . HitboxSize . X , meleeBullet . HitboxSize . Y , topPadding , bottomPadding , leftPadding , rightPadding , spaceOffsetX , spaceOffsetY , "MeleeBullet" )
2022-11-25 08:21:03 +08:00
space . Add ( newBulletCollider )
bulletShape := newBulletCollider . Shape . ( * resolv . ConvexPolygon )
Logger . Warn ( fmt . Sprintf ( "bullet <-: Added bullet collider to space: a=%v" , ConvexPolygonStr ( bulletShape ) ) )
if collision := newBulletCollider . Check ( 0 , 0 ) ; collision != nil {
for _ , obj := range collision . Objects {
objShape := obj . Shape . ( * resolv . ConvexPolygon )
if overlapped , pushbackX , pushbackY , overlapResult := CalcPushbacks ( 0 , 0 , bulletShape , objShape ) ; overlapped {
Logger . Warn ( fmt . Sprintf ( "bullet <-: Overlapped: a=%v, b=%v, pushbackX=%v, pushbackY=%v" , ConvexPolygonStr ( bulletShape ) , ConvexPolygonStr ( objShape ) , pushbackX , pushbackY ) )
} else {
Logger . Warn ( fmt . Sprintf ( "bullet <-: Collided BUT not overlapped: a=%v, b=%v, overlapResult=%v" , ConvexPolygonStr ( bulletShape ) , ConvexPolygonStr ( objShape ) , overlapResult ) )
}
}
}
}
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-11-25 08:21:03 +08:00
} else if o . HasTags ( "MeleeBullet" ) {
drawColor := color . RGBA { 0 , 0 , 255 , 255 }
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-12-14 21:30:01 +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 ( ) ) ) ,
)
}
}