2022-11-17 15:39:32 +00:00
const i18n = require ( 'LanguageData' ) ;
i18n . init ( window . language ) ; // languageID should be equal to the one we input in New Language ID input field
const OnlineMap = require ( './Map' ) ;
2022-12-10 04:30:44 +00:00
/ *
[ WARNING ] As when a character is standing on a barrier , if not carefully curated there MIGHT BE a bouncing sequence of "[(inAir -> dropIntoBarrier ->), (notInAir -> pushedOutOfBarrier ->)], [(inAir -> ..."
Moreover , this "snapIntoPlatformOverlap" should be small enough such that the jumping initial "velY" can escape from it by 1 renderFrame ( when jumping is triggered , the character is waived from snappig for 1 renderFrame ) .
* /
const snapIntoPlatformOverlap = 0.1 ;
2022-11-17 15:39:32 +00:00
cc . Class ( {
extends : OnlineMap ,
onDestroy ( ) {
console . warn ( "+++++++ Map onDestroy()" ) ;
} ,
onLoad ( ) {
const self = this ;
window . mapIns = self ;
2022-11-19 12:58:07 +00:00
cc . director . getCollisionManager ( ) . enabled = false ;
const mapNode = self . node ;
const canvasNode = mapNode . parent ;
self . mainCameraNode = self . canvasNode . getChildByName ( "Main Camera" ) ;
2022-11-17 15:39:32 +00:00
self . mainCamera = self . mainCameraNode . getComponent ( cc . Camera ) ;
for ( let child of self . mainCameraNode . children ) {
child . setScale ( 1 / self . mainCamera . zoomRatio ) ;
}
self . widgetsAboveAllNode = self . mainCameraNode . getChildByName ( "WidgetsAboveAll" ) ;
self . mainCameraNode . setPosition ( cc . v2 ( ) ) ;
/** Init required prefab ended. */
2022-11-19 12:58:07 +00:00
self . inputDelayFrames = 8 ;
self . inputScaleFrames = 2 ;
self . inputFrameUpsyncDelayTolerance = 2 ;
2022-11-17 15:39:32 +00:00
2022-12-01 03:35:56 +00:00
self . renderCacheSize = 1024 ;
2022-12-09 09:22:04 +00:00
self . serverFps = 60 ;
2022-11-19 12:58:07 +00:00
self . rollbackEstimatedDt = 0.016667 ;
self . rollbackEstimatedDtMillis = 16.667 ;
self . rollbackEstimatedDtNanos = 16666666 ;
2022-11-17 15:39:32 +00:00
2022-11-19 12:58:07 +00:00
self . worldToVirtualGridRatio = 1000 ;
self . virtualGridToWorldRatio = 1.0 / self . worldToVirtualGridRatio ;
2022-11-24 11:45:48 +00:00
self . meleeSkillConfig = {
1 : {
// for offender
2022-11-27 11:38:26 +00:00
startupFrames : 10 ,
2022-11-25 00:21:03 +00:00
activeFrames : 3 ,
2022-11-27 11:38:26 +00:00
recoveryFrames : 34 , // usually but not always "startupFrames+activeFrames", I hereby set it to be 1 frame more than the actual animation to avoid critical transition, i.e. when the animation is 1 frame from ending but "rdfPlayer.framesToRecover" is already counted 0 and the player triggers an other same attack, making an effective bullet trigger but no animation is played due to same animName is still playing
recoveryFramesOnBlock : 34 ,
recoveryFramesOnHit : 34 ,
2022-11-24 11:45:48 +00:00
moveforward : {
x : 0 ,
y : 0 ,
} ,
hitboxOffset : 12.0 , // should be about the radius of the PlayerCollider
hitboxSize : {
2022-11-27 11:38:26 +00:00
x : 23.0 ,
2022-11-27 13:33:34 +00:00
y : 32.0 ,
2022-11-24 11:45:48 +00:00
} ,
// for defender
hitStunFrames : 18 ,
blockStunFrames : 9 ,
2022-11-27 11:38:26 +00:00
pushback : 8.0 ,
2022-11-24 11:45:48 +00:00
releaseTriggerType : 1 , // 1: rising-edge, 2: falling-edge
damage : 5
}
} ;
2022-11-17 15:39:32 +00:00
const tiledMapIns = self . node . getComponent ( cc . TiledMap ) ;
2022-11-19 12:58:07 +00:00
const fullPathOfTmxFile = cc . js . formatStr ( "map/%s/map" , "dungeon" ) ;
2022-11-17 15:39:32 +00:00
cc . loader . loadRes ( fullPathOfTmxFile , cc . TiledMapAsset , ( err , tmxAsset ) => {
if ( null != err ) {
console . error ( err ) ;
return ;
}
tiledMapIns . tmxAsset = null ;
mapNode . removeAllChildren ( ) ;
self . _resetCurrentMatch ( ) ;
tiledMapIns . tmxAsset = tmxAsset ;
const newMapSize = tiledMapIns . getMapSize ( ) ;
const newTileSize = tiledMapIns . getTileSize ( ) ;
self . node . setContentSize ( newMapSize . width * newTileSize . width , newMapSize . height * newTileSize . height ) ;
self . node . setPosition ( cc . v2 ( 0 , 0 ) ) ;
let barrierIdCounter = 0 ;
const boundaryObjs = tileCollisionManager . extractBoundaryObjects ( self . node ) ;
for ( let boundaryObj of boundaryObjs . barriers ) {
const x0 = boundaryObj . anchor . x ,
y0 = boundaryObj . anchor . y ;
const newBarrier = self . collisionSys . createPolygon ( x0 , y0 , Array . from ( boundaryObj , p => {
return [ p . x , p . y ] ;
} ) ) ;
if ( self . showCriticalCoordinateLabels ) {
for ( let i = 0 ; i < boundaryObj . length ; ++ i ) {
const barrierVertLabelNode = new cc . Node ( ) ;
switch ( i % 4 ) {
case 0 :
barrierVertLabelNode . color = cc . Color . RED ;
break ;
case 1 :
barrierVertLabelNode . color = cc . Color . GRAY ;
break ;
case 2 :
barrierVertLabelNode . color = cc . Color . BLACK ;
break ;
default :
barrierVertLabelNode . color = cc . Color . MAGENTA ;
break ;
}
const wx = boundaryObj . anchor . x + boundaryObj [ i ] . x ,
wy = boundaryObj . anchor . y + boundaryObj [ i ] . y ;
barrierVertLabelNode . setPosition ( cc . v2 ( wx , wy ) ) ;
const barrierVertLabel = barrierVertLabelNode . addComponent ( cc . Label ) ;
barrierVertLabel . fontSize = 12 ;
barrierVertLabel . lineHeight = barrierVertLabel . fontSize + 1 ;
barrierVertLabel . string = ` ( ${ wx . toFixed ( 1 ) } , ${ wy . toFixed ( 1 ) } ) ` ;
safelyAddChild ( self . node , barrierVertLabelNode ) ;
setLocalZOrder ( barrierVertLabelNode , 5 ) ;
barrierVertLabelNode . active = true ;
}
}
// console.log("Created barrier: ", newBarrier);
++ barrierIdCounter ;
const collisionBarrierIndex = ( self . collisionBarrierIndexPrefix + barrierIdCounter ) ;
self . collisionSysMap . set ( collisionBarrierIndex , newBarrier ) ;
}
2022-11-19 12:58:07 +00:00
2022-11-25 05:24:03 +00:00
const startRdf = window . pb . protos . RoomDownsyncFrame . create ( {
2022-11-19 12:58:07 +00:00
id : window . MAGIC _ROOM _DOWNSYNC _FRAME _ID . BATTLE _START ,
players : {
2022-12-09 09:22:04 +00:00
10 : window . pb . protos . PlayerDownsync . create ( {
2022-11-19 12:58:07 +00:00
id : 10 ,
2022-11-20 13:07:45 +00:00
joinIndex : 1 ,
2022-12-09 09:22:04 +00:00
virtualGridX : - 50 * self . worldToVirtualGridRatio ,
virtualGridY : - 400 * self . worldToVirtualGridRatio ,
2022-11-25 00:21:03 +00:00
speed : 1 * self . worldToVirtualGridRatio ,
2022-11-22 08:19:04 +00:00
colliderRadius : 12 ,
2022-12-09 09:22:04 +00:00
characterState : window . ATK _CHARACTER _STATE . InAirIdle1 [ 0 ] ,
2022-11-23 07:04:32 +00:00
framesToRecover : 0 ,
2022-11-23 14:11:28 +00:00
dirX : 0 ,
dirY : 0 ,
2022-12-09 09:22:04 +00:00
velX : 0 ,
velY : 0 ,
inAir : true ,
} ) ,
11 : window . pb . protos . PlayerDownsync . create ( {
2022-11-22 08:19:04 +00:00
id : 11 ,
joinIndex : 2 ,
2022-12-09 09:22:04 +00:00
virtualGridX : 100 * self . worldToVirtualGridRatio ,
virtualGridY : - 350 * self . worldToVirtualGridRatio ,
2022-11-25 00:21:03 +00:00
speed : 1 * self . worldToVirtualGridRatio ,
2022-11-19 12:58:07 +00:00
colliderRadius : 12 ,
2022-12-09 09:22:04 +00:00
characterState : window . ATK _CHARACTER _STATE . InAirIdle1 [ 0 ] ,
2022-11-23 07:04:32 +00:00
framesToRecover : 0 ,
2022-11-23 14:11:28 +00:00
dirX : 0 ,
dirY : 0 ,
2022-12-09 09:22:04 +00:00
velX : 0 ,
velY : 0 ,
inAir : true ,
} ) ,
2022-11-19 12:58:07 +00:00
}
2022-11-25 05:24:03 +00:00
} ) ;
2022-11-19 12:58:07 +00:00
self . selfPlayerInfo = {
2022-11-27 13:33:34 +00:00
id : 11
2022-11-19 12:58:07 +00:00
} ;
2022-11-22 08:19:04 +00:00
self . _initPlayerRichInfoDict ( startRdf . players ) ;
2022-11-19 12:58:07 +00:00
self . onRoomDownsyncFrame ( startRdf ) ;
self . battleState = ALL _BATTLE _STATES . IN _BATTLE ;
2022-11-17 15:39:32 +00:00
} ) ;
} ,
update ( dt ) {
const self = this ;
if ( ALL _BATTLE _STATES . IN _BATTLE == self . battleState ) {
const elapsedMillisSinceLastFrameIdTriggered = performance . now ( ) - self . lastRenderFrameIdTriggeredAt ;
if ( elapsedMillisSinceLastFrameIdTriggered < ( self . rollbackEstimatedDtMillis ) ) {
// console.debug("Avoiding too fast frame@renderFrameId=", self.renderFrameId, ": elapsedMillisSinceLastFrameIdTriggered=", elapsedMillisSinceLastFrameIdTriggered);
return ;
}
try {
let st = performance . now ( ) ;
let prevSelfInput = null ,
currSelfInput = null ;
const noDelayInputFrameId = self . _convertToInputFrameId ( self . renderFrameId , 0 ) ; // It's important that "inputDelayFrames == 0" here
if ( self . shouldGenerateInputFrameUpsync ( self . renderFrameId ) ) {
const prevAndCurrInputs = self . _generateInputFrameUpsync ( noDelayInputFrameId ) ;
prevSelfInput = prevAndCurrInputs [ 0 ] ;
currSelfInput = prevAndCurrInputs [ 1 ] ;
}
2022-11-25 03:20:05 +00:00
const [ prevRdf , rdf ] = self . rollbackAndChase ( self . renderFrameId , self . renderFrameId + 1 , self . collisionSys , self . collisionSysMap , false ) ;
self . applyRoomDownsyncFrameDynamics ( rdf , prevRdf ) ;
2022-11-17 15:39:32 +00:00
let t3 = performance . now ( ) ;
} catch ( err ) {
console . error ( "Error during Map.update" , err ) ;
} finally {
++ self . renderFrameId ; // [WARNING] It's important to increment the renderFrameId AFTER all the operations above!!!
}
}
} ,
2022-12-09 09:22:04 +00:00
applyInputFrameDownsyncDynamicsOnSingleRenderFrame ( delayedInputFrame , currRenderFrame , collisionSys , collisionSysMap ) {
const self = this ;
const nextRenderFramePlayers = { } ;
for ( let playerId in currRenderFrame . players ) {
const currPlayerDownsync = currRenderFrame . players [ playerId ] ;
nextRenderFramePlayers [ playerId ] = {
id : playerId ,
virtualGridX : currPlayerDownsync . virtualGridX ,
virtualGridY : currPlayerDownsync . virtualGridY ,
dirX : currPlayerDownsync . dirX ,
dirY : currPlayerDownsync . dirY ,
velX : currPlayerDownsync . velX ,
velY : currPlayerDownsync . velY ,
characterState : currPlayerDownsync . characterState ,
inAir : true , // will be updated if collided with a barrier with "0 > pushbackY"
speed : currPlayerDownsync . speed ,
battleState : currPlayerDownsync . battleState ,
score : currPlayerDownsync . score ,
removed : currPlayerDownsync . removed ,
joinIndex : currPlayerDownsync . joinIndex ,
framesToRecover : ( 0 < currPlayerDownsync . framesToRecover ? currPlayerDownsync . framesToRecover - 1 : 0 ) ,
hp : currPlayerDownsync . hp ,
maxHp : currPlayerDownsync . maxHp ,
} ;
}
const nextRenderFrameMeleeBullets = [ ] ;
2022-12-10 04:30:44 +00:00
// Guaranteed determinism regardless of traversal order
const jumpTriggered = new Array ( self . playerRichInfoArr . length ) ;
const movements = new Array ( self . playerRichInfoArr . length ) ;
const bulletPushbacks = new Array ( self . playerRichInfoArr . length ) ;
const effPushbacks = new Array ( self . playerRichInfoArr . length ) ;
2022-12-09 09:22:04 +00:00
// Reset playerCollider position from the "virtual grid position"
for ( let j in self . playerRichInfoArr ) {
const joinIndex = parseInt ( j ) + 1 ;
2022-12-10 04:30:44 +00:00
jumpTriggered [ joinIndex - 1 ] = false ;
2022-12-09 09:22:04 +00:00
movements [ joinIndex - 1 ] = [ 0.0 , 0.0 ] ;
bulletPushbacks [ joinIndex - 1 ] = [ 0.0 , 0.0 ] ;
effPushbacks [ joinIndex - 1 ] = [ 0.0 , 0.0 ] ;
const playerRichInfo = self . playerRichInfoArr [ j ] ;
const playerId = playerRichInfo . id ;
const collisionPlayerIndex = self . collisionPlayerIndexPrefix + joinIndex ;
const playerCollider = collisionSysMap . get ( collisionPlayerIndex ) ;
const currPlayerDownsync = currRenderFrame . players [ playerId ] ;
const thatPlayerInNextFrame = nextRenderFramePlayers [ playerId ] ;
const newVx = currPlayerDownsync . virtualGridX ;
const newVy = currPlayerDownsync . virtualGridY ;
[ playerCollider . x , playerCollider . y ] = self . virtualGridToPolygonColliderAnchorPos ( newVx , newVy , self . playerRichInfoArr [ joinIndex - 1 ] . colliderRadius , self . playerRichInfoArr [ joinIndex - 1 ] . colliderRadius ) ;
2022-12-10 04:30:44 +00:00
// Process gravity before anyother interaction, by now "currPlayerDownsync.velX & velY" are properly snapped to be parallel to the edge of its standing platform if necessary
2022-12-09 09:22:04 +00:00
[ movements [ joinIndex - 1 ] [ 0 ] , movements [ joinIndex - 1 ] [ 1 ] ] = self . virtualGridToWorldPos ( currPlayerDownsync . velX , currPlayerDownsync . velY ) ;
playerCollider . x += movements [ joinIndex - 1 ] [ 0 ] ;
playerCollider . y += movements [ joinIndex - 1 ] [ 1 ] ;
if ( currPlayerDownsync . inAir ) {
thatPlayerInNextFrame . velX += self . gravityX ;
thatPlayerInNextFrame . velY += self . gravityY ;
}
}
// Check bullet-anything collisions first, because the pushbacks caused by bullets might later be reverted by player-barrier collision
const bulletColliders = new Map ( ) ; // Will all be removed at the end of `applyInputFrameDownsyncDynamicsOnSingleRenderFrame` due to the need for being rollback-compatible
const removedBulletsAtCurrFrame = new Set ( ) ;
for ( let k in currRenderFrame . meleeBullets ) {
const meleeBullet = currRenderFrame . meleeBullets [ k ] ;
if (
meleeBullet . originatedRenderFrameId + meleeBullet . startupFrames <= currRenderFrame . id
&&
meleeBullet . originatedRenderFrameId + meleeBullet . startupFrames + meleeBullet . activeFrames > currRenderFrame . id
) {
const collisionBulletIndex = self . collisionBulletIndexPrefix + meleeBullet . battleLocalId ;
const collisionOffenderIndex = self . collisionPlayerIndexPrefix + meleeBullet . offenderJoinIndex ;
const offenderCollider = collisionSysMap . get ( collisionOffenderIndex ) ;
const offender = currRenderFrame . players [ meleeBullet . offenderPlayerId ] ;
let xfac = 1 ; // By now, straight Punch offset doesn't respect "y-axis"
if ( 0 > offender . dirX ) {
xfac = - 1 ;
}
const [ offenderWx , offenderWy ] = self . virtualGridToWorldPos ( offender . virtualGridX , offender . virtualGridY ) ;
const bulletWx = offenderWx + xfac * meleeBullet . hitboxOffset ;
const bulletWy = offenderWy ;
const [ bulletCx , bulletCy ] = self . worldToPolygonColliderAnchorPos ( bulletWx , bulletWy , meleeBullet . hitboxSize . x * 0.5 , meleeBullet . hitboxSize . y * 0.5 ) ,
pts = [ [ 0 , 0 ] , [ meleeBullet . hitboxSize . x , 0 ] , [ meleeBullet . hitboxSize . x , meleeBullet . hitboxSize . y ] , [ 0 , meleeBullet . hitboxSize . y ] ] ;
const newBulletCollider = collisionSys . createPolygon ( bulletCx , bulletCy , pts ) ;
newBulletCollider . data = meleeBullet ;
collisionSysMap . set ( collisionBulletIndex , newBulletCollider ) ;
bulletColliders . set ( collisionBulletIndex , newBulletCollider ) ;
// console.log(`A meleeBullet is added to collisionSys at currRenderFrame.id=${currRenderFrame.id} as start-up frames ended and active frame is not yet ended: ${JSON.stringify(meleeBullet)}`);
}
}
collisionSys . update ( ) ;
const result1 = collisionSys . createResult ( ) ; // Can I reuse a "self.collisionSysResult" object throughout the whole battle?
bulletColliders . forEach ( ( bulletCollider , collisionBulletIndex ) => {
const potentials = bulletCollider . potentials ( ) ;
const offender = currRenderFrame . players [ bulletCollider . data . offenderPlayerId ] ;
let shouldRemove = false ;
for ( const potential of potentials ) {
if ( null != potential . data && potential . data . joinIndex == bulletCollider . data . offenderJoinIndex ) continue ;
if ( ! bulletCollider . collides ( potential , result1 ) ) continue ;
if ( null != potential . data && null !== potential . data . joinIndex ) {
const joinIndex = potential . data . joinIndex ;
let xfac = 1 ;
if ( 0 > offender . dirX ) {
xfac = - 1 ;
}
bulletPushbacks [ joinIndex - 1 ] [ 0 ] += xfac * bulletCollider . data . pushback ; // Only for straight punch, there's no y-pushback
bulletPushbacks [ joinIndex - 1 ] [ 1 ] += 0 ;
const thatAckedPlayerInNextFrame = nextRenderFramePlayers [ potential . data . id ] ;
2022-12-09 16:07:03 +00:00
if ( ! window . ATK _CHARACTER _STATE _IN _AIR _SET . has ( thatAckedPlayerInNextFrame . characterState ) ) {
thatAckedPlayerInNextFrame . characterState = window . ATK _CHARACTER _STATE . Atked1 [ 0 ] ;
} else {
thatAckedPlayerInNextFrame . characterState = window . ATK _CHARACTER _STATE . InAirAtked1 [ 0 ] ;
}
2022-12-09 09:22:04 +00:00
const oldFramesToRecover = thatAckedPlayerInNextFrame . framesToRecover ;
thatAckedPlayerInNextFrame . framesToRecover = ( oldFramesToRecover > bulletCollider . data . hitStunFrames ? oldFramesToRecover : bulletCollider . data . hitStunFrames ) ; // In case the hit player is already stun, we extend it
}
shouldRemove = true ;
}
if ( shouldRemove ) {
removedBulletsAtCurrFrame . add ( collisionBulletIndex ) ;
}
} ) ;
// [WARNING] Remove bullets from collisionSys ANYWAY for the convenience of rollback
for ( let k in currRenderFrame . meleeBullets ) {
const meleeBullet = currRenderFrame . meleeBullets [ k ] ;
const collisionBulletIndex = self . collisionBulletIndexPrefix + meleeBullet . battleLocalId ;
if ( collisionSysMap . has ( collisionBulletIndex ) ) {
const bulletCollider = collisionSysMap . get ( collisionBulletIndex ) ;
bulletCollider . remove ( ) ;
collisionSysMap . delete ( collisionBulletIndex ) ;
}
if ( removedBulletsAtCurrFrame . has ( collisionBulletIndex ) ) continue ;
nextRenderFrameMeleeBullets . push ( meleeBullet ) ;
}
// Process player inputs
if ( null != delayedInputFrame ) {
const delayedInputFrameForPrevRenderFrame = self . getCachedInputFrameDownsyncWithPrediction ( self . _convertToInputFrameId ( currRenderFrame . id - 1 , self . inputDelayFrames ) ) ;
const inputList = delayedInputFrame . inputList ;
for ( let j in self . playerRichInfoArr ) {
const joinIndex = parseInt ( j ) + 1 ;
effPushbacks [ joinIndex - 1 ] = [ 0.0 , 0.0 ] ;
const playerRichInfo = self . playerRichInfoArr [ j ] ;
const playerId = playerRichInfo . id ;
const collisionPlayerIndex = self . collisionPlayerIndexPrefix + joinIndex ;
const playerCollider = collisionSysMap . get ( collisionPlayerIndex ) ;
const currPlayerDownsync = currRenderFrame . players [ playerId ] ;
const thatPlayerInNextFrame = nextRenderFramePlayers [ playerId ] ;
if ( 0 < thatPlayerInNextFrame . framesToRecover ) {
// No need to process inputs for this player, but there might be bullet pushbacks on this player
playerCollider . x += bulletPushbacks [ joinIndex - 1 ] [ 0 ] ;
playerCollider . y += bulletPushbacks [ joinIndex - 1 ] [ 1 ] ;
if ( 0 != bulletPushbacks [ joinIndex - 1 ] [ 0 ] || 0 != bulletPushbacks [ joinIndex - 1 ] [ 1 ] ) {
console . log ( ` playerId= ${ playerId } , joinIndex= ${ joinIndex } is pushbacked back by ${ bulletPushbacks [ joinIndex - 1 ] } by bullet impacts, now its framesToRecover is ${ thatPlayerInNextFrame . framesToRecover } ` ) ;
}
continue ;
}
const decodedInput = self . ctrl . decodeInput ( inputList [ joinIndex - 1 ] ) ;
const prevDecodedInput = ( null == delayedInputFrameForPrevRenderFrame ? null : self . ctrl . decodeInput ( delayedInputFrameForPrevRenderFrame . inputList [ joinIndex - 1 ] ) ) ;
const prevBtnALevel = ( null == prevDecodedInput ? 0 : prevDecodedInput . btnALevel ) ;
2022-12-09 16:07:03 +00:00
const prevBtnBLevel = ( null == prevDecodedInput ? 0 : prevDecodedInput . btnBLevel ) ;
2022-12-10 04:30:44 +00:00
/ *
[ WARNING ] Player input alone WOULD NOT take "characterState" into any "ATK_CHARACTER_STATE_IN_AIR_SET" , only after the calculation of "effPushbacks" do we know exactly whether or not a player is "inAir" , the finalize the transition of "thatPlayerInNextFrame.characterState" .
* /
if ( 1 == decodedInput . btnBLevel && 0 == prevBtnBLevel ) {
const characStateAlreadyInAir = window . ATK _CHARACTER _STATE _IN _AIR _SET . has ( thatPlayerInNextFrame . characterState ) ;
const characStateIsInterruptWaivable = window . ATK _CHARACTER _STATE _INTERRUPT _WAIVE _SET . has ( thatPlayerInNextFrame . characterState ) ;
if (
! characStateAlreadyInAir
&&
characStateIsInterruptWaivable
) {
thatPlayerInNextFrame . velY = currPlayerDownsync . speed * 4.5 ;
jumpTriggered [ joinIndex - 1 ] = true ;
console . log ( ` playerId= ${ playerId } , joinIndex= ${ joinIndex } triggered a rising-edge of btnB at renderFrame.id= ${ currRenderFrame . id } , delayedInputFrame.id= ${ delayedInputFrame . inputFrameId } , nextVelY= ${ thatPlayerInNextFrame . velY } , characStateAlreadyInAir= ${ characStateAlreadyInAir } , characStateIsInterruptWaivable= ${ characStateIsInterruptWaivable } ` ) ;
}
}
2022-12-09 09:22:04 +00:00
if ( 1 == decodedInput . btnALevel && 0 == prevBtnALevel ) {
// console.log(`playerId=${playerId} triggered a rising-edge of btnA at renderFrame.id=${currRenderFrame.id}, delayedInputFrame.id=${delayedInputFrame.inputFrameId}`);
if ( self . bulletTriggerEnabled ) {
const punchSkillId = 1 ;
const punch = window . pb . protos . MeleeBullet . create ( self . meleeSkillConfig [ punchSkillId ] ) ;
thatPlayerInNextFrame . framesToRecover = punch . recoveryFrames ;
punch . battleLocalId = self . bulletBattleLocalIdCounter ++ ;
punch . offenderJoinIndex = joinIndex ;
punch . offenderPlayerId = playerId ;
punch . originatedRenderFrameId = currRenderFrame . id ;
nextRenderFrameMeleeBullets . push ( punch ) ;
// console.log(`A rising-edge of meleeBullet is created at renderFrame.id=${currRenderFrame.id}, delayedInputFrame.id=${delayedInputFrame.inputFrameId}: ${self._stringifyRecentInputCache(true)}`);
// console.log(`A rising-edge of meleeBullet is created at renderFrame.id=${currRenderFrame.id}, delayedInputFrame.id=${delayedInputFrame.inputFrameId}`);
2022-12-10 04:30:44 +00:00
thatPlayerInNextFrame . characterState = window . ATK _CHARACTER _STATE . Atk1 [ 0 ] ;
2022-12-09 09:22:04 +00:00
}
} else if ( 0 == decodedInput . btnALevel && 1 == prevBtnALevel ) {
// console.log(`playerId=${playerId} triggered a falling-edge of btnA at renderFrame.id=${currRenderFrame.id}, delayedInputFrame.id=${delayedInputFrame.inputFrameId}`);
} else {
2022-12-10 04:30:44 +00:00
// No bullet trigger, process joystick movement inputs (except for jumping).
if ( 0 != decodedInput . dx || 0 != decodedInput . dy ) {
// Update directions and thus would eventually update moving animation accordingly
thatPlayerInNextFrame . dirX = decodedInput . dx ;
thatPlayerInNextFrame . dirY = decodedInput . dy ;
thatPlayerInNextFrame . characterState = window . ATK _CHARACTER _STATE . Walking [ 0 ] ;
thatPlayerInNextFrame . velX = decodedInput . dx * currPlayerDownsync . speed ;
2022-12-09 09:22:04 +00:00
} else {
2022-12-10 04:30:44 +00:00
thatPlayerInNextFrame . characterState = window . ATK _CHARACTER _STATE . Idle1 [ 0 ] ;
thatPlayerInNextFrame . velX = 0 ;
2022-12-09 09:22:04 +00:00
}
}
}
}
collisionSys . update ( ) ; // by now all "bulletCollider"s are removed
const result2 = collisionSys . createResult ( ) ; // Can I reuse a "self.collisionSysResult" object throughout the whole battle?
for ( let j in self . playerRichInfoArr ) {
const joinIndex = parseInt ( j ) + 1 ;
const playerId = self . playerRichInfoArr [ j ] . id ;
const collisionPlayerIndex = self . collisionPlayerIndexPrefix + joinIndex ;
const playerCollider = collisionSysMap . get ( collisionPlayerIndex ) ;
const potentials = playerCollider . potentials ( ) ;
const currPlayerDownsync = currRenderFrame . players [ playerId ] ;
const thatPlayerInNextFrame = nextRenderFramePlayers [ playerId ] ;
let fallStopping = false ;
2022-12-10 04:30:44 +00:00
let [ snappedIntoPlatformEx , snappedIntoPlatformEy ] = [ null , null ] ;
2022-12-09 09:22:04 +00:00
for ( const potential of potentials ) {
// Test if the player collides with the wall
if ( ! playerCollider . collides ( potential , result2 ) ) continue ;
// Push the player out of the wall
let [ pushbackX , pushbackY ] = [ result2 . overlap * result2 . overlap _x , result2 . overlap * result2 . overlap _y ] ;
if ( null == potential . data ) {
// "null == potential.data" implies a barrier
2022-12-10 04:30:44 +00:00
const remainsNotInAir = ( ! currPlayerDownsync . inAir ) ;
2022-12-09 16:07:03 +00:00
const localFallStopping = ( currPlayerDownsync . inAir && 0 > pushbackY ) ; // prevents false fall-stopping on the lateral sides
2022-12-10 04:30:44 +00:00
if ( remainsNotInAir || localFallStopping ) {
fallStopping |= localFallStopping ;
[ pushbackX , pushbackY ] = [ ( result2 . overlap - snapIntoPlatformOverlap ) * result2 . overlap _x , ( result2 . overlap - snapIntoPlatformOverlap ) * result2 . overlap _y ]
// [overlay_x, overlap_y] is the unit vector that points into the platform; FIXME: Should only assign to [snappedIntoPlatformEx, snappedIntoPlatformEy] at most once!
snappedIntoPlatformEx = - result2 . overlap _y ;
snappedIntoPlatformEy = result2 . overlap _x ;
if ( snappedIntoPlatformEx * currPlayerDownsync . dirX + snappedIntoPlatformEy * currPlayerDownsync . dirY ) {
[ snappedIntoPlatformEx , snappedIntoPlatformEy ] = [ - snappedIntoPlatformEx , - snappedIntoPlatformEy ] ;
}
2022-12-09 09:22:04 +00:00
}
}
// What if we're on the edge of 2 barriers? Would adding up make an unexpected bounce?
effPushbacks [ joinIndex - 1 ] [ 0 ] += pushbackX ;
effPushbacks [ joinIndex - 1 ] [ 1 ] += pushbackY ;
}
2022-12-10 04:30:44 +00:00
if ( false == jumpTriggered [ joinIndex - 1 ] && null != snappedIntoPlatformEx && null != snappedIntoPlatformEy ) {
2022-12-09 09:22:04 +00:00
thatPlayerInNextFrame . inAir = false ;
2022-12-10 04:30:44 +00:00
if ( fallStopping ) {
thatPlayerInNextFrame . velY = 0 ; // shuts off the velocity component introduced by gravity, otherwise the character will slip down along a slope
2022-12-09 16:07:03 +00:00
}
2022-12-10 04:30:44 +00:00
const dotProd = thatPlayerInNextFrame . velX * snappedIntoPlatformEx + thatPlayerInNextFrame . velY * snappedIntoPlatformEy ;
[ thatPlayerInNextFrame . velX , thatPlayerInNextFrame . velY ] = [ dotProd * snappedIntoPlatformEx , dotProd * snappedIntoPlatformEy ] ;
2022-12-09 09:22:04 +00:00
}
2022-12-10 04:30:44 +00:00
if ( currPlayerDownsync . inAir != thatPlayerInNextFrame . inAir ) {
thatPlayerInNextFrame . characterState = window . toInAirConjugate ( thatPlayerInNextFrame . characterState ) ;
2022-12-09 09:22:04 +00:00
}
}
// Get players out of stuck barriers if there's any
for ( let j in self . playerRichInfoArr ) {
const joinIndex = parseInt ( j ) + 1 ;
const playerId = self . playerRichInfoArr [ j ] . id ;
const collisionPlayerIndex = self . collisionPlayerIndexPrefix + joinIndex ;
const playerCollider = collisionSysMap . get ( collisionPlayerIndex ) ;
const thatPlayerInNextFrame = nextRenderFramePlayers [ playerId ] ;
[ thatPlayerInNextFrame . virtualGridX , thatPlayerInNextFrame . virtualGridY ] = self . polygonColliderAnchorToVirtualGridPos ( playerCollider . x - effPushbacks [ joinIndex - 1 ] [ 0 ] , playerCollider . y - effPushbacks [ joinIndex - 1 ] [ 1 ] , self . playerRichInfoArr [ j ] . colliderRadius , self . playerRichInfoArr [ j ] . colliderRadius ) ;
}
return window . pb . protos . RoomDownsyncFrame . create ( {
id : currRenderFrame . id + 1 ,
players : nextRenderFramePlayers ,
meleeBullets : nextRenderFrameMeleeBullets ,
} ) ;
} ,
2022-11-17 15:39:32 +00:00
} ) ;