2022-10-19 15:10:11 +08:00
package dnmshared
import (
2022-11-09 14:20:26 +08:00
. "dnmshared/sharedprotos"
2022-10-22 13:38:10 +08:00
"fmt"
2022-10-19 15:10:11 +08:00
"github.com/kvartborg/vector"
"github.com/solarlune/resolv"
2022-10-19 17:32:18 +08:00
"math"
2022-10-22 13:38:10 +08:00
"strings"
2022-10-19 15:10:11 +08:00
)
2022-10-22 13:38:10 +08:00
func ConvexPolygonStr ( body * resolv . ConvexPolygon ) string {
var s [ ] string = make ( [ ] string , len ( body . Points ) )
for i , p := range body . Points {
2022-11-12 20:34:38 +08:00
s [ i ] = fmt . Sprintf ( "[%.2f, %.2f]" , p [ 0 ] + body . X , p [ 1 ] + body . Y )
2022-10-22 13:38:10 +08:00
}
2022-11-12 20:34:38 +08:00
return fmt . Sprintf ( "{\n%s\n}" , strings . Join ( s , ",\n" ) )
2022-10-22 13:38:10 +08:00
}
2022-10-19 15:10:11 +08:00
func GenerateRectCollider ( origX , origY , w , h , spaceOffsetX , spaceOffsetY float64 , tag string ) * resolv . Object {
2022-11-12 20:34:38 +08:00
cx , cy := WorldToPolygonColliderAnchorPos ( origX , origY , w * 0.5 , h * 0.5 , spaceOffsetX , spaceOffsetY )
2022-12-11 18:14:02 +08:00
return GenerateRectColliderInCollisionSpace ( cx , cy , w , h , tag )
2022-11-24 17:48:07 +08:00
}
func GenerateRectColliderInCollisionSpace ( cx , cy , w , h float64 , tag string ) * resolv . Object {
2022-11-12 20:34:38 +08:00
collider := resolv . NewObject ( cx , cy , w , h , tag )
2022-10-19 17:32:18 +08:00
shape := resolv . NewRectangle ( 0 , 0 , w , h )
collider . SetShape ( shape )
return collider
2022-10-19 15:10:11 +08:00
}
func GenerateConvexPolygonCollider ( unalignedSrc * Polygon2D , spaceOffsetX , spaceOffsetY float64 , tag string ) * resolv . Object {
2022-10-19 17:32:18 +08:00
aligned := AlignPolygon2DToBoundingBox ( unalignedSrc )
var w , h float64 = 0 , 0
shape := resolv . NewConvexPolygon ( )
for i , pi := range aligned . Points {
for j , pj := range aligned . Points {
if i == j {
continue
}
if math . Abs ( pj . X - pi . X ) > w {
w = math . Abs ( pj . X - pi . X )
}
if math . Abs ( pj . Y - pi . Y ) > h {
h = math . Abs ( pj . Y - pi . Y )
}
}
}
for i := 0 ; i < len ( aligned . Points ) ; i ++ {
p := aligned . Points [ i ]
shape . AddPoints ( p . X , p . Y )
}
collider := resolv . NewObject ( aligned . Anchor . X + spaceOffsetX , aligned . Anchor . Y + spaceOffsetY , w , h , tag )
collider . SetShape ( shape )
return collider
2022-10-19 15:10:11 +08:00
}
2022-11-12 11:41:18 +08:00
func CalcPushbacks ( oldDx , oldDy float64 , playerShape , barrierShape * resolv . ConvexPolygon ) ( bool , float64 , float64 , * SatResult ) {
2022-10-22 13:38:10 +08:00
origX , origY := playerShape . Position ( )
defer func ( ) {
playerShape . SetPosition ( origX , origY )
} ( )
playerShape . SetPosition ( origX + oldDx , origY + oldDy )
overlapResult := & SatResult {
Overlap : 0 ,
OverlapX : 0 ,
OverlapY : 0 ,
AContainedInB : true ,
BContainedInA : true ,
Axis : vector . Vector { 0 , 0 } ,
}
if overlapped := IsPolygonPairOverlapped ( playerShape , barrierShape , overlapResult ) ; overlapped {
pushbackX , pushbackY := overlapResult . Overlap * overlapResult . OverlapX , overlapResult . Overlap * overlapResult . OverlapY
2022-11-12 11:41:18 +08:00
return true , pushbackX , pushbackY , overlapResult
2022-10-22 13:38:10 +08:00
} else {
2022-11-12 11:41:18 +08:00
return false , 0 , 0 , overlapResult
2022-10-22 13:38:10 +08:00
}
}
2022-10-19 17:32:18 +08:00
type SatResult struct {
2022-10-22 13:38:10 +08:00
Overlap float64
OverlapX float64
OverlapY float64
AContainedInB bool
BContainedInA bool
Axis vector . Vector
2022-10-19 17:32:18 +08:00
}
2022-10-19 15:10:11 +08:00
2022-10-22 13:38:10 +08:00
func IsPolygonPairOverlapped ( a , b * resolv . ConvexPolygon , result * SatResult ) bool {
2022-10-19 17:32:18 +08:00
aCnt , bCnt := len ( a . Points ) , len ( b . Points )
// Single point case
if 1 == aCnt && 1 == bCnt {
if nil != result {
result . Overlap = 0
}
return a . Points [ 0 ] . X ( ) == b . Points [ 0 ] . X ( ) && a . Points [ 0 ] . Y ( ) == b . Points [ 0 ] . Y ( )
}
if 1 < aCnt {
for _ , axis := range a . SATAxes ( ) {
2022-10-22 13:38:10 +08:00
if isPolygonPairSeparatedByDir ( a , b , axis . Unit ( ) , result ) {
2022-10-19 17:32:18 +08:00
return false
}
}
}
if 1 < bCnt {
for _ , axis := range b . SATAxes ( ) {
2022-10-22 13:38:10 +08:00
if isPolygonPairSeparatedByDir ( a , b , axis . Unit ( ) , result ) {
2022-10-19 17:32:18 +08:00
return false
}
}
}
return true
}
2022-10-19 15:10:11 +08:00
2022-10-22 13:38:10 +08:00
func isPolygonPairSeparatedByDir ( a , b * resolv . ConvexPolygon , e vector . Vector , result * SatResult ) bool {
/ *
[ WARNING ] This function is deliberately made private , it shouldn ' t be used alone ( i . e . not along the norms of a polygon ) , otherwise the pushbacks calculated would be meaningless .
Consider the following example
a : {
anchor : [ 1337.19 1696.74 ]
points : [ [ 0 0 ] [ 24 0 ] [ 24 24 ] [ 0 24 ] ]
} ,
b : {
anchor : [ 1277.72 1570.56 ]
points : [ [ 642.57 319.16 ] [ 0 319.16 ] [ 5.73 0 ] [ 643.75 0.90 ] ]
}
e = ( - 2.98 , 1.49 ) . Unit ( )
* /
2022-10-19 17:32:18 +08:00
var aStart , aEnd , bStart , bEnd float64 = math . MaxFloat64 , - math . MaxFloat64 , math . MaxFloat64 , - math . MaxFloat64
for _ , p := range a . Points {
2022-10-22 13:38:10 +08:00
dot := ( p . X ( ) + a . X ) * e . X ( ) + ( p . Y ( ) + a . Y ) * e . Y ( )
2022-10-19 17:32:18 +08:00
if aStart > dot {
aStart = dot
}
if aEnd < dot {
aEnd = dot
}
}
for _ , p := range b . Points {
2022-10-22 13:38:10 +08:00
dot := ( p . X ( ) + b . X ) * e . X ( ) + ( p . Y ( ) + b . Y ) * e . Y ( )
2022-10-19 17:32:18 +08:00
if bStart > dot {
bStart = dot
}
if bEnd < dot {
bEnd = dot
}
}
if aStart > bEnd || aEnd < bStart {
// Separated by unit vector "e"
return true
}
if nil != result {
2022-10-22 13:38:10 +08:00
result . Axis = e
2022-10-19 17:32:18 +08:00
overlap := float64 ( 0 )
if aStart < bStart {
result . AContainedInB = false
if aEnd < bEnd {
overlap = aEnd - bStart
result . BContainedInA = false
} else {
option1 := aEnd - bStart
option2 := bEnd - aStart
if option1 < option2 {
overlap = option1
} else {
overlap = - option2
}
}
} else {
result . BContainedInA = false
if aEnd > bEnd {
overlap = aStart - bEnd
result . AContainedInB = false
} else {
option1 := aEnd - bStart
option2 := bEnd - aStart
if option1 < option2 {
overlap = option1
} else {
overlap = - option2
}
}
}
currentOverlap := result . Overlap
absoluteOverlap := overlap
if overlap < 0 {
absoluteOverlap = - overlap
}
if 0 == currentOverlap || currentOverlap > absoluteOverlap {
var sign float64 = 1
if overlap < 0 {
sign = - 1
}
result . Overlap = absoluteOverlap
result . OverlapX = e . X ( ) * sign
result . OverlapY = e . Y ( ) * sign
}
}
// the specified unit vector "e" doesn't separate "a" and "b", overlap result is generated
return false
2022-10-19 15:10:11 +08:00
}
2022-11-12 20:34:38 +08:00
func WorldToVirtualGridPos ( wx , wy , worldToVirtualGridRatio float64 ) ( int32 , int32 ) {
// [WARNING] Introduces loss of precision!
// In JavaScript floating numbers suffer from seemingly non-deterministic arithmetics, and even if certain libs solved this issue by approaches such as fixed-point-number, they might not be used in other libs -- e.g. the "collision libs" we're interested in -- thus couldn't kill all pains.
var virtualGridX int32 = int32 ( math . Round ( wx * worldToVirtualGridRatio ) )
var virtualGridY int32 = int32 ( math . Round ( wy * worldToVirtualGridRatio ) )
return virtualGridX , virtualGridY
}
func VirtualGridToWorldPos ( vx , vy int32 , virtualGridToWorldRatio float64 ) ( float64 , float64 ) {
// No loss of precision
var wx float64 = float64 ( vx ) * virtualGridToWorldRatio
var wy float64 = float64 ( vy ) * virtualGridToWorldRatio
return wx , wy
}
func WorldToPolygonColliderAnchorPos ( wx , wy , halfBoundingW , halfBoundingH , collisionSpaceOffsetX , collisionSpaceOffsetY float64 ) ( float64 , float64 ) {
return wx - halfBoundingW + collisionSpaceOffsetX , wy - halfBoundingH + collisionSpaceOffsetY
}
func PolygonColliderAnchorToWorldPos ( cx , cy , halfBoundingW , halfBoundingH , collisionSpaceOffsetX , collisionSpaceOffsetY float64 ) ( float64 , float64 ) {
return cx + halfBoundingW - collisionSpaceOffsetX , cy + halfBoundingH - collisionSpaceOffsetY
}
func PolygonColliderAnchorToVirtualGridPos ( cx , cy , halfBoundingW , halfBoundingH , collisionSpaceOffsetX , collisionSpaceOffsetY float64 , worldToVirtualGridRatio float64 ) ( int32 , int32 ) {
wx , wy := PolygonColliderAnchorToWorldPos ( cx , cy , halfBoundingW , halfBoundingH , collisionSpaceOffsetX , collisionSpaceOffsetY )
return WorldToVirtualGridPos ( wx , wy , worldToVirtualGridRatio )
}
func VirtualGridToPolygonColliderAnchorPos ( vx , vy int32 , halfBoundingW , halfBoundingH , collisionSpaceOffsetX , collisionSpaceOffsetY float64 , virtualGridToWorldRatio float64 ) ( float64 , float64 ) {
wx , wy := VirtualGridToWorldPos ( vx , vy , virtualGridToWorldRatio )
return WorldToPolygonColliderAnchorPos ( wx , wy , halfBoundingW , halfBoundingH , collisionSpaceOffsetX , collisionSpaceOffsetY )
}