TombEngine/TR5Main/Game/Lara/lara.cpp

1077 lines
27 KiB
C++

#include "framework.h"
#include "Game/Lara/lara.h"
#include "Game/Lara/lara_basic.h"
#include "Game/Lara/lara_helpers.h"
#include "Game/Lara/lara_jump.h"
#include "Game/Lara/lara_tests.h"
#include "Game/Lara/lara_monkey.h"
#include "Game/Lara/lara_crawl.h"
#include "Game/Lara/lara_objects.h"
#include "Game/Lara/lara_hang.h"
#include "Game/Lara/lara_helpers.h"
#include "Game/Lara/lara_slide.h"
#include "Game/Lara/lara_fire.h"
#include "Game/Lara/lara_surface.h"
#include "Game/Lara/lara_swim.h"
#include "Game/Lara/lara_one_gun.h"
#include "Game/Lara/lara_cheat.h"
#include "Game/Lara/lara_climb.h"
#include "Game/Lara/lara_collide.h"
#include "Game/Lara/lara_overhang.h"
#include "Game/Lara/lara_initialise.h"
#include "Game/animation.h"
#include "Game/camera.h"
#include "Game/collision/collide_item.h"
#include "Game/control/flipeffect.h"
#include "Game/control/volume.h"
#include "Game/effects/lara_fx.h"
#include "Game/effects/tomb4fx.h"
#include "Game/gui.h"
#include "Game/items.h"
#include "Game/misc.h"
#include "Game/savegame.h"
#include "Scripting/GameFlowScript.h"
#include "Sound/sound.h"
#include "Renderer/Renderer11.h"
using namespace TEN::Effects::Lara;
using namespace TEN::Control::Volumes;
using std::function;
using TEN::Renderer::g_Renderer;
LaraInfo Lara;
ITEM_INFO* LaraItem;
CollisionInfo LaraCollision = {};
byte LaraNodeUnderwater[NUM_LARA_MESHES];
function<LaraRoutineFunction> lara_control_routines[NUM_LARA_STATES + 1] =
{
lara_as_walk_forward,
lara_as_run_forward,
lara_as_idle,
lara_as_jump_forward,//33
lara_as_pose,//4
lara_as_run_back,//5
lara_as_turn_right_slow,//6
lara_as_turn_left_slow,//7
lara_as_death,//8
lara_as_freefall,//9
lara_as_hang,
lara_as_reach,
lara_as_splat,
lara_as_underwater_idle,//13
lara_void_func,//14
lara_as_jump_prepare,//15
lara_as_walk_back,//16
lara_as_underwater_swim_forward,//17
lara_as_underwater_inertia,//18
lara_as_controlled_no_look,//19
lara_as_turn_right_fast,//20
lara_as_step_right,//21
lara_as_step_left,//22
lara_as_roll_back,
lara_as_slide_forward,//24
lara_as_jump_back,//25
lara_as_jump_right,//26
lara_as_jump_left,//27
lara_as_jump_up,//28
lara_as_fall_back,//29
lara_as_shimmy_left,//30
lara_as_shimmy_right,//31
lara_as_slide_back,//32
lara_as_surface_idle,//33
lara_as_surface_swim_forward,//34
lara_as_surface_dive,//35
lara_as_pushable_push,//36
lara_as_pushable_pull,//37
lara_as_pushable_grab,//38
lara_as_pickup,//39
lara_as_switch_on,//40
lara_as_switch_off,//41
lara_as_use_key,//42
lara_as_use_puzzle,//43
lara_as_underwater_death,//44
lara_as_roll_forward,//45
lara_as_special,//46
lara_as_surface_swim_back,//47
lara_as_surface_swim_left,//48
lara_as_surface_swim_right,//49
lara_void_func,//50
lara_void_func,//51
lara_as_swan_dive,//52
lara_as_freefall_dive,//53
lara_as_handstand,//54
lara_as_surface_climb_out,//55
lara_as_climb_idle,//56
lara_as_climb_up,//57
lara_as_climb_left,//58
lara_as_climb_end,//59
lara_as_climb_right,//60
lara_as_climb_down,//61
lara_as_auto_jump,//62
lara_void_func,//63
lara_void_func,//64
lara_as_wade_forward,//65
lara_as_underwater_roll_180,//66
lara_as_pickup_flare,//67
lara_void_func,//68
lara_void_func,//69
lara_as_zip_line,//70
lara_as_crouch_idle,//71
lara_as_crouch_roll,//72
lara_as_sprint,//73
lara_as_sprint_dive,//74
lara_as_monkey_idle,//75
lara_as_monkey_forward,//76
lara_as_monkey_shimmy_left,//77
lara_as_monkey_shimmy_right,//78
lara_as_monkey_turn_180,//79
lara_as_crawl_idle,//80
lara_as_crawl_forward,//81
lara_as_monkey_turn_left,//82
lara_as_monkey_turn_right,//83
lara_as_crawl_turn_left,//84
lara_as_crawl_turn_right,//85
lara_as_crawl_back,//86
lara_as_controlled_no_look,
lara_as_controlled_no_look,
lara_as_controlled,
lara_as_rope_turn_clockwise,
lara_as_rope_turn_counter_clockwise,
lara_as_controlled,
lara_as_controlled,
lara_as_controlled,
lara_as_controlled_no_look,
lara_as_controlled_no_look,
lara_as_controlled,
lara_as_pickup,//98
lara_as_pole_idle,//99
lara_as_pole_up,//100
lara_as_pole_down,//101
lara_as_pole_turn_clockwise,//102
lara_as_pole_turn_counter_clockwise,//103
lara_as_pulley,//104
lara_as_crouch_turn_left,//105
lara_as_crouch_turn_right,//106
lara_as_corner,//107
lara_as_corner,//108
lara_as_corner,//109
lara_as_corner,//110
lara_as_rope_idle,//111
lara_as_rope_up,//112
lara_as_rope_down,//113
lara_as_rope_idle,//114
lara_as_rope_idle,//115
lara_void_func,
lara_as_controlled,
lara_as_swimcheat,
lara_as_tightrope_idle,//119
lara_as_controlled_no_look,//120
lara_as_tightrope_walk,//121
lara_as_tightrope_fall,//122
lara_as_tightrope_fall,//123
lara_as_null,//124
#ifdef NEW_TIGHTROPE
lara_as_tightrope_dismount,//125
#else // !NEW_TIGHTROPE
lara_as_null,//125
#endif
lara_as_switch_on,//126
lara_as_null,//127
lara_as_horizontal_bar_swing,//128
lara_as_horizontal_bar_leap,//129
lara_as_null,//130
lara_as_controlled_no_look,//131
lara_as_controlled_no_look,//132
lara_as_null,//133
lara_as_null,//134
lara_as_null,//135
lara_as_null,//136
lara_as_null,//137
lara_as_null,//138
lara_as_null,//139
lara_as_null,//140
lara_as_null,//141
lara_as_null,//142
lara_as_slopeclimb,//143
lara_as_slopeclimbup,//144
lara_as_slopeclimbdown,//145
lara_as_controlled_no_look,//146
lara_as_null,//147
lara_as_null,//148
lara_as_slopefall,//149
lara_as_climb_stepoff_left,
lara_as_climb_stepoff_right,
lara_as_turn_left_fast,
lara_as_controlled,
lara_as_controlled,
lara_as_controlled,//155
lara_as_slopehang,
lara_as_slopeshimmy,
lara_as_sclimbstart,
lara_as_sclimbstop,
lara_as_sclimbend,
lara_as_null,//161
lara_as_null,//162
lara_as_monkey_back,//163
lara_as_vault,//164
lara_as_vault,//165
lara_as_vault,//166
lara_as_vault,//167
lara_as_vault,//168
lara_as_vault,//169
lara_as_idle,//170
};
function<LaraRoutineFunction> lara_collision_routines[NUM_LARA_STATES + 1] =
{
lara_col_walk_forward,
lara_col_run_forward,
lara_col_idle,
lara_col_jump_forward,//3
lara_col_idle,//4
lara_col_run_back,
lara_col_turn_right_slow,
lara_col_turn_left_slow,
lara_col_death,
lara_col_freefall,//9
lara_col_hang,
lara_col_reach,
lara_col_splat,
lara_col_underwater_idle,
lara_col_land,
lara_col_jump_prepare,//15
lara_col_walk_back,
lara_col_underwater_swim_forward,
lara_col_underwater_inertia,
lara_default_col,//19
lara_col_turn_right_fast,
lara_col_step_right,
lara_col_step_left,
lara_col_roll_back,
lara_col_slide_forward,//24
lara_col_jump_back,//25
lara_col_jump_right,//26
lara_col_jump_left,//27
lara_col_jump_up,//28
lara_col_fall_back,//29
lara_col_shimmy_left,
lara_col_shimmy_right,
lara_col_slide_back,//32
lara_col_surface_idle,//33
lara_col_surface_swim_forward,//34
lara_col_surface_dive,//35
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_col_underwater_death,//44
lara_col_roll_forward,//45
lara_void_func,//46
lara_col_surface_swim_back,//47
lara_col_surface_swim_left,//48
lara_col_surface_swim_right,//49
lara_void_func,
lara_void_func,
lara_col_swan_dive,//52
lara_col_freefall_dive,//53
lara_default_col,
lara_default_col,//55
lara_col_climb_idle,
lara_col_climb_up,
lara_col_climb_left,
lara_col_climb_end,
lara_col_climb_right,
lara_col_climb_down,
lara_void_func,//62
lara_void_func,
lara_void_func,
lara_col_wade_forward,
lara_col_underwater_roll_180,
lara_default_col,
lara_void_func,
lara_void_func,
lara_void_func,
lara_col_crouch_idle,
lara_col_crouch_roll,
lara_col_sprint,
lara_col_sprint_dive,
lara_col_monkey_idle,
lara_col_monkey_forward,//76
lara_col_monkey_shimmy_left,//77
lara_col_monkey_shimmy_right,//78
lara_col_monkey_turn_180,//79
lara_col_crawl_idle,
lara_col_crawl_forward,
lara_col_monkey_turn_left,//81
lara_col_monkey_turn_right,//82
lara_col_crawl_turn_left,
lara_col_crawl_turn_right,
lara_col_crawl_back,
lara_void_func,
lara_col_crawl_to_hang,
lara_default_col,
lara_void_func,
lara_void_func,
lara_default_col,
lara_void_func,
lara_void_func,
lara_col_turn_switch,
lara_void_func,
lara_void_func,
lara_default_col,
lara_col_pole_idle,
lara_col_pole_up,
lara_col_pole_down,
lara_col_pole_turn_clockwise,
lara_col_pole_turn_counter_clockwise,
lara_default_col,
lara_col_crouch_turn_left,
lara_col_crouch_turn_right,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_col_rope_idle,
lara_void_func,
lara_void_func,
lara_col_rope_swing,
lara_col_rope_swing,
lara_void_func,
lara_void_func,
lara_col_underwater_swim_forward,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_default_col,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_void_func,
lara_col_slopeclimb,
lara_default_col, // lara_col_slopeclimbup
lara_default_col, // lara_col_slopeclimbdown
lara_void_func,
lara_void_func,
lara_void_func,
lara_default_col, // lara_col_slopefall
lara_default_col,
lara_default_col,
lara_col_turn_left_fast,
lara_default_col,
lara_default_col,
lara_default_col,
lara_col_slopehang, // lara_col_slopehang
lara_col_slopeshimmy, // lara_col_slopeshimmy
lara_default_col, // lara_col_sclimbstart
lara_default_col, // lara_col_sclimbstop
lara_default_col, // lara_col_sclimbend
lara_void_func,//161
lara_void_func,//162
lara_col_monkey_back,//163
lara_void_func,//164
lara_void_func,//165
lara_void_func,//166
lara_void_func,//167
lara_void_func,//168
lara_void_func,//169
lara_col_idle,//170
};
void LaraControl(ITEM_INFO* item, CollisionInfo* coll)
{
auto* lara = GetLaraInfo(item);
if (lara->Control.Weapon.HasFired)
{
AlertNearbyGuards(item);
lara->Control.Weapon.HasFired = false;
}
if (lara->PoisonPotency)
{
if (lara->PoisonPotency > LARA_POISON_POTENCY_MAX)
lara->PoisonPotency = LARA_POISON_POTENCY_MAX;
if (!(Wibble & 0xFF))
item->HitPoints -= lara->PoisonPotency;
}
if (lara->Control.IsMoving)
{
if (lara->Control.Count.PositionAdjust > LARA_POSITION_ADJUST_MAX_TIME)
{
lara->Control.IsMoving = false;
lara->Control.HandStatus = HandStatus::Free;
}
++lara->Control.Count.PositionAdjust;
}
if (!lara->Control.Locked)
lara->LocationPad = 128;
int oldX = item->Position.xPos;
int oldY = item->Position.yPos;
int oldZ = item->Position.zPos;
// Set hands free failsafe.
if (lara->Control.HandStatus == HandStatus::Busy &&
item->Animation.ActiveState == LS_IDLE &&
item->Animation.TargetState == LS_IDLE &&
item->Animation.AnimNumber == LA_STAND_IDLE &&
!item->Animation.Airborne)
{
lara->Control.HandStatus = HandStatus::Free;
}
if (lara->SprintEnergy < LARA_SPRINT_MAX && item->Animation.ActiveState != LS_SPRINT)
lara->SprintEnergy++;
lara->Control.IsLow = false;
bool isWater = TestEnvironment(ENV_FLAG_WATER, item);
bool isSwamp = TestEnvironment(ENV_FLAG_SWAMP, item);
int waterDepth = GetWaterDepth(item);
int waterHeight = GetWaterHeight(item);
int heightFromWater;
if (waterHeight != NO_HEIGHT)
heightFromWater = item->Position.yPos - waterHeight;
else
heightFromWater = NO_HEIGHT;
lara->WaterSurfaceDist = -heightFromWater;
if (lara->Vehicle == NO_ITEM)
WadeSplash(item, waterHeight, waterDepth);
if (lara->Vehicle == NO_ITEM && lara->ExtraAnim == -1)
{
switch (lara->Control.WaterStatus)
{
case WaterStatus::Dry:
if (heightFromWater == NO_HEIGHT || heightFromWater < WADE_DEPTH)
break;
Camera.targetElevation = -ANGLE(22.0f);
// Water is deep enough to swim; dispatch dive.
if (waterDepth >= SWIM_DEPTH && !isSwamp)
{
if (isWater)
{
item->Position.yPos += 100;
item->Animation.Airborne = false;
lara->Air = LARA_AIR_MAX;
lara->Control.WaterStatus = WaterStatus::Underwater;
UpdateItemRoom(item, 0);
StopSoundEffect(SFX_TR4_LARA_FALL);
if (item->Animation.ActiveState == LS_SWAN_DIVE)
{
SetAnimation(item, LA_SWANDIVE_DIVE);
item->Position.xRot = -ANGLE(45.0f);
item->Animation.VerticalVelocity *= 2;
lara->Control.HandStatus = HandStatus::Free;
}
else if (item->Animation.ActiveState == LS_FREEFALL_DIVE)
{
SetAnimation(item, LA_SWANDIVE_DIVE);
item->Position.xRot = -ANGLE(85.0f);
item->Animation.VerticalVelocity *= 2;
lara->Control.HandStatus = HandStatus::Free;
}
else
{
SetAnimation(item, LA_FREEFALL_DIVE);
item->Position.xRot = -ANGLE(45.0f);
item->Animation.VerticalVelocity = 3 * item->Animation.VerticalVelocity / 2;
}
ResetLaraFlex(item);
Splash(item);
}
}
// Water is at wade depth; update water status and do special handling.
else if (heightFromWater >= WADE_DEPTH)
{
lara->Control.WaterStatus = WaterStatus::Wade;
// Make splash ONLY within this particular threshold before swim depth while airborne (WadeSplash() above interferes otherwise).
if (waterDepth > (SWIM_DEPTH - CLICK(1)) &&
item->Animation.Airborne && !isSwamp)
{
item->Animation.TargetState = LS_IDLE;
Splash(item);
}
// Lara is grounded; don't splash again.
else if (!item->Animation.Airborne)
item->Animation.TargetState = LS_IDLE;
else if (isSwamp)
{
if (item->Animation.ActiveState == LS_SWAN_DIVE ||
item->Animation.ActiveState == LS_FREEFALL_DIVE)
{
item->Position.yPos = waterHeight + (SECTOR(1) - 24);
}
SetAnimation(item, LA_WADE);
}
}
break;
case WaterStatus::Underwater:
if (isWater ||
waterDepth == DEEP_WATER || abs(heightFromWater) >= CLICK(1) ||
item->Animation.AnimNumber == LA_UNDERWATER_RESURFACE ||
item->Animation.AnimNumber == LA_ONWATER_DIVE)
{
if (!isWater)
{
if (waterDepth == DEEP_WATER || abs(heightFromWater) >= CLICK(1))
{
SetAnimation(item, LA_FALL_START);
ResetLaraLean(item);
ResetLaraFlex(item);
item->Animation.Velocity = item->Animation.VerticalVelocity / 4;
item->Animation.VerticalVelocity = 0;
item->Animation.Airborne = true;
lara->Control.WaterStatus = WaterStatus::Dry;
}
else
{
SetAnimation(item, LA_UNDERWATER_RESURFACE);
ResetLaraLean(item);
ResetLaraFlex(item);
item->Position.yPos = waterHeight;
item->Animation.VerticalVelocity = 0;
lara->Control.WaterStatus = WaterStatus::TreadWater;
UpdateItemRoom(item, -(STEPUP_HEIGHT - 3));
SoundEffect(SFX_TR4_LARA_BREATH, &item->Position, 2);
}
}
}
else
{
SetAnimation(item, LA_UNDERWATER_RESURFACE);
ResetLaraLean(item);
ResetLaraFlex(item);
item->Position.yPos = waterHeight + 1;
item->Animation.VerticalVelocity = 0;
lara->Control.WaterStatus = WaterStatus::TreadWater;
UpdateItemRoom(item, 0);
SoundEffect(SFX_TR4_LARA_BREATH, &item->Position, 2);
}
break;
case WaterStatus::TreadWater:
if (!isWater)
{
if (heightFromWater <= WADE_DEPTH)
{
SetAnimation(item, LA_FALL_START);
item->Animation.Velocity = item->Animation.VerticalVelocity / 4;
item->Animation.Airborne = true;
lara->Control.WaterStatus = WaterStatus::Dry;
}
else
{
SetAnimation(item, LA_STAND_IDLE);
lara->Control.WaterStatus = WaterStatus::Wade;
}
ResetLaraLean(item);
ResetLaraFlex(item);
item->Animation.VerticalVelocity = 0;
}
break;
case WaterStatus::Wade:
Camera.targetElevation = -ANGLE(22.0f);
if (heightFromWater >= WADE_DEPTH)
{
if (heightFromWater > SWIM_DEPTH && !isSwamp)
{
SetAnimation(item, LA_ONWATER_IDLE);
ResetLaraLean(item);
ResetLaraFlex(item);
item->Position.yPos += 1 - heightFromWater;
item->Animation.VerticalVelocity = 0;
item->Animation.Airborne = false;
lara->Control.WaterStatus = WaterStatus::TreadWater;
UpdateItemRoom(item, 0);
}
}
else
{
lara->Control.WaterStatus = WaterStatus::Dry;
if (item->Animation.ActiveState == LS_WADE_FORWARD)
item->Animation.TargetState = LS_RUN_FORWARD;
}
break;
}
}
if (item->HitPoints <= 0)
{
item->HitPoints = -1;
if (lara->Control.Count.Death == 0)
StopSoundTracks();
lara->Control.Count.Death++;
if ((item->Flags & 0x100))
{
lara->Control.Count.Death++;
return;
}
}
switch (lara->Control.WaterStatus)
{
case WaterStatus::Dry:
case WaterStatus::Wade:
if (isSwamp && lara->WaterSurfaceDist < -(LARA_HEIGHT + 8)) // TODO: Find best height. @Sezz 2021.11.10
{
if (item->HitPoints >= 0)
{
lara->Air -= 6;
if (lara->Air < 0)
{
lara->Air = -1;
item->HitPoints -= 10;
}
}
}
else if (lara->Air < LARA_AIR_MAX && item->HitPoints >= 0)
{
if (lara->Vehicle == NO_ITEM) // only for the upv !!
{
lara->Air += 10;
if (lara->Air > LARA_AIR_MAX)
lara->Air = LARA_AIR_MAX;
}
}
LaraAboveWater(item, coll);
break;
case WaterStatus::Underwater:
if (item->HitPoints >= 0)
{
auto* level = g_GameFlow->GetLevel(CurrentLevel);
if (level->LaraType != LaraType::Divesuit)
lara->Air--;
if (lara->Air < 0)
{
// if (LaraDrawType == LARA_TYPE::DIVESUIT && lara->anxiety < 251)
// lara->anxiety += 4;
item->HitPoints -= 5;
lara->Air = -1;
}
}
LaraUnderwater(item, coll);
break;
case WaterStatus::TreadWater:
if (item->HitPoints >= 0)
{
lara->Air += 10;
if (lara->Air > LARA_AIR_MAX)
lara->Air = LARA_AIR_MAX;
}
LaraWaterSurface(item, coll);
break;
case WaterStatus::FlyCheat:
LaraCheat(item, coll);
break;
}
Statistics.Game.Distance += sqrt(
pow(item->Position.xPos - oldX, 2) +
pow(item->Position.yPos - oldY, 2) +
pow(item->Position.zPos - oldZ, 2));
}
void LaraAboveWater(ITEM_INFO* item, CollisionInfo* coll)
{
auto* lara = GetLaraInfo(item);
coll->Setup.UpperCeilingBound = NO_UPPER_BOUND;
coll->Setup.OldPosition.x = item->Position.xPos;
coll->Setup.OldPosition.y = item->Position.yPos;
coll->Setup.OldPosition.z = item->Position.zPos;
coll->Setup.OldState = item->Animation.ActiveState;
coll->Setup.OldAnimNumber = item->Animation.AnimNumber;
coll->Setup.OldFrameNumber = item->Animation.FrameNumber;
coll->Setup.EnableObjectPush = true;
coll->Setup.EnableSpasm = true;
coll->Setup.FloorSlopeIsWall = false;
coll->Setup.FloorSlopeIsPit = false;
coll->Setup.CeilingSlopeIsWall = false;
coll->Setup.DeathFlagIsPit = false;
coll->Setup.NoMonkeyFlagIsWall = false;
coll->Setup.Mode = CollisionProbeMode::Quadrants;
if (TrInput & IN_LOOK && lara->Control.CanLook &&
lara->ExtraAnim == NO_ITEM)
{
LookLeftRight(item);
}
else if (coll->Setup.Height > LARA_HEIGHT - LARA_HEADROOM) // TEMP HACK: Look feature will need a dedicated refactor; ResetLook() interferes with crawl flexing. @Sezz 2021.12.10
ResetLook(item);
// TODO: Move radius and height default resets above look checks once look feature is refactored.
coll->Setup.Radius = LARA_RAD;
coll->Setup.Height = LARA_HEIGHT;
lara->Control.CanLook = true;
UpdateItemRoom(item, -LARA_HEIGHT / 2);
// Process vehicles.
if (HandleLaraVehicle(item, coll))
return;
// Temp. debug stuff
//---
// Kill Lara.
if (KeyMap[DIK_D])
item->HitPoints = 0;
// Say no.
static bool dbNo = false;
if (KeyMap[DIK_N] && !dbNo)
SayNo();
dbNo = KeyMap[DIK_N] ? true : false;
static PHD_3DPOS posO = item->Position;
static short roomNumO = item->RoomNumber;
static CAMERA_INFO camO = Camera;
// Save position.
if (KeyMap[DIK_Q] && TrInput & IN_WALK)
{
posO = item->Position;
roomNumO = item->RoomNumber;
camO = Camera;
}
// Restore position.
if (KeyMap[DIK_E])
{
item->Position = posO;
item->RoomNumber = roomNumO;
Camera = camO;
}
// Forward 1 unit.
if (KeyMap[DIK_I])
MoveItem(item, item->Position.yRot, 1);
// Back 1 unit.
else if (KeyMap[DIK_K])
MoveItem(item, item->Position.yRot + ANGLE(180.0f), 1);
// Left 1 unit.
else if (KeyMap[DIK_J])
MoveItem(item, item->Position.yRot - ANGLE(90.0f), 1);
// Right 1 unit.
else if (KeyMap[DIK_L])
MoveItem(item, item->Position.yRot + ANGLE(90.0f), 1);
//---
// Temp. debug stuff.
static bool doRoutines = true;
static bool dbT = false;
if (KeyMap[DIK_T] && !dbT)
doRoutines = !doRoutines;
dbT = KeyMap[DIK_T] ? true : false;
static bool dbU = false;
if (doRoutines || KeyMap[DIK_U] && !dbU)
{
HandleLaraMovementParameters(item, coll);
// Handle current Lara status.
lara_control_routines[item->Animation.ActiveState](item, coll);
AnimateLara(item);
if (lara->ExtraAnim == -1)
{
// Check for collision with items.
DoObjectCollision(item, coll);
// Handle Lara collision.
if (lara->Vehicle == NO_ITEM)
lara_collision_routines[item->Animation.ActiveState](item, coll);
}
lara->ExtraVelocity = PHD_VECTOR();
}
dbU = KeyMap[DIK_U] ? true : false;
//if (lara->gunType == LaraWeaponType::Crossbow && !LaserSight)
// TrInput &= ~IN_ACTION;
// Handle weapons.
LaraGun(item);
// Handle breath.
LaraBreath(item);
// Test for flags and triggers.
ProcessSectorFlags(item);
TestTriggers(item, false);
TestVolumes(item);
}
void LaraWaterSurface(ITEM_INFO* item, CollisionInfo* coll)
{
auto* lara = GetLaraInfo(item);
Camera.targetElevation = -ANGLE(22.0f);
coll->Setup.LowerFloorBound = NO_LOWER_BOUND;
coll->Setup.UpperFloorBound = -CLICK(0.5f);
coll->Setup.LowerCeilingBound = LARA_RAD;
coll->Setup.UpperCeilingBound = NO_UPPER_BOUND;
coll->Setup.OldPosition.x = item->Position.xPos;
coll->Setup.OldPosition.y = item->Position.yPos;
coll->Setup.OldPosition.z = item->Position.zPos;
coll->Setup.FloorSlopeIsWall = false;
coll->Setup.FloorSlopeIsPit = false;
coll->Setup.CeilingSlopeIsWall = false;
coll->Setup.DeathFlagIsPit = false;
coll->Setup.NoMonkeyFlagIsWall = false;
coll->Setup.EnableObjectPush = false;
coll->Setup.EnableSpasm = false;
coll->Setup.Mode = CollisionProbeMode::FreeForward;
coll->Setup.Radius = LARA_RAD;
coll->Setup.Height = LARA_HEIGHT_SURFACE;
if (TrInput & IN_LOOK && lara->Control.CanLook)
LookLeftRight(item);
else
ResetLook(item);
lara->Control.CanLook = true;
lara->Control.Count.Pose = 0;
lara_control_routines[item->Animation.ActiveState](item, coll);
// Reset turn rate.
int sign = copysign(1, lara->Control.TurnRate);
if (abs(lara->Control.TurnRate) > ANGLE(2.0f))
lara->Control.TurnRate -= ANGLE(2.0f) * sign;
else if (abs(lara->Control.TurnRate) > ANGLE(0.5f))
lara->Control.TurnRate -= ANGLE(0.5f) * sign;
else
lara->Control.TurnRate = 0;
item->Position.yRot += lara->Control.TurnRate;
// Reset lean.
if (!lara->Control.IsMoving && !(TrInput & (IN_LEFT | IN_RIGHT)))
ResetLaraLean(item, 8.0f);
if (lara->WaterCurrentActive && lara->Control.WaterStatus != WaterStatus::FlyCheat)
LaraWaterCurrent(item, coll);
AnimateLara(item);
item->Position.xPos += item->Animation.VerticalVelocity * phd_sin(lara->Control.MoveAngle) / 4;
item->Position.zPos += item->Animation.VerticalVelocity * phd_cos(lara->Control.MoveAngle) / 4;
DoObjectCollision(item, coll);
if (lara->Vehicle == NO_ITEM)
lara_collision_routines[item->Animation.ActiveState](item, coll);
lara->ExtraVelocity = PHD_VECTOR();
UpdateItemRoom(item, LARA_RAD);
LaraGun(item);
ProcessSectorFlags(item);
TestTriggers(item, false);
TestVolumes(item);
}
void LaraUnderwater(ITEM_INFO* item, CollisionInfo* coll)
{
auto* lara = GetLaraInfo(item);
coll->Setup.LowerFloorBound = NO_LOWER_BOUND;
coll->Setup.UpperFloorBound = -(LARA_RAD_UNDERWATER + (LARA_RAD_UNDERWATER / 3));
coll->Setup.LowerCeilingBound = LARA_RAD_UNDERWATER + (LARA_RAD_UNDERWATER / 3);
coll->Setup.UpperCeilingBound = NO_UPPER_BOUND;
coll->Setup.OldPosition.x = item->Position.xPos;
coll->Setup.OldPosition.y = item->Position.yPos;
coll->Setup.OldPosition.z = item->Position.zPos;
coll->Setup.FloorSlopeIsWall = false;
coll->Setup.FloorSlopeIsPit = false;
coll->Setup.CeilingSlopeIsWall = false;
coll->Setup.DeathFlagIsPit = false;
coll->Setup.NoMonkeyFlagIsWall = false;
coll->Setup.EnableObjectPush = true;
coll->Setup.EnableSpasm = false;
coll->Setup.Mode = CollisionProbeMode::Quadrants;
coll->Setup.Radius = LARA_RAD_UNDERWATER;
coll->Setup.Height = LARA_HEIGHT;
if (TrInput & IN_LOOK && lara->Control.CanLook)
LookLeftRight(item);
else
ResetLook(item);
lara->Control.CanLook = true;
lara->Control.Count.Pose = 0;
lara_control_routines[item->Animation.ActiveState](item, coll);
auto* level = g_GameFlow->GetLevel(CurrentLevel);
if (level->LaraType == LaraType::Divesuit)
{
if (lara->Control.TurnRate < -ANGLE(0.5f))
lara->Control.TurnRate += ANGLE(0.5f);
else if (lara->Control.TurnRate > ANGLE(0.5f))
lara->Control.TurnRate -= ANGLE(0.5f);
else
lara->Control.TurnRate = 0;
}
else if (lara->Control.TurnRate < -ANGLE(2.0f))
lara->Control.TurnRate += ANGLE(2.0f);
else if (lara->Control.TurnRate > ANGLE(2.0f))
lara->Control.TurnRate -= ANGLE(2.0f);
else
lara->Control.TurnRate = 0;
item->Position.yRot += lara->Control.TurnRate;
if (level->LaraType == LaraType::Divesuit)
UpdateLaraSubsuitAngles(item);
if (!lara->Control.IsMoving && !(TrInput & (IN_LEFT | IN_RIGHT)))
ResetLaraLean(item, 8.0f, true, false);
if (item->Position.xRot < -ANGLE(85.0f))
item->Position.xRot = -ANGLE(85.0f);
else if (item->Position.xRot > ANGLE(85.0f))
item->Position.xRot = ANGLE(85.0f);
if (level->LaraType == LaraType::Divesuit)
{
if (item->Position.zRot > ANGLE(44.0f))
item->Position.zRot = ANGLE(44.0f);
else if (item->Position.zRot < -ANGLE(44.0f))
item->Position.zRot = -ANGLE(44.0f);
}
else
{
if (item->Position.zRot > ANGLE(22.0f))
item->Position.zRot = ANGLE(22.0f);
else if (item->Position.zRot < -ANGLE(22.0f))
item->Position.zRot = -ANGLE(22.0f);
}
if (lara->WaterCurrentActive && lara->Control.WaterStatus != WaterStatus::FlyCheat)
LaraWaterCurrent(item, coll);
AnimateLara(item);
item->Position.xPos += phd_cos(item->Position.xRot) * item->Animation.VerticalVelocity * phd_sin(item->Position.yRot) / 4;
item->Position.yPos -= item->Animation.VerticalVelocity * phd_sin(item->Position.xRot) / 4;
item->Position.zPos += phd_cos(item->Position.xRot) * item->Animation.VerticalVelocity * phd_cos(item->Position.yRot) / 4;
DoObjectCollision(item, coll);
if (/*lara->ExtraAnim == -1 &&*/ lara->Vehicle == NO_ITEM)
lara_collision_routines[item->Animation.ActiveState](item, coll);
lara->ExtraVelocity = PHD_VECTOR();
UpdateItemRoom(item, 0);
LaraGun(item);
ProcessSectorFlags(item);
TestTriggers(item, false);
TestVolumes(item);
}
void LaraCheat(ITEM_INFO* item, CollisionInfo* coll)
{
auto* lara = GetLaraInfo(item);
item->HitPoints = LARA_HEALTH_MAX;
LaraUnderwater(item, coll);
if (TrInput & IN_WALK && !(TrInput & IN_LOOK))
{
if (TestEnvironment(ENV_FLAG_WATER, item) || (lara->WaterSurfaceDist > 0 && lara->WaterSurfaceDist != NO_HEIGHT))
{
lara->Control.WaterStatus = WaterStatus::Underwater;
SetAnimation(item, LA_UNDERWATER_IDLE);
ResetLaraFlex(item);
}
else
{
lara->Control.WaterStatus = WaterStatus::Dry;
SetAnimation(item, LA_STAND_SOLID);
item->Position.zRot = 0;
item->Position.xRot = 0;
ResetLaraFlex(item);
}
lara->Control.HandStatus = HandStatus::Free;
LaraInitialiseMeshes(item);
item->HitPoints = LARA_HEALTH_MAX;
}
}