chain_chomp.inc.c (19554B)
1 2 /** 3 * Behavior for bhvChainChomp, bhvChainChompChainPart, bhvWoodenPost, and bhvChainChompGate. 4 * bhvChainChomp spawns its bhvWoodenPost in its behavior script. It spawns 5 chain 5 * parts. Part 0 is the "pivot", which is positioned at the wooden post while 6 * the chomp is chained up. Parts 1-4 are the other parts, starting from the 7 * chain chomp and moving toward the pivot. 8 * Processing order is bhvWoodenPost, bhvChainChompGate, bhvChainChomp, bhvChainChompChainPart. 9 * The chain parts are processed starting at the post and ending at the chomp. 10 */ 11 12 /** 13 * Hitbox for chain chomp. 14 */ 15 static struct ObjectHitbox sChainChompHitbox = { 16 /* interactType: */ INTERACT_MR_BLIZZARD, 17 /* downOffset: */ 0, 18 /* damageOrCoinValue: */ 3, 19 /* health: */ 1, 20 /* numLootCoins: */ 0, 21 /* radius: */ 80, 22 /* height: */ 160, 23 /* hurtboxRadius: */ 80, 24 /* hurtboxHeight: */ 160, 25 }; 26 27 /** 28 * Update function for chain chomp part / pivot. 29 */ 30 void bhv_chain_chomp_chain_part_update(void) { 31 if (o->parentObj->oAction == CHAIN_CHOMP_ACT_UNLOAD_CHAIN) { 32 obj_mark_for_deletion(o); 33 } else if (o->oBhvParams2ndByte != CHAIN_CHOMP_CHAIN_PART_BP_PIVOT) { 34 struct ChainSegment *segment = &o->parentObj->oChainChompSegments[o->oBhvParams2ndByte]; 35 36 // Set position relative to the pivot 37 o->oPosX = o->parentObj->parentObj->oPosX + segment->posX; 38 o->oPosY = o->parentObj->parentObj->oPosY + segment->posY; 39 o->oPosZ = o->parentObj->parentObj->oPosZ + segment->posZ; 40 } else if (o->parentObj->oChainChompReleaseStatus != CHAIN_CHOMP_NOT_RELEASED) { 41 cur_obj_update_floor_and_walls(); 42 cur_obj_move_standard(78); 43 } 44 } 45 46 /** 47 * When mario gets close enough, allocate chain segments and spawn their objects. 48 */ 49 static void chain_chomp_act_uninitialized(void) { 50 struct ChainSegment *segments; 51 s32 i; 52 53 if (o->oDistanceToMario < 3000.0f) { 54 segments = mem_pool_alloc(gObjectMemoryPool, 5 * sizeof(struct ChainSegment)); 55 if (segments != NULL) { 56 // Each segment represents the offset of a chain part to the pivot. 57 // Segment 0 connects the pivot to the chain chomp itself. Segment 58 // 1 connects the pivot to the chain part next to the chain chomp 59 // (chain part 1), etc. 60 o->oChainChompSegments = segments; 61 for (i = 0; i <= 4; i++) { 62 chain_segment_init(&segments[i]); 63 } 64 65 cur_obj_set_pos_to_home(); 66 67 // Spawn the pivot and set to parent 68 if ((o->parentObj = 69 spawn_object(o, CHAIN_CHOMP_CHAIN_PART_BP_PIVOT, bhvChainChompChainPart)) 70 != NULL) { 71 // Spawn the non-pivot chain parts, starting from the chain 72 // chomp and moving toward the pivot 73 for (i = 1; i <= 4; i++) { 74 spawn_object_relative(i, 0, 0, 0, o, MODEL_METALLIC_BALL, bhvChainChompChainPart); 75 } 76 77 o->oAction = CHAIN_CHOMP_ACT_MOVE; 78 cur_obj_unhide(); 79 } 80 } 81 } 82 } 83 84 /** 85 * Apply gravity to each chain part, and cap its distance to the previous 86 * part as well as from the pivot. 87 */ 88 static void chain_chomp_update_chain_segments(void) { 89 struct ChainSegment *prevSegment; 90 struct ChainSegment *segment; 91 f32 offsetX; 92 f32 offsetY; 93 f32 offsetZ; 94 f32 offset; 95 f32 segmentVelY; 96 f32 maxTotalOffset; 97 s32 i; 98 99 if (o->oVelY < 0.0f) { 100 segmentVelY = o->oVelY; 101 } else { 102 segmentVelY = -20.0f; 103 } 104 105 // Segment 0 connects the pivot to the chain chomp itself, and segment i>0 106 // connects the pivot to chain part i (1 is closest to the chain chomp). 107 108 for (i = 1; i <= 4; i++) { 109 prevSegment = &o->oChainChompSegments[i - 1]; 110 segment = &o->oChainChompSegments[i]; 111 112 // Apply gravity 113 114 if ((segment->posY += segmentVelY) < 0.0f) { 115 segment->posY = 0.0f; 116 } 117 118 // Cap distance to previous chain part (so that the tail follows the 119 // chomp) 120 121 offsetX = segment->posX - prevSegment->posX; 122 offsetY = segment->posY - prevSegment->posY; 123 offsetZ = segment->posZ - prevSegment->posZ; 124 offset = sqrtf(offsetX * offsetX + offsetY * offsetY + offsetZ * offsetZ); 125 126 if (offset > o->oChainChompMaxDistBetweenChainParts) { 127 offset = o->oChainChompMaxDistBetweenChainParts / offset; 128 offsetX *= offset; 129 offsetY *= offset; 130 offsetZ *= offset; 131 } 132 133 // Cap distance to pivot (so that it stretches when the chomp moves far 134 // from the wooden post) 135 136 offsetX += prevSegment->posX; 137 offsetY += prevSegment->posY; 138 offsetZ += prevSegment->posZ; 139 offset = sqrtf(offsetX * offsetX + offsetY * offsetY + offsetZ * offsetZ); 140 141 maxTotalOffset = o->oChainChompMaxDistFromPivotPerChainPart * (5 - i); 142 if (offset > maxTotalOffset) { 143 offset = maxTotalOffset / offset; 144 offsetX *= offset; 145 offsetY *= offset; 146 offsetZ *= offset; 147 } 148 149 segment->posX = offsetX; 150 segment->posY = offsetY; 151 segment->posZ = offsetZ; 152 } 153 } 154 155 /** 156 * Lunging increases the maximum distance from the pivot and changes the maximum 157 * distance between chain parts. Restore these values to normal. 158 */ 159 static void chain_chomp_restore_normal_chain_lengths(void) { 160 approach_f32_ptr(&o->oChainChompMaxDistFromPivotPerChainPart, 750.0f / 5, 4.0f); 161 o->oChainChompMaxDistBetweenChainParts = o->oChainChompMaxDistFromPivotPerChainPart; 162 } 163 164 /** 165 * Turn toward mario. Wait a bit and then enter the lunging sub-action. 166 */ 167 static void chain_chomp_sub_act_turn(void) { 168 o->oGravity = -4.0f; 169 170 chain_chomp_restore_normal_chain_lengths(); 171 obj_move_pitch_approach(0, 0x100); 172 173 if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) { 174 cur_obj_rotate_yaw_toward(o->oAngleToMario, 0x400); 175 if (abs_angle_diff(o->oAngleToMario, o->oMoveAngleYaw) < 0x800) { 176 if (o->oTimer > 30) { 177 if (cur_obj_check_anim_frame(0)) { 178 cur_obj_reverse_animation(); 179 if (o->oTimer > 40) { 180 // Increase the maximum distance from the pivot and enter 181 // the lunging sub-action. 182 cur_obj_play_sound_2(SOUND_GENERAL_CHAIN_CHOMP2); 183 184 o->oSubAction = CHAIN_CHOMP_SUB_ACT_LUNGE; 185 o->oChainChompMaxDistFromPivotPerChainPart = 900.0f / 5; 186 187 o->oForwardVel = 140.0f; 188 o->oVelY = 20.0f; 189 o->oGravity = 0.0f; 190 o->oChainChompTargetPitch = obj_get_pitch_from_vel(); 191 } 192 } else { 193 o->oTimer--; 194 } 195 } else { 196 o->oForwardVel = 0.0f; 197 } 198 } else { 199 cur_obj_play_sound_2(SOUND_GENERAL_CHAIN_CHOMP1); 200 o->oForwardVel = 10.0f; 201 o->oVelY = 20.0f; 202 } 203 } else { 204 cur_obj_rotate_yaw_toward(o->oAngleToMario, 400); 205 o->oTimer = 0; 206 } 207 } 208 209 static void chain_chomp_sub_act_lunge(void) { 210 obj_face_pitch_approach(o->oChainChompTargetPitch, 0x400); 211 212 if (o->oForwardVel != 0.0f) { 213 f32 val04; 214 215 if (o->oChainChompRestrictedByChain == TRUE) { 216 o->oForwardVel = o->oVelY = 0.0f; 217 o->oChainChompUnk104 = 30.0f; 218 } 219 220 // TODO: What is this 221 if ((val04 = 900.0f - o->oChainChompDistToPivot) > 220.0f) { 222 val04 = 220.0f; 223 } 224 225 o->oChainChompMaxDistBetweenChainParts = 226 val04 / 220.0f * o->oChainChompMaxDistFromPivotPerChainPart; 227 o->oTimer = 0; 228 } else { 229 // Turn toward pivot 230 cur_obj_rotate_yaw_toward(atan2s(o->oChainChompSegments[0].posZ, o->oChainChompSegments[0].posX), 231 0x1000); 232 233 if (o->oChainChompUnk104 != 0.0f) { 234 approach_f32_ptr(&o->oChainChompUnk104, 0.0f, 0.8f); 235 } else { 236 o->oSubAction = CHAIN_CHOMP_SUB_ACT_TURN; 237 } 238 239 o->oChainChompMaxDistBetweenChainParts = o->oChainChompUnk104; 240 if (gGlobalTimer % 2 != 0) { 241 o->oChainChompMaxDistBetweenChainParts = -o->oChainChompUnk104; 242 } 243 } 244 245 if (o->oTimer < 30) { 246 cur_obj_reverse_animation(); 247 } 248 } 249 250 /** 251 * Fall to the ground and interrupt mario into a cutscene action. 252 */ 253 static void chain_chomp_released_trigger_cutscene(void) { 254 o->oForwardVel = 0.0f; 255 o->oGravity = -4.0f; 256 257 //! Can delay this if we get into a cutscene-unfriendly action after the 258 // last post ground pound and before this 259 if (set_mario_npc_dialog(MARIO_DIALOG_LOOK_UP) == MARIO_DIALOG_STATUS_SPEAK 260 && (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) && cutscene_object(CUTSCENE_STAR_SPAWN, o) == 1) { 261 o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_LUNGE_AROUND; 262 o->oTimer = 0; 263 } 264 } 265 266 /** 267 * Lunge 4 times, each time moving toward mario +/- 0x2000 angular units. 268 * Finally, begin a lunge toward x=1450, z=562 (near the gate). 269 */ 270 static void chain_chomp_released_lunge_around(void) { 271 chain_chomp_restore_normal_chain_lengths(); 272 273 // Finish bounce 274 if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) { 275 // Before first bounce, turn toward mario and wait 2 seconds 276 if (o->oChainChompNumLunges == 0) { 277 if (cur_obj_rotate_yaw_toward(o->oAngleToMario, 800)) { 278 if (o->oTimer > 60) { 279 o->oChainChompNumLunges++; 280 // enable wall collision 281 o->oWallHitboxRadius = 200.0f; 282 } 283 } else { 284 o->oTimer = 0; 285 } 286 } else { 287 if (++o->oChainChompNumLunges <= 5) { 288 cur_obj_play_sound_2(SOUND_GENERAL_CHAIN_CHOMP1); 289 o->oMoveAngleYaw = o->oAngleToMario + random_sign() * 0x2000; 290 o->oForwardVel = 30.0f; 291 o->oVelY = 50.0f; 292 } else { 293 o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_BREAK_GATE; 294 o->oHomeX = 1450.0f; 295 o->oHomeZ = 562.0f; 296 o->oMoveAngleYaw = cur_obj_angle_to_home(); 297 o->oForwardVel = cur_obj_lateral_dist_to_home() / 8; 298 o->oVelY = 50.0f; 299 } 300 } 301 } 302 } 303 304 /** 305 * Continue lunging until a wall collision occurs. Mark the gate as destroyed, 306 * wait for the chain chomp to land, and then begin a jump toward the final 307 * target, x=3288, z=-1770. 308 */ 309 static void chain_chomp_released_break_gate(void) { 310 if (!o->oChainChompHitGate) { 311 // If hit wall, assume it's the gate and bounce off of it 312 //! The wall may not be the gate 313 //! If the chain chomp gets stuck, it may never hit a wall, resulting 314 // in a softlock 315 if (o->oMoveFlags & OBJ_MOVE_HIT_WALL) { 316 o->oChainChompHitGate = TRUE; 317 o->oMoveAngleYaw = cur_obj_reflect_move_angle_off_wall(); 318 o->oForwardVel *= 0.4f; 319 } 320 } else if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) { 321 o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_JUMP_AWAY; 322 o->oHomeX = 3288.0f; 323 o->oHomeZ = -1770.0f; 324 o->oMoveAngleYaw = cur_obj_angle_to_home(); 325 o->oForwardVel = cur_obj_lateral_dist_to_home() / 50.0f; 326 o->oVelY = 120.0f; 327 } 328 } 329 330 /** 331 * Wait until the chain chomp lands. 332 */ 333 static void chain_chomp_released_jump_away(void) { 334 if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) { 335 gObjCutsceneDone = TRUE; 336 o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_END_CUTSCENE; 337 } 338 } 339 340 /** 341 * Release mario and transition to the unload chain action. 342 */ 343 static void chain_chomp_released_end_cutscene(void) { 344 if (cutscene_object(CUTSCENE_STAR_SPAWN, o) == -1) { 345 set_mario_npc_dialog(MARIO_DIALOG_STOP); 346 o->oAction = CHAIN_CHOMP_ACT_UNLOAD_CHAIN; 347 } 348 } 349 350 /** 351 * All chain chomp movement behavior, including the cutscene after being 352 * released. 353 */ 354 static void chain_chomp_act_move(void) { 355 f32 maxDistToPivot; 356 357 // Unload chain if mario is far enough 358 if (o->oChainChompReleaseStatus == CHAIN_CHOMP_NOT_RELEASED && o->oDistanceToMario > 4000.0f) { 359 o->oAction = CHAIN_CHOMP_ACT_UNLOAD_CHAIN; 360 o->oForwardVel = o->oVelY = 0.0f; 361 } else { 362 cur_obj_update_floor_and_walls(); 363 364 switch (o->oChainChompReleaseStatus) { 365 case CHAIN_CHOMP_NOT_RELEASED: 366 switch (o->oSubAction) { 367 case CHAIN_CHOMP_SUB_ACT_TURN: 368 chain_chomp_sub_act_turn(); 369 break; 370 case CHAIN_CHOMP_SUB_ACT_LUNGE: 371 chain_chomp_sub_act_lunge(); 372 break; 373 } 374 break; 375 case CHAIN_CHOMP_RELEASED_TRIGGER_CUTSCENE: 376 chain_chomp_released_trigger_cutscene(); 377 break; 378 case CHAIN_CHOMP_RELEASED_LUNGE_AROUND: 379 chain_chomp_released_lunge_around(); 380 break; 381 case CHAIN_CHOMP_RELEASED_BREAK_GATE: 382 chain_chomp_released_break_gate(); 383 break; 384 case CHAIN_CHOMP_RELEASED_JUMP_AWAY: 385 chain_chomp_released_jump_away(); 386 break; 387 case CHAIN_CHOMP_RELEASED_END_CUTSCENE: 388 chain_chomp_released_end_cutscene(); 389 break; 390 } 391 392 cur_obj_move_standard(78); 393 394 // Segment 0 connects the pivot to the chain chomp itself 395 o->oChainChompSegments[0].posX = o->oPosX - o->parentObj->oPosX; 396 o->oChainChompSegments[0].posY = o->oPosY - o->parentObj->oPosY; 397 o->oChainChompSegments[0].posZ = o->oPosZ - o->parentObj->oPosZ; 398 399 o->oChainChompDistToPivot = 400 sqrtf(o->oChainChompSegments[0].posX * o->oChainChompSegments[0].posX 401 + o->oChainChompSegments[0].posY * o->oChainChompSegments[0].posY 402 + o->oChainChompSegments[0].posZ * o->oChainChompSegments[0].posZ); 403 404 // If the chain is fully stretched 405 maxDistToPivot = o->oChainChompMaxDistFromPivotPerChainPart * 5; 406 if (o->oChainChompDistToPivot > maxDistToPivot) { 407 f32 ratio = maxDistToPivot / o->oChainChompDistToPivot; 408 o->oChainChompDistToPivot = maxDistToPivot; 409 410 o->oChainChompSegments[0].posX *= ratio; 411 o->oChainChompSegments[0].posY *= ratio; 412 o->oChainChompSegments[0].posZ *= ratio; 413 414 if (o->oChainChompReleaseStatus == CHAIN_CHOMP_NOT_RELEASED) { 415 // Restrict chain chomp position 416 o->oPosX = o->parentObj->oPosX + o->oChainChompSegments[0].posX; 417 o->oPosY = o->parentObj->oPosY + o->oChainChompSegments[0].posY; 418 o->oPosZ = o->parentObj->oPosZ + o->oChainChompSegments[0].posZ; 419 420 o->oChainChompRestrictedByChain = TRUE; 421 } else { 422 // Move pivot like the chain chomp is pulling it along 423 f32 oldPivotY = o->parentObj->oPosY; 424 425 o->parentObj->oPosX = o->oPosX - o->oChainChompSegments[0].posX; 426 o->parentObj->oPosY = o->oPosY - o->oChainChompSegments[0].posY; 427 o->parentObj->oVelY = o->parentObj->oPosY - oldPivotY; 428 o->parentObj->oPosZ = o->oPosZ - o->oChainChompSegments[0].posZ; 429 } 430 } else { 431 o->oChainChompRestrictedByChain = FALSE; 432 } 433 434 chain_chomp_update_chain_segments(); 435 436 // Begin a lunge if mario tries to attack 437 if (obj_check_attacks(&sChainChompHitbox, o->oAction) != 0) { 438 o->oSubAction = CHAIN_CHOMP_SUB_ACT_LUNGE; 439 o->oChainChompMaxDistFromPivotPerChainPart = 900.0f / 5; 440 o->oForwardVel = 0.0f; 441 o->oVelY = 300.0f; 442 o->oGravity = -4.0f; 443 o->oChainChompTargetPitch = -0x3000; 444 } 445 } 446 } 447 448 /** 449 * Hide and free the chain chomp segments. The chain objects will unload 450 * themselves when they see that the chain chomp is in this action. 451 */ 452 static void chain_chomp_act_unload_chain(void) { 453 cur_obj_hide(); 454 mem_pool_free(gObjectMemoryPool, o->oChainChompSegments); 455 456 o->oAction = CHAIN_CHOMP_ACT_UNINITIALIZED; 457 458 if (o->oChainChompReleaseStatus != CHAIN_CHOMP_NOT_RELEASED) { 459 obj_mark_for_deletion(o); 460 } 461 } 462 463 /** 464 * Update function for chain chomp. 465 */ 466 void bhv_chain_chomp_update(void) { 467 switch (o->oAction) { 468 case CHAIN_CHOMP_ACT_UNINITIALIZED: 469 chain_chomp_act_uninitialized(); 470 break; 471 case CHAIN_CHOMP_ACT_MOVE: 472 chain_chomp_act_move(); 473 break; 474 case CHAIN_CHOMP_ACT_UNLOAD_CHAIN: 475 chain_chomp_act_unload_chain(); 476 break; 477 } 478 } 479 480 /** 481 * Update function for wooden post. 482 */ 483 void bhv_wooden_post_update(void) { 484 // When ground pounded by mario, drop by -45 + -20 485 if (!o->oWoodenPostMarioPounding) { 486 if ((o->oWoodenPostMarioPounding = cur_obj_is_mario_ground_pounding_platform())) { 487 cur_obj_play_sound_2(SOUND_GENERAL_POUND_WOOD_POST); 488 o->oWoodenPostSpeedY = -70.0f; 489 } 490 } else if (approach_f32_ptr(&o->oWoodenPostSpeedY, 0.0f, 25.0f)) { 491 // Stay still until mario is done ground pounding 492 o->oWoodenPostMarioPounding = cur_obj_is_mario_ground_pounding_platform(); 493 } else if ((o->oWoodenPostOffsetY += o->oWoodenPostSpeedY) < -190.0f) { 494 // Once pounded, if this is the chain chomp's post, release the chain 495 // chomp 496 o->oWoodenPostOffsetY = -190.0f; 497 if (o->parentObj != o) { 498 play_puzzle_jingle(); 499 o->parentObj->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_TRIGGER_CUTSCENE; 500 o->parentObj = o; 501 } 502 } 503 504 if (o->oWoodenPostOffsetY != 0.0f) { 505 o->oPosY = o->oHomeY + o->oWoodenPostOffsetY; 506 } else if (!(o->oBhvParams & WOODEN_POST_BP_NO_COINS_MASK)) { 507 // Reset the timer once mario is far enough 508 if (o->oDistanceToMario > 400.0f) { 509 o->oTimer = o->oWoodenPostTotalMarioAngle = 0; 510 } else { 511 // When mario runs around the post 3 times within 200 frames, spawn 512 // coins 513 o->oWoodenPostTotalMarioAngle += (s16)(o->oAngleToMario - o->oWoodenPostPrevAngleToMario); 514 if (absi(o->oWoodenPostTotalMarioAngle) > 0x30000 && o->oTimer < 200) { 515 obj_spawn_loot_yellow_coins(o, 5, 20.0f); 516 set_object_respawn_info_bits(o, 1); 517 } 518 } 519 520 o->oWoodenPostPrevAngleToMario = o->oAngleToMario; 521 } 522 } 523 524 /** 525 * Init function for chain chomp gate. 526 */ 527 void bhv_chain_chomp_gate_init(void) { 528 o->parentObj = cur_obj_nearest_object_with_behavior(bhvChainChomp); 529 } 530 531 /** 532 * Update function for chain chomp gate 533 */ 534 void bhv_chain_chomp_gate_update(void) { 535 if (o->parentObj->oChainChompHitGate) { 536 spawn_mist_particles_with_sound(SOUND_GENERAL_WALL_EXPLOSION); 537 set_camera_shake_from_point(SHAKE_POS_SMALL, o->oPosX, o->oPosY, o->oPosZ); 538 spawn_mist_particles_variable(0, 127, 200.0f); 539 spawn_triangle_break_particles(30, MODEL_DIRT_ANIMATION, 3.0f, 4); 540 obj_mark_for_deletion(o); 541 } 542 }