mario_misc.c (25419B)
1 #include <PR/ultratypes.h> 2 3 #include "sm64.h" 4 #include "area.h" 5 #include "audio/external.h" 6 #include "behavior_actions.h" 7 #include "behavior_data.h" 8 #include "camera.h" 9 #include "dialog_ids.h" 10 #include "engine/behavior_script.h" 11 #include "engine/graph_node.h" 12 #include "engine/math_util.h" 13 #include "envfx_snow.h" 14 #include "game_init.h" 15 #include "goddard/renderer.h" 16 #include "interaction.h" 17 #include "level_update.h" 18 #include "mario_actions_cutscene.h" 19 #include "mario_misc.h" 20 #include "memory.h" 21 #include "object_helpers.h" 22 #include "object_list_processor.h" 23 #include "rendering_graph_node.h" 24 #include "save_file.h" 25 #include "skybox.h" 26 #include "sound_init.h" 27 28 #define TOAD_STAR_1_REQUIREMENT 12 29 #define TOAD_STAR_2_REQUIREMENT 25 30 #define TOAD_STAR_3_REQUIREMENT 35 31 32 #define TOAD_STAR_1_DIALOG DIALOG_082 33 #define TOAD_STAR_2_DIALOG DIALOG_076 34 #define TOAD_STAR_3_DIALOG DIALOG_083 35 36 #define TOAD_STAR_1_DIALOG_AFTER DIALOG_154 37 #define TOAD_STAR_2_DIALOG_AFTER DIALOG_155 38 #define TOAD_STAR_3_DIALOG_AFTER DIALOG_156 39 40 enum ToadMessageStates { 41 TOAD_MESSAGE_FADED, 42 TOAD_MESSAGE_OPAQUE, 43 TOAD_MESSAGE_OPACIFYING, 44 TOAD_MESSAGE_FADING, 45 TOAD_MESSAGE_TALKING 46 }; 47 48 enum UnlockDoorStarStates { 49 UNLOCK_DOOR_STAR_RISING, 50 UNLOCK_DOOR_STAR_WAITING, 51 UNLOCK_DOOR_STAR_SPAWNING_PARTICLES, 52 UNLOCK_DOOR_STAR_DONE 53 }; 54 55 /** 56 * The eye texture on succesive frames of Mario's blink animation. 57 * He intentionally blinks twice each time. 58 */ 59 static s8 gMarioBlinkAnimation[7] = { 1, 2, 1, 0, 1, 2, 1 }; 60 61 /** 62 * The scale values per frame for Mario's foot/hand for his attack animation 63 * There are 3 scale animations in groups of 6 frames. 64 * The first animation starts at frame index 3 and goes down, the others start at frame index 5. 65 * The values get divided by 10 before assigning, so e.g. 12 gives a scale factor 1.2. 66 * All combined, this means e.g. the first animation scales Mario's fist by {2.4, 1.6, 1.2, 1.0} on 67 * successive frames. 68 */ 69 static s8 gMarioAttackScaleAnimation[3 * 6] = { 70 10, 12, 16, 24, 10, 10, 10, 14, 20, 30, 10, 10, 10, 16, 20, 26, 26, 20, 71 }; 72 73 struct MarioBodyState gBodyStates[2]; // 2nd is never accessed in practice, most likely Luigi related 74 struct GraphNodeObject gMirrorMario; // copy of Mario's geo node for drawing mirror Mario 75 76 // This whole file is weirdly organized. It has to be the same file due 77 // to rodata boundaries and function aligns, which means the programmer 78 // treated this like a "misc" file for vaguely Mario related things 79 // (message NPC related things, the Mario head geo, and Mario geo 80 // functions) 81 82 /** 83 * Geo node script that draws Mario's head on the title screen. 84 */ 85 Gfx *geo_draw_mario_head_goddard(s32 callContext, struct GraphNode *node, Mat4 *c) { 86 Gfx *gfx = NULL; 87 s16 sfx = 0; 88 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 89 UNUSED Mat4 *transform = c; 90 91 if (callContext == GEO_CONTEXT_RENDER) { 92 if (gPlayer1Controller->controllerData != NULL && !gWarpTransition.isActive) { 93 gd_copy_p1_contpad(gPlayer1Controller->controllerData); 94 } 95 gfx = (Gfx *) PHYSICAL_TO_VIRTUAL(gdm_gettestdl(asGenerated->parameter)); 96 gGoddardVblankCallback = gd_vblank; 97 sfx = gd_sfx_to_play(); 98 play_menu_sounds(sfx); 99 } 100 return gfx; 101 } 102 103 static void toad_message_faded(void) { 104 if (gCurrentObject->oDistanceToMario > 700.0f) { 105 gCurrentObject->oToadMessageRecentlyTalked = FALSE; 106 } 107 if (!gCurrentObject->oToadMessageRecentlyTalked && gCurrentObject->oDistanceToMario < 600.0f) { 108 gCurrentObject->oToadMessageState = TOAD_MESSAGE_OPACIFYING; 109 } 110 } 111 112 static void toad_message_opaque(void) { 113 if (gCurrentObject->oDistanceToMario > 700.0f) { 114 gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADING; 115 } else if (!gCurrentObject->oToadMessageRecentlyTalked) { 116 gCurrentObject->oInteractionSubtype = INT_SUBTYPE_NPC; 117 if (gCurrentObject->oInteractStatus & INT_STATUS_INTERACTED) { 118 gCurrentObject->oInteractStatus = 0; 119 gCurrentObject->oToadMessageState = TOAD_MESSAGE_TALKING; 120 play_toads_jingle(); 121 } 122 } 123 } 124 125 static void toad_message_talking(void) { 126 if (cur_obj_update_dialog_with_cutscene(MARIO_DIALOG_LOOK_DOWN, 127 DIALOG_FLAG_TURN_TO_MARIO, CUTSCENE_DIALOG, gCurrentObject->oToadMessageDialogID)) { 128 gCurrentObject->oToadMessageRecentlyTalked = TRUE; 129 gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADING; 130 switch (gCurrentObject->oToadMessageDialogID) { 131 case TOAD_STAR_1_DIALOG: 132 gCurrentObject->oToadMessageDialogID = TOAD_STAR_1_DIALOG_AFTER; 133 bhv_spawn_star_no_level_exit(STAR_INDEX_ACT_1); 134 break; 135 case TOAD_STAR_2_DIALOG: 136 gCurrentObject->oToadMessageDialogID = TOAD_STAR_2_DIALOG_AFTER; 137 bhv_spawn_star_no_level_exit(STAR_INDEX_ACT_2); 138 break; 139 case TOAD_STAR_3_DIALOG: 140 gCurrentObject->oToadMessageDialogID = TOAD_STAR_3_DIALOG_AFTER; 141 bhv_spawn_star_no_level_exit(STAR_INDEX_ACT_3); 142 break; 143 } 144 } 145 } 146 147 static void toad_message_opacifying(void) { 148 if ((gCurrentObject->oOpacity += 6) == 255) { 149 gCurrentObject->oToadMessageState = TOAD_MESSAGE_OPAQUE; 150 } 151 } 152 153 static void toad_message_fading(void) { 154 if ((gCurrentObject->oOpacity -= 6) == 81) { 155 gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADED; 156 } 157 } 158 159 void bhv_toad_message_loop(void) { 160 if (gCurrentObject->header.gfx.node.flags & GRAPH_RENDER_ACTIVE) { 161 gCurrentObject->oInteractionSubtype = 0; 162 switch (gCurrentObject->oToadMessageState) { 163 case TOAD_MESSAGE_FADED: 164 toad_message_faded(); 165 break; 166 case TOAD_MESSAGE_OPAQUE: 167 toad_message_opaque(); 168 break; 169 case TOAD_MESSAGE_OPACIFYING: 170 toad_message_opacifying(); 171 break; 172 case TOAD_MESSAGE_FADING: 173 toad_message_fading(); 174 break; 175 case TOAD_MESSAGE_TALKING: 176 toad_message_talking(); 177 break; 178 } 179 } 180 } 181 182 void bhv_toad_message_init(void) { 183 s32 saveFlags = save_file_get_flags(); 184 s32 starCount = save_file_get_total_star_count(gCurrSaveFileNum - 1, COURSE_MIN - 1, COURSE_MAX - 1); 185 s32 dialogID = (gCurrentObject->oBhvParams >> 24) & 0xFF; 186 s32 enoughStars = TRUE; 187 188 switch (dialogID) { 189 case TOAD_STAR_1_DIALOG: 190 enoughStars = (starCount >= TOAD_STAR_1_REQUIREMENT); 191 if (saveFlags & SAVE_FLAG_COLLECTED_TOAD_STAR_1) { 192 dialogID = TOAD_STAR_1_DIALOG_AFTER; 193 } 194 break; 195 case TOAD_STAR_2_DIALOG: 196 enoughStars = (starCount >= TOAD_STAR_2_REQUIREMENT); 197 if (saveFlags & SAVE_FLAG_COLLECTED_TOAD_STAR_2) { 198 dialogID = TOAD_STAR_2_DIALOG_AFTER; 199 } 200 break; 201 case TOAD_STAR_3_DIALOG: 202 enoughStars = (starCount >= TOAD_STAR_3_REQUIREMENT); 203 if (saveFlags & SAVE_FLAG_COLLECTED_TOAD_STAR_3) { 204 dialogID = TOAD_STAR_3_DIALOG_AFTER; 205 } 206 break; 207 } 208 209 if (enoughStars) { 210 gCurrentObject->oToadMessageDialogID = dialogID; 211 gCurrentObject->oToadMessageRecentlyTalked = FALSE; 212 gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADED; 213 gCurrentObject->oOpacity = 81; 214 } else { 215 obj_mark_for_deletion(gCurrentObject); 216 } 217 } 218 219 static void star_door_unlock_spawn_particles(s16 angleOffset) { 220 struct Object *sparkleParticle = spawn_object(gCurrentObject, 0, bhvSparkleSpawn); 221 222 sparkleParticle->oPosX += 223 100.0f * sins((gCurrentObject->oUnlockDoorStarTimer * 0x2800) + angleOffset); 224 sparkleParticle->oPosZ += 225 100.0f * coss((gCurrentObject->oUnlockDoorStarTimer * 0x2800) + angleOffset); 226 // Particles are spawned lower each frame 227 sparkleParticle->oPosY -= gCurrentObject->oUnlockDoorStarTimer * 10.0f; 228 } 229 230 void bhv_unlock_door_star_init(void) { 231 gCurrentObject->oUnlockDoorStarState = UNLOCK_DOOR_STAR_RISING; 232 gCurrentObject->oUnlockDoorStarTimer = 0; 233 gCurrentObject->oUnlockDoorStarYawVel = 0x1000; 234 gCurrentObject->oPosX += 30.0f * sins(gMarioState->faceAngle[1] - 0x4000); 235 gCurrentObject->oPosY += 160.0f; 236 gCurrentObject->oPosZ += 30.0f * coss(gMarioState->faceAngle[1] - 0x4000); 237 gCurrentObject->oMoveAngleYaw = 0x7800; 238 obj_scale(gCurrentObject, 0.5f); 239 } 240 241 void bhv_unlock_door_star_loop(void) { 242 UNUSED u8 filler1[4]; 243 s16 prevYaw = gCurrentObject->oMoveAngleYaw; 244 UNUSED u8 filler2[4]; 245 246 // Speed up the star every frame 247 if (gCurrentObject->oUnlockDoorStarYawVel < 0x2400) { 248 gCurrentObject->oUnlockDoorStarYawVel += 0x60; 249 } 250 switch (gCurrentObject->oUnlockDoorStarState) { 251 case UNLOCK_DOOR_STAR_RISING: 252 gCurrentObject->oPosY += 3.4f; // Raise the star up in the air 253 gCurrentObject->oMoveAngleYaw += 254 gCurrentObject->oUnlockDoorStarYawVel; // Apply yaw velocity 255 obj_scale(gCurrentObject, gCurrentObject->oUnlockDoorStarTimer / 50.0f 256 + 0.5f); // Scale the star to be bigger 257 if (++gCurrentObject->oUnlockDoorStarTimer == 30) { 258 gCurrentObject->oUnlockDoorStarTimer = 0; 259 gCurrentObject->oUnlockDoorStarState++; // Sets state to UNLOCK_DOOR_STAR_WAITING 260 } 261 break; 262 case UNLOCK_DOOR_STAR_WAITING: 263 gCurrentObject->oMoveAngleYaw += 264 gCurrentObject->oUnlockDoorStarYawVel; // Apply yaw velocity 265 if (++gCurrentObject->oUnlockDoorStarTimer == 30) { 266 play_sound(SOUND_MENU_STAR_SOUND, 267 gCurrentObject->header.gfx.cameraToObject); // Play final sound 268 cur_obj_hide(); // Hide the object 269 gCurrentObject->oUnlockDoorStarTimer = 0; 270 gCurrentObject 271 ->oUnlockDoorStarState++; // Sets state to UNLOCK_DOOR_STAR_SPAWNING_PARTICLES 272 } 273 break; 274 case UNLOCK_DOOR_STAR_SPAWNING_PARTICLES: 275 // Spawn two particles, opposite sides of the star. 276 star_door_unlock_spawn_particles(0); 277 star_door_unlock_spawn_particles(0x8000); 278 if (gCurrentObject->oUnlockDoorStarTimer++ == 20) { 279 gCurrentObject->oUnlockDoorStarTimer = 0; 280 gCurrentObject->oUnlockDoorStarState++; // Sets state to UNLOCK_DOOR_STAR_DONE 281 } 282 break; 283 case UNLOCK_DOOR_STAR_DONE: // The object stays loaded for an additional 50 frames so that the 284 // sound doesn't immediately stop. 285 if (gCurrentObject->oUnlockDoorStarTimer++ == 50) { 286 obj_mark_for_deletion(gCurrentObject); 287 } 288 break; 289 } 290 // Checks if the angle has cycled back to 0. 291 // This means that the code will execute when the star completes a full revolution. 292 if (prevYaw > (s16) gCurrentObject->oMoveAngleYaw) { 293 play_sound( 294 SOUND_GENERAL_SHORT_STAR, 295 gCurrentObject->header.gfx.cameraToObject); // Play a sound every time the star spins once 296 } 297 } 298 299 /** 300 * Generate a display list that sets the correct blend mode and color for mirror Mario. 301 */ 302 static Gfx *make_gfx_mario_alpha(struct GraphNodeGenerated *node, s16 alpha) { 303 Gfx *gfx; 304 Gfx *gfxHead = NULL; 305 306 if (alpha == 255) { 307 node->fnNode.node.flags = (node->fnNode.node.flags & 0xFF) | (LAYER_OPAQUE << 8); 308 gfxHead = alloc_display_list(2 * sizeof(*gfxHead)); 309 gfx = gfxHead; 310 } else { 311 node->fnNode.node.flags = (node->fnNode.node.flags & 0xFF) | (LAYER_TRANSPARENT << 8); 312 gfxHead = alloc_display_list(3 * sizeof(*gfxHead)); 313 gfx = gfxHead; 314 gDPSetAlphaCompare(gfx++, G_AC_DITHER); 315 } 316 gDPSetEnvColor(gfx++, 255, 255, 255, alpha); 317 gSPEndDisplayList(gfx); 318 return gfxHead; 319 } 320 321 /** 322 * Sets the correct blend mode and color for mirror Mario. 323 */ 324 Gfx *geo_mirror_mario_set_alpha(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 325 UNUSED u8 filler1[4]; 326 Gfx *gfx = NULL; 327 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 328 struct MarioBodyState *bodyState = &gBodyStates[asGenerated->parameter]; 329 s16 alpha; 330 UNUSED u8 filler2[4]; 331 332 if (callContext == GEO_CONTEXT_RENDER) { 333 alpha = (bodyState->modelState & 0x100) ? (bodyState->modelState & 0xFF) : 255; 334 gfx = make_gfx_mario_alpha(asGenerated, alpha); 335 } 336 return gfx; 337 } 338 339 /** 340 * Determines if Mario is standing or running for the level of detail of his model. 341 * If Mario is standing still, he is always high poly. If he is running, 342 * his level of detail depends on the distance to the camera. 343 */ 344 Gfx *geo_switch_mario_stand_run(s32 callContext, struct GraphNode *node, UNUSED Mat4 *mtx) { 345 struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node; 346 struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases]; 347 348 if (callContext == GEO_CONTEXT_RENDER) { 349 // assign result. 0 if moving, 1 if stationary. 350 switchCase->selectedCase = ((bodyState->action & ACT_FLAG_STATIONARY) == 0); 351 } 352 return NULL; 353 } 354 355 /** 356 * Geo node script that makes Mario blink 357 */ 358 Gfx *geo_switch_mario_eyes(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 359 struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node; 360 struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases]; 361 s16 blinkFrame; 362 363 if (callContext == GEO_CONTEXT_RENDER) { 364 if (bodyState->eyeState == 0) { 365 blinkFrame = ((switchCase->numCases * 32 + gAreaUpdateCounter) >> 1) & 0x1F; 366 if (blinkFrame < 7) { 367 switchCase->selectedCase = gMarioBlinkAnimation[blinkFrame]; 368 } else { 369 switchCase->selectedCase = 0; 370 } 371 } else { 372 switchCase->selectedCase = bodyState->eyeState - 1; 373 } 374 } 375 return NULL; 376 } 377 378 /** 379 * Makes Mario's upper body tilt depending on the rotation stored in his bodyState 380 */ 381 Gfx *geo_mario_tilt_torso(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 382 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 383 struct MarioBodyState *bodyState = &gBodyStates[asGenerated->parameter]; 384 s32 action = bodyState->action; 385 386 if (callContext == GEO_CONTEXT_RENDER) { 387 struct GraphNodeRotation *rotNode = (struct GraphNodeRotation *) node->next; 388 389 if (action != ACT_BUTT_SLIDE && action != ACT_HOLD_BUTT_SLIDE && action != ACT_WALKING 390 && action != ACT_RIDING_SHELL_GROUND) { 391 vec3s_copy(bodyState->torsoAngle, gVec3sZero); 392 } 393 rotNode->rotation[0] = bodyState->torsoAngle[1]; 394 rotNode->rotation[1] = bodyState->torsoAngle[2]; 395 rotNode->rotation[2] = bodyState->torsoAngle[0]; 396 } 397 return NULL; 398 } 399 400 /** 401 * Makes Mario's head rotate with the camera angle when in C-up mode 402 */ 403 Gfx *geo_mario_head_rotation(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 404 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 405 struct MarioBodyState *bodyState = &gBodyStates[asGenerated->parameter]; 406 s32 action = bodyState->action; 407 408 if (callContext == GEO_CONTEXT_RENDER) { 409 struct GraphNodeRotation *rotNode = (struct GraphNodeRotation *) node->next; 410 struct Camera *camera = gCurGraphNodeCamera->config.camera; 411 412 if (camera->mode == CAMERA_MODE_C_UP) { 413 rotNode->rotation[0] = gPlayerCameraState->headRotation[1]; 414 rotNode->rotation[2] = gPlayerCameraState->headRotation[0]; 415 } else if (action & ACT_FLAG_WATER_OR_TEXT) { 416 rotNode->rotation[0] = bodyState->headAngle[1]; 417 rotNode->rotation[1] = bodyState->headAngle[2]; 418 rotNode->rotation[2] = bodyState->headAngle[0]; 419 } else { 420 vec3s_set(bodyState->headAngle, 0, 0, 0); 421 vec3s_set(rotNode->rotation, 0, 0, 0); 422 } 423 } 424 return NULL; 425 } 426 427 /** 428 * Switch between hand models. 429 * Possible options are described in the MarioHandGSCId enum. 430 */ 431 Gfx *geo_switch_mario_hand(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 432 struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node; 433 struct MarioBodyState *bodyState = &gBodyStates[0]; 434 435 if (callContext == GEO_CONTEXT_RENDER) { 436 if (bodyState->handState == MARIO_HAND_FISTS) { 437 // switch between fists (0) and open (1) 438 switchCase->selectedCase = ((bodyState->action & ACT_FLAG_SWIMMING_OR_FLYING) != 0); 439 } else { 440 if (switchCase->numCases == 0) { 441 switchCase->selectedCase = 442 (bodyState->handState < 5) ? bodyState->handState : MARIO_HAND_OPEN; 443 } else { 444 switchCase->selectedCase = 445 (bodyState->handState < 2) ? bodyState->handState : MARIO_HAND_FISTS; 446 } 447 } 448 } 449 return NULL; 450 } 451 452 /** 453 * Increase Mario's hand / foot size when he punches / kicks. 454 * Since animation geo nodes only support rotation, this scaling animation 455 * was scripted separately. The node with this script should be placed before 456 * a scaling node containing the hand / foot geo layout. 457 * ! Since the animation gets updated in GEO_CONTEXT_RENDER, drawing Mario multiple times 458 * (such as in the mirror room) results in a faster and desynced punch / kick animation. 459 */ 460 Gfx *geo_mario_hand_foot_scaler(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 461 static s16 sMarioAttackAnimCounter = 0; 462 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 463 struct GraphNodeScale *scaleNode = (struct GraphNodeScale *) node->next; 464 struct MarioBodyState *bodyState = &gBodyStates[0]; 465 466 if (callContext == GEO_CONTEXT_RENDER) { 467 scaleNode->scale = 1.0f; 468 if (asGenerated->parameter == bodyState->punchState >> 6) { 469 if (sMarioAttackAnimCounter != gAreaUpdateCounter && (bodyState->punchState & 0x3F) > 0) { 470 bodyState->punchState -= 1; 471 sMarioAttackAnimCounter = gAreaUpdateCounter; 472 } 473 scaleNode->scale = 474 gMarioAttackScaleAnimation[asGenerated->parameter * 6 + (bodyState->punchState & 0x3F)] 475 / 10.0f; 476 } 477 } 478 return NULL; 479 } 480 481 /** 482 * Switch between normal cap, wing cap, vanish cap and metal cap. 483 */ 484 Gfx *geo_switch_mario_cap_effect(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 485 struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node; 486 struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases]; 487 488 if (callContext == GEO_CONTEXT_RENDER) { 489 switchCase->selectedCase = bodyState->modelState >> 8; 490 } 491 return NULL; 492 } 493 494 /** 495 * Determine whether Mario's head is drawn with or without a cap on. 496 * Also sets the visibility of the wing cap wings on or off. 497 */ 498 Gfx *geo_switch_mario_cap_on_off(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 499 struct GraphNode *next = node->next; 500 struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node; 501 struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases]; 502 503 if (callContext == GEO_CONTEXT_RENDER) { 504 switchCase->selectedCase = bodyState->capState & 1; 505 while (next != node) { 506 if (next->type == GRAPH_NODE_TYPE_TRANSLATION_ROTATION) { 507 if (bodyState->capState & 2) { 508 next->flags |= GRAPH_RENDER_ACTIVE; 509 } else { 510 next->flags &= ~GRAPH_RENDER_ACTIVE; 511 } 512 } 513 next = next->next; 514 } 515 } 516 return NULL; 517 } 518 519 /** 520 * Geo node script that makes the wings on Mario's wing cap flap. 521 * Should be placed before a rotation node. 522 */ 523 Gfx *geo_mario_rotate_wing_cap_wings(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 524 s16 rotX; 525 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 526 527 if (callContext == GEO_CONTEXT_RENDER) { 528 struct GraphNodeRotation *rotNode = (struct GraphNodeRotation *) node->next; 529 530 if (!gBodyStates[asGenerated->parameter >> 1].wingFlutter) { 531 rotX = (coss((gAreaUpdateCounter & 0xF) << 12) + 1.0f) * 4096.0f; 532 } else { 533 rotX = (coss((gAreaUpdateCounter & 7) << 13) + 1.0f) * 6144.0f; 534 } 535 if (!(asGenerated->parameter & 1)) { 536 rotNode->rotation[0] = -rotX; 537 } else { 538 rotNode->rotation[0] = rotX; 539 } 540 } 541 return NULL; 542 } 543 544 /** 545 * Geo node that updates the held object node and the HOLP. 546 */ 547 Gfx *geo_switch_mario_hand_grab_pos(s32 callContext, struct GraphNode *b, Mat4 *mtx) { 548 struct GraphNodeHeldObject *asHeldObj = (struct GraphNodeHeldObject *) b; 549 Mat4 *curTransform = mtx; 550 struct MarioState *marioState = &gMarioStates[asHeldObj->playerIndex]; 551 552 if (callContext == GEO_CONTEXT_RENDER) { 553 asHeldObj->objNode = NULL; 554 if (marioState->heldObj != NULL) { 555 asHeldObj->objNode = marioState->heldObj; 556 switch (marioState->marioBodyState->grabPos) { 557 case GRAB_POS_LIGHT_OBJ: 558 if (marioState->action & ACT_FLAG_THROWING) { 559 vec3s_set(asHeldObj->translation, 50, 0, 0); 560 } else { 561 vec3s_set(asHeldObj->translation, 50, 0, 110); 562 } 563 break; 564 case GRAB_POS_HEAVY_OBJ: 565 vec3s_set(asHeldObj->translation, 145, -173, 180); 566 break; 567 case GRAB_POS_BOWSER: 568 vec3s_set(asHeldObj->translation, 80, -270, 1260); 569 break; 570 } 571 } 572 } else if (callContext == GEO_CONTEXT_HELD_OBJ) { 573 // ! The HOLP is set here, which is why it only updates when the held object is drawn. 574 // This is why it won't update during a pause buffered hitstun or when the camera is very far 575 // away. 576 get_pos_from_transform_mtx(marioState->marioBodyState->heldObjLastPosition, *curTransform, 577 *gCurGraphNodeCamera->matrixPtr); 578 } 579 return NULL; 580 } 581 582 // X position of the mirror 583 #define MIRROR_X 4331.53 584 585 /** 586 * Geo node that creates a clone of Mario's geo node and updates it to becomes 587 * a mirror image of the player. 588 */ 589 Gfx *geo_render_mirror_mario(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 590 f32 mirroredX; 591 struct Object *mario = gMarioStates[0].marioObj; 592 593 switch (callContext) { 594 case GEO_CONTEXT_CREATE: 595 init_graph_node_object(NULL, &gMirrorMario, NULL, gVec3fZero, gVec3sZero, gVec3fOne); 596 break; 597 case GEO_CONTEXT_AREA_LOAD: 598 geo_add_child(node, &gMirrorMario.node); 599 break; 600 case GEO_CONTEXT_AREA_UNLOAD: 601 geo_remove_child(&gMirrorMario.node); 602 break; 603 case GEO_CONTEXT_RENDER: 604 if (mario->header.gfx.pos[0] > 1700.0f) { 605 // TODO: Is this a geo layout copy or a graph node copy? 606 gMirrorMario.sharedChild = mario->header.gfx.sharedChild; 607 gMirrorMario.areaIndex = mario->header.gfx.areaIndex; 608 vec3s_copy(gMirrorMario.angle, mario->header.gfx.angle); 609 vec3f_copy(gMirrorMario.pos, mario->header.gfx.pos); 610 vec3f_copy(gMirrorMario.scale, mario->header.gfx.scale); 611 612 gMirrorMario.animInfo = mario->header.gfx.animInfo; 613 mirroredX = MIRROR_X - gMirrorMario.pos[0]; 614 gMirrorMario.pos[0] = mirroredX + MIRROR_X; 615 gMirrorMario.angle[1] = -gMirrorMario.angle[1]; 616 gMirrorMario.scale[0] *= -1.0f; 617 ((struct GraphNode *) &gMirrorMario)->flags |= GRAPH_RENDER_ACTIVE; 618 } else { 619 ((struct GraphNode *) &gMirrorMario)->flags &= ~GRAPH_RENDER_ACTIVE; 620 } 621 break; 622 } 623 return NULL; 624 } 625 626 /** 627 * Since Mirror Mario has an x scale of -1, the mesh becomes inside out. 628 * This node corrects that by changing the culling mode accordingly. 629 */ 630 Gfx *geo_mirror_mario_backface_culling(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) { 631 struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node; 632 Gfx *gfx = NULL; 633 634 if (callContext == GEO_CONTEXT_RENDER && gCurGraphNodeObject == &gMirrorMario) { 635 gfx = alloc_display_list(3 * sizeof(*gfx)); 636 637 if (asGenerated->parameter == 0) { 638 gSPClearGeometryMode(&gfx[0], G_CULL_BACK); 639 gSPSetGeometryMode(&gfx[1], G_CULL_FRONT); 640 gSPEndDisplayList(&gfx[2]); 641 } else { 642 gSPClearGeometryMode(&gfx[0], G_CULL_FRONT); 643 gSPSetGeometryMode(&gfx[1], G_CULL_BACK); 644 gSPEndDisplayList(&gfx[2]); 645 } 646 asGenerated->fnNode.node.flags = (asGenerated->fnNode.node.flags & 0xFF) | (LAYER_OPAQUE << 8); 647 } 648 649 return gfx; 650 }