317 lines
11 KiB
C
317 lines
11 KiB
C
// tuxie.c.inc
|
|
|
|
void play_penguin_walking_sound(s32 walk) {
|
|
s32 sound;
|
|
if (o->oSoundStateID == 0) {
|
|
if (walk == PENGUIN_WALK_BABY)
|
|
sound = SOUND_OBJ_BABY_PENGUIN_WALK;
|
|
else // PENGUIN_WALK_BIG
|
|
sound = SOUND_OBJ_BIG_PENGUIN_WALK;
|
|
set_obj_anim_with_accel_and_sound(1, 11, sound);
|
|
}
|
|
}
|
|
|
|
void tuxies_mother_act_2(void) {
|
|
f32 sp24;
|
|
UNUSED s32 unused;
|
|
struct Object *sp1C = cur_obj_find_nearest_object_with_behavior(bhvSmallPenguin, &sp24);
|
|
|
|
if (cur_obj_find_nearby_held_actor(bhvUnused20E0, 1000.0f) != NULL) {
|
|
if (o->oSubAction == 0) {
|
|
cur_obj_init_animation_with_sound(0);
|
|
o->oForwardVel = 10.0f;
|
|
if (800.0f < cur_obj_lateral_dist_from_mario_to_home())
|
|
o->oSubAction = 1;
|
|
cur_obj_rotate_yaw_toward(o->oAngleToMario, 0x400);
|
|
} else {
|
|
o->oForwardVel = 0.0f;
|
|
cur_obj_init_animation_with_sound(3);
|
|
if (cur_obj_lateral_dist_from_mario_to_home() < 700.0f)
|
|
o->oSubAction = 0;
|
|
}
|
|
} else {
|
|
o->oForwardVel = 0.0f;
|
|
cur_obj_init_animation_with_sound(3);
|
|
}
|
|
if (sp1C != NULL && sp24 < 300.0f && sp1C->oHeldState != HELD_FREE) {
|
|
o->oAction = 1;
|
|
sp1C->oSmallPenguinUnk88 = 1;
|
|
o->prevObj = sp1C;
|
|
}
|
|
}
|
|
|
|
void tuxies_mother_act_1(void) {
|
|
s32 sp2C;
|
|
s32 sp28;
|
|
s32 dialogID;
|
|
switch (o->oSubAction) {
|
|
case 0:
|
|
cur_obj_init_animation_with_sound(3);
|
|
if (!cur_obj_is_mario_on_platform()) {
|
|
sp2C = (o->oBehParams >> 0x10) & 0xFF;
|
|
sp28 = (o->prevObj->oBehParams >> 0x10) & 0xFF;
|
|
if (sp2C == sp28)
|
|
dialogID = DIALOG_058;
|
|
else
|
|
dialogID = DIALOG_059;
|
|
if (cur_obj_update_dialog_with_cutscene(2, 1, CUTSCENE_DIALOG, dialogID)) {
|
|
if (dialogID == DIALOG_058)
|
|
o->oSubAction = 1;
|
|
else
|
|
o->oSubAction = 2;
|
|
o->prevObj->oInteractionSubtype |= INT_SUBTYPE_DROP_IMMEDIATELY;
|
|
}
|
|
} else
|
|
cur_obj_init_animation_with_sound(0);
|
|
break;
|
|
case 1:
|
|
if (o->prevObj->oHeldState == HELD_FREE) {
|
|
//! This line is was almost certainly supposed to be something
|
|
// like o->prevObj->oInteractionSubtype &= ~INT_SUBTYPE_DROP_IMMEDIATELY;
|
|
// however, this code uses the value of o->oInteractionSubtype
|
|
// rather than its offset to rawData. For this object,
|
|
// o->oInteractionSubtype is always 0, so the result is this:
|
|
// o->prevObj->oUnknownUnk88 &= ~INT_SUBTYPE_DROP_IMMEDIATELY
|
|
// which has no effect as o->prevObj->oUnknownUnk88 is always 0
|
|
// or 1, which is not affected by the bitwise AND.
|
|
o->prevObj->OBJECT_FIELD_S32(o->oInteractionSubtype) &= ~INT_SUBTYPE_DROP_IMMEDIATELY;
|
|
obj_set_behavior(o->prevObj, bhvUnused20E0);
|
|
#ifndef VERSION_JP
|
|
cur_obj_spawn_star_at_y_offset(3167.0f, -4300.0f, 5108.0f, 200.0f);
|
|
#else
|
|
spawn_default_star(3500.0f, -4300.0f, 4650.0f);
|
|
#endif
|
|
o->oAction = 2;
|
|
}
|
|
break;
|
|
case 2:
|
|
if (o->prevObj->oHeldState == HELD_FREE) {
|
|
//! Same bug as above
|
|
o->prevObj->OBJECT_FIELD_S32(o->oInteractionSubtype) &= ~INT_SUBTYPE_DROP_IMMEDIATELY;
|
|
obj_set_behavior(o->prevObj, bhvPenguinBaby);
|
|
o->oAction = 2;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void tuxies_mother_act_0(void) {
|
|
s32 sp2C;
|
|
f32 sp28;
|
|
struct Object *sp24;
|
|
sp2C = 0;
|
|
sp24 = cur_obj_find_nearest_object_with_behavior(bhvSmallPenguin, &sp28);
|
|
cur_obj_scale(4.0f);
|
|
cur_obj_init_animation_with_sound(3);
|
|
if (sp28 < 500.0f)
|
|
sp2C = 1;
|
|
if (sp24 != NULL && sp28 < 300.0f && sp24->oHeldState != HELD_FREE) {
|
|
o->oAction = 1;
|
|
sp24->oSmallPenguinUnk88 = 1;
|
|
o->prevObj = sp24;
|
|
} else {
|
|
switch (o->oSubAction) {
|
|
case 0:
|
|
if (cur_obj_can_mario_activate_textbox_2(300.0f, 100.0f))
|
|
if (sp2C == 0)
|
|
o->oSubAction++;
|
|
break;
|
|
case 1:
|
|
if (cur_obj_update_dialog_with_cutscene(2, 1, CUTSCENE_DIALOG, DIALOG_057))
|
|
o->oSubAction++;
|
|
break;
|
|
case 2:
|
|
if (o->oDistanceToMario > 450.0f)
|
|
o->oSubAction = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (cur_obj_check_anim_frame(1))
|
|
cur_obj_play_sound_2(SOUND_OBJ_BIG_PENGUIN_YELL);
|
|
}
|
|
|
|
void (*sTuxiesMotherActions[])(void) = { tuxies_mother_act_0, tuxies_mother_act_1,
|
|
tuxies_mother_act_2 };
|
|
|
|
void bhv_tuxies_mother_loop(void) {
|
|
o->activeFlags |= 0x400;
|
|
cur_obj_update_floor_and_walls();
|
|
cur_obj_call_action_function(sTuxiesMotherActions);
|
|
cur_obj_move_standard(-78);
|
|
play_penguin_walking_sound(PENGUIN_WALK_BIG);
|
|
o->oInteractStatus = 0;
|
|
}
|
|
|
|
void small_penguin_dive_with_mario(void) {
|
|
if (mario_is_dive_sliding()) {
|
|
o->oSmallPenguinUnk100 = o->oAction;
|
|
o->oAction = 3;
|
|
}
|
|
}
|
|
|
|
void small_penguin_act_2(void) {
|
|
s32 sp1C = 0;
|
|
if (o->oTimer == 0)
|
|
if (cur_obj_dist_to_nearest_object_with_behavior(bhvTuxiesMother) < 1000.0f)
|
|
sp1C = 1;
|
|
cur_obj_init_animation_with_sound(0);
|
|
o->oForwardVel = o->oSmallPenguinUnk104 + 3.0f;
|
|
cur_obj_rotate_yaw_toward(o->oAngleToMario + 0x8000, o->oSmallPenguinUnk110 + 0x600);
|
|
if (o->oDistanceToMario > o->oSmallPenguinUnk108 + 500.0f)
|
|
o->oAction = 0;
|
|
small_penguin_dive_with_mario();
|
|
if (sp1C)
|
|
o->oAction = 5;
|
|
}
|
|
|
|
void small_penguin_act_1(void) {
|
|
cur_obj_init_animation_with_sound(0);
|
|
o->oForwardVel = o->oSmallPenguinUnk104 + 3.0f;
|
|
cur_obj_rotate_yaw_toward(o->oAngleToMario, o->oSmallPenguinUnk110 + 0x600);
|
|
if (o->oDistanceToMario < o->oSmallPenguinUnk108 + 300.0f)
|
|
o->oAction = 0;
|
|
if (o->oDistanceToMario > 1100.0f)
|
|
o->oAction = 0;
|
|
small_penguin_dive_with_mario();
|
|
}
|
|
|
|
void small_penguin_act_3(void) {
|
|
if (o->oTimer > 5) {
|
|
if (o->oTimer == 6)
|
|
cur_obj_play_sound_2(SOUND_OBJ_BABY_PENGUIN_DIVE);
|
|
cur_obj_init_animation_with_sound(1);
|
|
if (o->oTimer > 25)
|
|
if (!mario_is_dive_sliding())
|
|
o->oAction = 4;
|
|
}
|
|
}
|
|
|
|
void small_penguin_act_4(void) {
|
|
if (o->oTimer > 20) {
|
|
o->oForwardVel = 0.0f;
|
|
cur_obj_init_animation_with_sound(2);
|
|
if (o->oTimer > 40)
|
|
o->oAction = o->oSmallPenguinUnk100;
|
|
}
|
|
}
|
|
|
|
void small_penguin_act_0(void) {
|
|
s32 sp1C;
|
|
|
|
sp1C = 0;
|
|
cur_obj_init_animation_with_sound(3);
|
|
if (o->oTimer == 0) {
|
|
o->oSmallPenguinUnk110 = (s32)(random_float() * 0x400);
|
|
o->oSmallPenguinUnk108 = random_float() * 100.0f;
|
|
o->oSmallPenguinUnk104 = random_float();
|
|
o->oForwardVel = 0.0f;
|
|
if (cur_obj_dist_to_nearest_object_with_behavior(bhvTuxiesMother) < 1000.0f)
|
|
sp1C = 1;
|
|
}
|
|
if (o->oDistanceToMario < 1000.0f && o->oSmallPenguinUnk108 + 600.0f < o->oDistanceToMario)
|
|
o->oAction = 1;
|
|
else if (o->oDistanceToMario < o->oSmallPenguinUnk108 + 300.0f)
|
|
o->oAction = 2;
|
|
if (sp1C)
|
|
o->oAction = 5;
|
|
if (cur_obj_mario_far_away())
|
|
cur_obj_set_pos_to_home();
|
|
}
|
|
|
|
void small_penguin_act_5(void) {
|
|
f32 sp24;
|
|
s16 sp22;
|
|
struct Object *sp1C = cur_obj_nearest_object_with_behavior(bhvTuxiesMother);
|
|
if (sp1C != NULL) {
|
|
if (o->oDistanceToMario < 1000.0f)
|
|
o->oForwardVel = 2.0f;
|
|
else
|
|
o->oForwardVel = 0.0f;
|
|
sp24 = dist_between_objects(o, sp1C);
|
|
sp22 = obj_angle_to_object(o, sp1C);
|
|
if (sp24 > 200.0f)
|
|
cur_obj_rotate_yaw_toward(sp22, 0x400);
|
|
else
|
|
cur_obj_rotate_yaw_toward(sp22 + 0x8000, 0x400);
|
|
cur_obj_init_animation_with_sound(0);
|
|
}
|
|
small_penguin_dive_with_mario();
|
|
}
|
|
|
|
void (*sSmallPenguinActions[])(void) = {
|
|
small_penguin_act_0, small_penguin_act_1, small_penguin_act_2,
|
|
small_penguin_act_3, small_penguin_act_4, small_penguin_act_5
|
|
};
|
|
|
|
void small_penguin_free_actions(void) {
|
|
if (o->oSmallPenguinUnk88 != 0) {
|
|
o->oAction = 5;
|
|
o->oSmallPenguinUnk88 = 0;
|
|
}
|
|
cur_obj_update_floor_and_walls();
|
|
cur_obj_call_action_function(sSmallPenguinActions);
|
|
cur_obj_move_standard(-78);
|
|
play_penguin_walking_sound(PENGUIN_WALK_BABY);
|
|
}
|
|
|
|
void bhv_small_penguin_loop(void) {
|
|
switch (o->oHeldState) {
|
|
case HELD_FREE:
|
|
small_penguin_free_actions();
|
|
break;
|
|
case HELD_HELD:
|
|
cur_obj_unrender_and_reset_state(0, 0);
|
|
if (cur_obj_has_behavior(bhvPenguinBaby))
|
|
obj_set_behavior(o, bhvSmallPenguin);
|
|
obj_copy_pos(o, gMarioObject);
|
|
if (gGlobalTimer % 30 == 0)
|
|
#ifndef VERSION_JP
|
|
play_sound(SOUND_OBJ2_BABY_PENGUIN_YELL, gMarioObject->header.gfx.cameraToObject);
|
|
#else
|
|
play_sound(SOUND_OBJ2_BABY_PENGUIN_YELL, o->header.gfx.cameraToObject);
|
|
#endif
|
|
break;
|
|
case HELD_THROWN:
|
|
cur_obj_get_thrown_or_placed(0, 0, 0);
|
|
break;
|
|
case HELD_DROPPED:
|
|
cur_obj_get_dropped();
|
|
break;
|
|
}
|
|
}
|
|
|
|
/** Geo switch logic for Tuxie's mother's eyes. Cases 0-4. Interestingly, case
|
|
* 4 is unused, and is the eye state seen in Shoshinkai 1995 footage.
|
|
*/
|
|
Gfx *geo_switch_tuxie_mother_eyes(s32 run, struct GraphNode *node, UNUSED Mat4 *mtx) {
|
|
struct Object *obj;
|
|
struct GraphNodeSwitchCase *switchCase;
|
|
s32 timer;
|
|
|
|
if (run == TRUE) {
|
|
obj = (struct Object *) gCurGraphNodeObject;
|
|
switchCase = (struct GraphNodeSwitchCase *) node;
|
|
switchCase->selectedCase = 0;
|
|
|
|
// timer logic for blinking. uses cases 0-2.
|
|
timer = gGlobalTimer % 50;
|
|
if (timer < 43)
|
|
switchCase->selectedCase = 0;
|
|
else if (timer < 45)
|
|
switchCase->selectedCase = 1;
|
|
else if (timer < 47)
|
|
switchCase->selectedCase = 2;
|
|
else
|
|
switchCase->selectedCase = 1;
|
|
|
|
/** make Tuxie's Mother have angry eyes if Mario takes the correct baby
|
|
* after giving it back. The easiest way to check this is to see if she's
|
|
* moving, since she only does when she's chasing Mario.
|
|
*/
|
|
if (segmented_to_virtual(bhvTuxiesMother) == obj->behavior)
|
|
if (obj->oForwardVel > 5.0f)
|
|
switchCase->selectedCase = 3;
|
|
}
|
|
return NULL;
|
|
}
|