TombEngine/TR5Main/Game/Lara/lara_swim.cpp

694 lines
No EOL
14 KiB
C++

#include "framework.h"
#include "lara_swim.h"
#include "control.h"
#include "camera.h"
#include "items.h"
#include "box.h"
#include "Lara.h"
#include "lara_surface.h"
#include "effect.h"
#include "lara_fire.h"
#include "draw.h"
#include "camera.h"
#include "level.h"
#include "input.h"
#include "sound.h"
#include "GameFlowScript.h"
typedef struct SUBSUIT_INFO
{
short XRot;
short dXRot;
short XRotVel;
short Vel[2];
short YVel;
};
SUBSUIT_INFO Subsuit;
byte SubHitCount = 0;
void LaraWaterCurrent(COLL_INFO* coll)
{
if (Lara.currentActive)
{
SINK_INFO* sink = &g_Level.Sinks[Lara.currentActive - 1];
short angle = mGetAngle(sink->x, sink->z, LaraItem->pos.xPos, LaraItem->pos.zPos);
Lara.currentXvel += (sink->strength * 1024 * phd_sin(angle - ANGLE(90)) - Lara.currentXvel) / 16;
Lara.currentZvel += (sink->strength * 1024 * phd_cos(angle - ANGLE(90)) - Lara.currentZvel) / 16;
LaraItem->pos.yPos += ((sink->y - LaraItem->pos.yPos) / 16);
}
else
{
int shift = 0;
if (abs(Lara.currentXvel) <= 16)
shift = (abs(Lara.currentXvel) > 8) + 2;
else
shift = 4;
Lara.currentXvel -= Lara.currentXvel >> shift;
if (abs(Lara.currentXvel) < 4)
Lara.currentXvel = 0;
if (abs(Lara.currentZvel) <= 16)
shift = (abs(Lara.currentZvel) > 8) + 2;
else
shift = 4;
Lara.currentZvel -= Lara.currentZvel >> shift;
if (abs(Lara.currentZvel) < 4)
Lara.currentZvel = 0;
if (!Lara.currentXvel && !Lara.currentZvel)
return;
}
LaraItem->pos.xPos += (Lara.currentXvel / 256);
LaraItem->pos.zPos += (Lara.currentZvel / 256);
Lara.currentActive = 0;
coll->facing = phd_atan(LaraItem->pos.zPos - coll->old.z, LaraItem->pos.xPos - coll->old.x);
GetCollisionInfo(coll, LaraItem->pos.xPos, LaraItem->pos.yPos + 200, LaraItem->pos.zPos, LaraItem->roomNumber, 400);
if (coll->collType == CT_FRONT)
{
if (LaraItem->pos.xRot > ANGLE(35))
LaraItem->pos.xRot += ANGLE(1);
else if (LaraItem->pos.xRot < -ANGLE(35))
LaraItem->pos.xRot -= ANGLE(1);
else
LaraItem->fallspeed = 0;
}
else if (coll->collType == CT_TOP)
{
LaraItem->pos.xRot -= ANGLE(1);
}
else if (coll->collType == CT_TOP_FRONT)
{
LaraItem->fallspeed = 0;
}
else if (coll->collType == CT_LEFT)
{
LaraItem->pos.yRot += ANGLE(5);
}
else if (coll->collType == CT_RIGHT)
{
LaraItem->pos.yRot -= ANGLE(5);
}
if (coll->midFloor < 0 && coll->midFloor != NO_HEIGHT)
LaraItem->pos.yPos += coll->midFloor;
ShiftItem(LaraItem, coll);
coll->old.x = LaraItem->pos.xPos;
coll->old.y = LaraItem->pos.yPos;
coll->old.z = LaraItem->pos.zPos;
}
int GetWaterDepth(int x, int y, int z, short roomNumber)
{
FLOOR_INFO* floor;
ROOM_INFO* r = &g_Level.Rooms[roomNumber];
short roomIndex = NO_ROOM;
do
{
int zFloor = (z - r->z) / SECTOR(1);
int xFloor = (x - r->x) / SECTOR(1);
if (zFloor <= 0)
{
zFloor = 0;
if (xFloor < 1)
xFloor = 1;
else if (xFloor > r->ySize - 2)
xFloor = r->ySize - 2;
}
else if (zFloor >= r->xSize - 1)
{
zFloor = r->xSize - 1;
if (xFloor < 1)
xFloor = 1;
else if (xFloor > r->ySize - 2)
xFloor = r->ySize - 2;
}
else if (xFloor < 0)
xFloor = 0;
else if (xFloor >= r->ySize)
xFloor = r->ySize - 1;
floor = &r->floor[zFloor + xFloor * r->xSize];
roomIndex = GetDoor(floor);
if (roomIndex != NO_ROOM)
{
roomNumber = roomIndex;
r = &g_Level.Rooms[roomIndex];
}
} while (roomIndex != NO_ROOM);
if (r->flags & (ENV_FLAG_WATER|ENV_FLAG_SWAMP))
{
while (floor->skyRoom != NO_ROOM)
{
r = &g_Level.Rooms[floor->skyRoom];
if (!(r->flags & (ENV_FLAG_WATER|ENV_FLAG_SWAMP)))
{
int wh = floor->ceiling * 256;
floor = GetFloor(x, y, z, &roomNumber);
return (GetFloorHeight(floor, x, y, z) - wh);
}
floor = &XZ_GET_SECTOR(r, x - r->x, z - r->z);
}
return 0x7FFF;
}
else
{
while (floor->pitRoom != NO_ROOM)
{
r = &g_Level.Rooms[floor->pitRoom];
if (r->flags & (ENV_FLAG_WATER|ENV_FLAG_SWAMP))
{
int wh = floor->floor * 256;
floor = GetFloor(x, y, z, &roomNumber);
return (GetFloorHeight(floor, x, y, z) - wh);
}
floor = &XZ_GET_SECTOR(r, x - r->x, z - r->z);
}
return NO_HEIGHT;
}
}
void lara_col_waterroll(ITEM_INFO* item, COLL_INFO* coll)
{
LaraSwimCollision(item, coll);
}
void lara_col_uwdeath(ITEM_INFO* item, COLL_INFO* coll)
{
item->hitPoints = -1;
Lara.air = -1;
Lara.gunStatus = LG_HANDS_BUSY;
int wh = GetWaterHeight(item->pos.xPos, item->pos.yPos, item->pos.zPos, item->roomNumber);
if (wh != NO_HEIGHT)
{
if (wh < item->pos.yPos - 100)
item->pos.yPos -= 5;
}
LaraSwimCollision(item, coll);
}
void lara_col_dive(ITEM_INFO* item, COLL_INFO* coll)
{
LaraSwimCollision(item, coll);
}
void lara_col_tread(ITEM_INFO* item, COLL_INFO* coll)
{
LaraSwimCollision(item, coll);
}
void lara_col_glide(ITEM_INFO* item, COLL_INFO* coll)
{
LaraSwimCollision(item, coll);
}
void lara_col_swim(ITEM_INFO* item, COLL_INFO* coll)
{
LaraSwimCollision(item, coll);
}
void lara_as_waterroll(ITEM_INFO* item, COLL_INFO* coll)
{
item->fallspeed = 0;
}
void lara_as_uwdeath(ITEM_INFO* item, COLL_INFO* coll)
{
Lara.look = 0;
item->fallspeed -= 8;
if (item->fallspeed <= 0)
item->fallspeed = 0;
if (item->pos.xRot < -ANGLE(2) || item->pos.xRot > ANGLE(2))
{
if (item->pos.xRot >= 0)
item->pos.xRot -= ANGLE(2);
else
item->pos.xRot += ANGLE(2);
}
else
{
item->pos.xRot = 0;
}
}
void lara_as_dive(ITEM_INFO* item, COLL_INFO* coll)
{
if (TrInput & IN_FORWARD)
{
item->pos.xRot -= ANGLE(1);
}
}
void lara_as_tread(ITEM_INFO* item, COLL_INFO* coll)
{
if (item->hitPoints <= 0)
{
item->goalAnimState = LS_WATER_DEATH;
return;
}
if (TrInput & IN_ROLL && LaraDrawType != LARA_DIVESUIT)
{
item->currentAnimState = LS_UNDERWATER_ROLL;
item->animNumber = LA_UNDERWATER_ROLL_180_START;
item->frameNumber = g_Level.Anims[item->animNumber].frameBase;
}
else
{
if (TrInput & IN_LOOK)
LookUpDown();
if (LaraDrawType == LARA_DIVESUIT)
SwimTurnSubsuit(item);
else
SwimTurn(item);
if (TrInput & IN_JUMP)
item->goalAnimState = LS_UNDERWATER_FORWARD;
item->fallspeed -= 6;
if (item->fallspeed < 0)
item->fallspeed = 0;
if (Lara.gunStatus == LG_HANDS_BUSY)
Lara.gunStatus = LG_NO_ARMS;
}
}
void lara_as_glide(ITEM_INFO* item, COLL_INFO* coll)
{
if (item->hitPoints <= 0)
{
item->goalAnimState = LS_WATER_DEATH;
return;
}
if (TrInput & IN_ROLL)
{
if (LaraDrawType != LARA_DIVESUIT)
{
item->currentAnimState = LS_UNDERWATER_ROLL;
item->animNumber = LA_UNDERWATER_ROLL_180_START;
item->frameNumber = g_Level.Anims[item->animNumber].frameBase;
return;
}
}
else if (LaraDrawType != LARA_DIVESUIT)
{
SwimTurn(item);
}
else
{
SwimTurnSubsuit(item);
}
if (TrInput & IN_JUMP)
item->goalAnimState = LS_UNDERWATER_FORWARD;
item->fallspeed -= 6;
if (item->fallspeed < 0)
item->fallspeed = 0;
if (item->fallspeed <= 133)
item->goalAnimState = LS_UNDERWATER_STOP;
}
void lara_as_swim(ITEM_INFO* item, COLL_INFO* coll)
{
if (item->hitPoints <= 0)
{
item->goalAnimState = LS_WATER_DEATH;
return;
}
if (TrInput & IN_ROLL)
{
if (LaraDrawType != LARA_DIVESUIT)
{
item->currentAnimState = LS_UNDERWATER_ROLL;
item->animNumber = LA_UNDERWATER_ROLL_180_START;
item->frameNumber = g_Level.Anims[item->animNumber].frameBase;
return;
}
}
else if (LaraDrawType != LARA_DIVESUIT)
{
SwimTurn(item);
}
else
{
SwimTurnSubsuit(item);
}
item->fallspeed += 8;
if (item->fallspeed > 200)
item->fallspeed = 200;
if (!(TrInput & IN_JUMP))
item->goalAnimState = LS_UNDERWATER_INERTIA;
}
void UpdateSubsuitAngles()
{
if (Subsuit.YVel != 0)
{
LaraItem->pos.yPos += Subsuit.YVel / 4;
Subsuit.YVel = ceil(0.9375 * Subsuit.YVel - 1); // YVel * (15/16)
}
Subsuit.Vel[0] = Subsuit.Vel[1] = -4 * LaraItem->fallspeed;
if (Subsuit.XRot >= Subsuit.dXRot)
{
if (Subsuit.XRot > Subsuit.dXRot)
{
if (Subsuit.XRot > 0 && Subsuit.dXRot < 0)
Subsuit.XRot = ceil(0.75 * Subsuit.XRot);
Subsuit.XRot -= ANGLE(2);
if (Subsuit.XRot < Subsuit.dXRot)
{
Subsuit.XRot = Subsuit.dXRot;
}
}
}
else
{
if (Subsuit.XRot < 0 && Subsuit.dXRot > 0)
Subsuit.XRot = ceil(0.75 * Subsuit.XRot);
Subsuit.XRot += ANGLE(2);
if (Subsuit.XRot > Subsuit.dXRot)
{
Subsuit.XRot = Subsuit.dXRot;
}
}
if (Subsuit.dXRot != 0)
{
short rot = Subsuit.dXRot / 8;
if (rot < -ANGLE(2))
rot = -ANGLE(2);
else if (rot > ANGLE(2))
rot = ANGLE(2);
LaraItem->pos.xRot += rot;
}
Subsuit.Vel[0] += abs(Subsuit.XRot / 8);
Subsuit.Vel[1] += abs(Subsuit.XRot / 8);
if (Lara.turnRate > 0)
{
Subsuit.Vel[0] += 2 * abs(Lara.turnRate);
}
else if (Lara.turnRate < 0)
{
Subsuit.Vel[1] += 2 * abs(Lara.turnRate);
}
if (Subsuit.Vel[0] > 1536)
Subsuit.Vel[0] = 1536;
if (Subsuit.Vel[1] > 1536)
Subsuit.Vel[1] = 1536;
if (Subsuit.Vel[0] != 0 || Subsuit.Vel[1] != 0)
{
SoundEffect(SFX_TR5_LARA_UNDERWATER_ENGINE, &LaraItem->pos, (((Subsuit.Vel[0] + Subsuit.Vel[1]) * 4) & 0x1F00) + 10);
}
}
void SwimTurnSubsuit(ITEM_INFO* item)
{
if (item->pos.yPos < 14080)
Subsuit.YVel += ((14080 - item->pos.yPos) / 16);
if (TrInput & IN_FORWARD && item->pos.xRot > -ANGLE(85))
{
Subsuit.dXRot = -ANGLE(45);
}
else if (TrInput & IN_BACK && item->pos.xRot < ANGLE(85))
{
Subsuit.dXRot = ANGLE(45);
}
else
{
Subsuit.dXRot = 0;
}
if (TrInput & IN_LEFT)
{
Lara.turnRate -= 136;
if (Lara.turnRate < -ANGLE(6))
Lara.turnRate = -ANGLE(6);
item->pos.zRot -= ANGLE(3);
}
else if (TrInput & IN_RIGHT)
{
Lara.turnRate += 136;
if (Lara.turnRate > ANGLE(6))
Lara.turnRate = ANGLE(6);
item->pos.zRot += ANGLE(3);
}
}
void SwimTurn(ITEM_INFO* item)
{
if (TrInput & IN_FORWARD)
{
item->pos.xRot -= ANGLE(2);
}
else if (TrInput & IN_BACK)
{
item->pos.xRot += ANGLE(2);
}
if (TrInput & IN_LEFT)
{
Lara.turnRate -= 409;
if (Lara.turnRate < -ANGLE(6))
Lara.turnRate = -ANGLE(6);
item->pos.zRot -= ANGLE(3);
}
else if (TrInput & IN_RIGHT)
{
Lara.turnRate += 409;
if (Lara.turnRate > ANGLE(6))
Lara.turnRate = ANGLE(6);
item->pos.zRot += ANGLE(3);
}
}
void LaraSwimCollision(ITEM_INFO* item, COLL_INFO* coll)
{
int oldX = item->pos.xPos;
int oldY = item->pos.yPos;
int oldZ = item->pos.zPos;
short oldXrot = item->pos.xRot;
short oldYrot = item->pos.yRot;
short oldZrot = item->pos.zRot;
if (item->pos.xRot < -ANGLE(90) || item->pos.xRot > ANGLE(90))
{
Lara.moveAngle = item->pos.yRot + ANGLE(180);
coll->facing = item->pos.yRot - ANGLE(180);
}
else
{
Lara.moveAngle = item->pos.yRot;
coll->facing = item->pos.yRot;
}
short height = 762 * phd_sin(item->pos.xRot);
height = abs(height);
if (height < ((LaraDrawType == LARA_DIVESUIT) * 64) + 200)
height = ((LaraDrawType == LARA_DIVESUIT) * 64) + 200;
coll->badNeg = -64;
COLL_INFO c1;
COLL_INFO c2;
memcpy(&c1, coll, sizeof(COLL_INFO));
memcpy(&c2, coll, sizeof(COLL_INFO) - 2);
GetCollisionInfo(coll, item->pos.xPos, height / 2 + item->pos.yPos, item->pos.zPos, item->roomNumber, height);
c1.facing += ANGLE(45);
GetCollisionInfo(&c1, item->pos.xPos, height / 2 + item->pos.yPos, item->pos.zPos, item->roomNumber, height);
c2.facing -= ANGLE(45);
GetCollisionInfo(&c2, item->pos.xPos, height / 2 + item->pos.yPos, item->pos.zPos, item->roomNumber, height);
ShiftItem(item, coll);
int flag = 0;
switch (coll->collType)
{
case CT_FRONT:
if (item->pos.xRot <= ANGLE(25))
{
if (item->pos.xRot >= -ANGLE(25))
{
if (item->pos.xRot > ANGLE(5))
item->pos.xRot += ANGLE(0.5);
else if (item->pos.xRot < -ANGLE(5))
item->pos.xRot -= ANGLE(0.5);
else if (item->pos.xRot > 0)
item->pos.xRot += 45;
else if (item->pos.xRot < 0)
item->pos.xRot -= 45;
else
{
item->fallspeed = 0;
flag = 1;
}
}
else
{
item->pos.xRot -= ANGLE(1);
flag = 1;
}
}
else
{
item->pos.xRot += ANGLE(1);
flag = 1;
}
if (c1.collType == CT_LEFT)
{
item->pos.yRot += ANGLE(2);
}
else if (c1.collType == CT_RIGHT)
{
item->pos.yRot -= ANGLE(2);
}
else if (c2.collType == CT_LEFT)
{
item->pos.yRot += ANGLE(2);
}
else if (c2.collType == CT_RIGHT)
{
item->pos.yRot -= ANGLE(2);
}
break;
case CT_TOP:
if (item->pos.xRot >= -8190)
{
flag = 1;
item->pos.xRot -= ANGLE(1);
}
break;
case CT_TOP_FRONT:
item->fallspeed = 0;
flag = 1;
break;
case CT_LEFT:
item->pos.yRot += ANGLE(2);
flag = 1;
break;
case CT_RIGHT:
item->pos.yRot -= ANGLE(2);
flag = 1;
break;
case CT_CLAMP:
flag = 2;
item->pos.xPos = coll->old.x;
item->pos.yPos = coll->old.y;
item->pos.zPos = coll->old.z;
item->fallspeed = 0;
break;
}
if (coll->midFloor < 0 && coll->midFloor != NO_HEIGHT)
{
flag = 1;
item->pos.xRot += ANGLE(1);
item->pos.yPos += coll->midFloor;
}
if (oldX == item->pos.xPos
&& oldY == item->pos.yPos
&& oldZ == item->pos.zPos
&& oldXrot == item->pos.xRot
&& oldYrot == item->pos.yRot
|| flag != 1)
{
if (flag == 2)
return;
}
else if (item->fallspeed > 100)
{
if (LaraDrawType == 5)
{
SoundEffect(SFX_TR5_SWIMSUIT_METAL_CLASH, &LaraItem->pos, ((2 * GetRandomControl() + 0x8000) * 256) | 6);
}
if (Lara.anxiety < 96)
{
Lara.anxiety += 16;
}
}
if (Lara.waterStatus != LW_FLYCHEAT && Lara.ExtraAnim == NO_ITEM)
LaraTestWaterDepth(item, coll);
}
void LaraTestWaterDepth(ITEM_INFO* item, COLL_INFO* coll)
{
short roomNumber = item->roomNumber;
FLOOR_INFO* floor = GetFloor(item->pos.xPos, item->pos.yPos, item->pos.zPos, &roomNumber);
int wd = GetWaterDepth(item->pos.xPos, item->pos.yPos, item->pos.zPos, roomNumber);
if (wd == NO_HEIGHT)
{
item->fallspeed = 0;
item->pos.xPos = coll->old.x;
item->pos.yPos = coll->old.y;
item->pos.zPos = coll->old.z;
}
else if (wd <= 512)
{
item->animNumber = LA_UNDERWATER_TO_STAND;
item->currentAnimState = LS_ONWATER_EXIT;
item->goalAnimState = LS_STOP;
item->pos.zRot = 0;
item->pos.xRot = 0;
item->speed = 0;
item->fallspeed = 0;
item->gravityStatus = false;
item->frameNumber = g_Level.Anims[item->animNumber].frameBase;
Lara.waterStatus = LW_WADE;
item->pos.yPos = GetFloorHeight(floor, item->pos.xPos, item->pos.yPos, item->pos.zPos);
}
}