raze/source/games/duke/src/input.cpp

704 lines
20 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
Copyright (C) 1996, 2003 - 3D Realms Entertainment
Copyright (C) 2000, 2003 - Matt Saettler (EDuke Enhancements)
2020-06-28 07:03:31 +00:00
Copyright (C) 2020 - Christoph Oelckers
This file is part of Enhanced Duke Nukem 3D version 1.5 - Atomic Edition
Duke Nukem 3D is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
Original Source: 1996 - Todd Replogle
Prepared for public release: 03/21/2003 - Charlie Wiederhold, 3D Realms
EDuke enhancements integrated: 04/13/2003 - Matt Saettler
Note: EDuke source was in transition. Changes are in-progress in the
source as it is released.
*/
//-------------------------------------------------------------------------
#include "ns.h"
#include "global.h"
#include "gamecontrol.h"
#include "v_video.h"
#include "dukeactor.h"
EXTERN_CVAR(Float, m_sensitivity_x)
EXTERN_CVAR(Float, m_yaw)
BEGIN_DUKE_NS
//---------------------------------------------------------------------------
//
2020-05-17 11:25:39 +00:00
// handles all HUD related input, i.e. inventory item selection and activation plus weapon selection.
//
// Note: This doesn't restrict the events to WW2GI - since the other games do
// not define any by default there is no harm done keeping the code clean.
//
//---------------------------------------------------------------------------
2020-05-17 11:25:39 +00:00
void hud_input(int plnum)
2020-05-17 11:25:39 +00:00
{
int i, k;
uint8_t dainv;
2022-02-07 10:04:19 +00:00
player_struct* p;
2020-05-17 11:25:39 +00:00
p = &ps[plnum];
2020-11-02 19:23:48 +00:00
auto pact = p->GetActor();
2020-05-17 11:25:39 +00:00
i = p->aim_mode;
p->aim_mode = !PlayerInput(plnum, SB_AIMMODE);
2020-05-17 11:25:39 +00:00
if (p->aim_mode < i)
p->sync.actions |= SB_CENTERVIEW;
2020-05-17 11:25:39 +00:00
// Backup weapon here as hud_input() is the first function where any one of the weapon variables can change.
2020-11-29 08:00:00 +00:00
p->backupweapon();
// Set-up crouch bools.
const int sectorLotag = p->insector() ? p->cursector->lotag : 0;
const bool crouchable = sectorLotag != ST_2_UNDERWATER && (sectorLotag != ST_1_ABOVE_WATER || p->spritebridge);
const bool disableToggle = p->jetpack_on || (!crouchable && p->on_ground) || (isRRRA() && (p->OnMotorcycle || p->OnBoat));
if (isRR() && (p->sync.actions & SB_CROUCH)) p->sync.actions &= ~SB_JUMP;
if (crouch_toggle && (!crouchable || disableToggle))
{
crouch_toggle = false;
p->sync.actions &= ~SB_CROUCH;
}
if (p->OnMotorcycle || p->OnBoat)
{
// mask out all actions not compatible with vehicles.
p->sync.actions &= ~(SB_WEAPONMASK_BITS | SB_TURNAROUND | SB_CENTERVIEW | SB_HOLSTER | SB_JUMP | SB_CROUCH | SB_RUN |
SB_AIM_UP | SB_AIM_DOWN | SB_AIMMODE | SB_LOOK_UP | SB_LOOK_DOWN | SB_LOOK_LEFT | SB_LOOK_RIGHT);
}
else
{
if ((isRR() && p->drink_amt > 88)) p->sync.actions |= SB_LOOK_LEFT;
if ((isRR() && p->drink_amt > 99)) p->sync.actions |= SB_LOOK_DOWN;
}
2020-05-17 11:25:39 +00:00
if (isRR())
{
if (PlayerInput(plnum, SB_QUICK_KICK) && p->last_pissed_time == 0)
2020-05-17 11:25:39 +00:00
{
if (!isRRRA() || p->GetActor()->spr.extra > 0)
2020-05-17 11:25:39 +00:00
{
p->last_pissed_time = 4000;
2020-11-02 19:23:48 +00:00
S_PlayActorSound(437, pact);
if (p->GetActor()->spr.extra <= gs.max_player_health - gs.max_player_health / 10)
2020-05-17 11:25:39 +00:00
{
p->GetActor()->spr.extra += 2;
p->last_extra = p->GetActor()->spr.extra;
2020-05-17 11:25:39 +00:00
}
else if (p->GetActor()->spr.extra < gs.max_player_health)
p->GetActor()->spr.extra = gs.max_player_health;
2020-05-17 11:25:39 +00:00
}
}
}
else
{
if (PlayerInput(plnum, SB_QUICK_KICK) && p->quick_kick == 0 && (p->curr_weapon != KNEE_WEAPON || p->kickback_pic == 0))
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_QUICKKICK, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
p->quick_kick = 14;
if (!p->quick_kick_msg && plnum == screenpeek) FTA(QUOTE_MIGHTY_FOOT, p);
p->quick_kick_msg = true;
2020-05-17 11:25:39 +00:00
}
}
}
if (!PlayerInput(plnum, SB_QUICK_KICK)) p->quick_kick_msg = false;
2020-05-17 11:25:39 +00:00
if (!PlayerInputBits(plnum, SB_INTERFACE_BITS))
2020-05-17 11:25:39 +00:00
p->interface_toggle_flag = 0;
else if (p->interface_toggle_flag == 0)
{
p->interface_toggle_flag = 1;
// Don't go on if paused or dead.
2020-07-15 16:10:31 +00:00
if (paused) return;
if (p->GetActor()->spr.extra <= 0) return;
// Activate an inventory item. This just forwards to the other inventory bits. If the inventory selector was taken out of the playsim this could be removed.
2020-11-02 23:20:51 +00:00
if (PlayerInput(plnum, SB_INVUSE) && p->newOwner == nullptr)
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_INVENTORY, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
if (p->inven_icon > ICON_NONE && p->inven_icon <= ICON_HEATS) PlayerSetItemUsed(plnum, p->inven_icon);
2020-05-17 11:25:39 +00:00
}
}
if (!isRR() && PlayerUseItem(plnum, ICON_HEATS))
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_USENIGHTVISION, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0 && p->heat_amount > 0)
{
p->heat_on = !p->heat_on;
p->inven_icon = 5;
2020-11-02 19:23:48 +00:00
S_PlayActorSound(NITEVISION_ONOFF, pact);
FTA(106 + (!p->heat_on), p);
}
2020-05-17 11:25:39 +00:00
}
if (PlayerUseItem(plnum, ICON_STEROIDS))
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_USESTEROIDS, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
if (p->steroids_amount == 400)
{
p->steroids_amount--;
2020-11-02 19:23:48 +00:00
S_PlayActorSound(DUKE_TAKEPILLS, pact);
p->inven_icon = ICON_STEROIDS;
FTA(12, p);
}
2020-05-17 11:25:39 +00:00
}
return;
2020-05-17 11:25:39 +00:00
}
if (PlayerInput(plnum, SB_INVPREV) || PlayerInput(plnum, SB_INVNEXT))
{
p->invdisptime = 26 * 2;
2020-05-17 11:25:39 +00:00
if (PlayerInput(plnum, SB_INVNEXT)) k = 1;
else k = 0;
2020-05-17 11:25:39 +00:00
dainv = p->inven_icon;
2020-05-17 11:25:39 +00:00
i = 0;
CHECKINV1:
2020-05-17 11:25:39 +00:00
if (i < 9)
2020-05-17 11:25:39 +00:00
{
i++;
switch (dainv)
{
case 4:
if (p->jetpack_amount > 0 && i > 1)
break;
if (k) dainv = 5;
else dainv = 3;
goto CHECKINV1;
case 6:
if (p->scuba_amount > 0 && i > 1)
break;
if (k) dainv = 7;
else dainv = 5;
goto CHECKINV1;
case 2:
if (p->steroids_amount > 0 && i > 1)
break;
if (k) dainv = 3;
else dainv = 1;
goto CHECKINV1;
case 3:
if (p->holoduke_amount > 0 && i > 1)
break;
if (k) dainv = 4;
else dainv = 2;
goto CHECKINV1;
case 0:
case 1:
if (p->firstaid_amount > 0 && i > 1)
break;
if (k) dainv = 2;
else dainv = 7;
goto CHECKINV1;
case 5:
if (p->heat_amount > 0 && i > 1)
break;
if (k) dainv = 6;
else dainv = 4;
goto CHECKINV1;
case 7:
if (p->boot_amount > 0 && i > 1)
break;
if (k) dainv = 1;
else dainv = 6;
goto CHECKINV1;
}
2020-05-17 11:25:39 +00:00
}
else dainv = 0;
2020-05-17 11:25:39 +00:00
// These events force us to keep the inventory selector in the playsim as opposed to the UI where it really belongs.
if (PlayerInput(plnum, SB_INVPREV))
{
SetGameVarID(g_iReturnVarID, dainv, nullptr, plnum);
OnEvent(EVENT_INVENTORYLEFT, plnum, nullptr, -1);
dainv = GetGameVarID(g_iReturnVarID, nullptr, plnum).safeValue();
}
if (PlayerInput(plnum, SB_INVNEXT))
{
SetGameVarID(g_iReturnVarID, dainv, nullptr, plnum);
OnEvent(EVENT_INVENTORYRIGHT, plnum, nullptr, -1);
dainv = GetGameVarID(g_iReturnVarID, nullptr, plnum).safeValue();
}
p->inven_icon = dainv;
// Someone must have really hated constant data, doing this with a switch/case (and of course also with literal numbers...)
static const uint8_t invquotes[] = { QUOTE_MEDKIT, QUOTE_STEROIDS, QUOTE_HOLODUKE, QUOTE_JETPACK, QUOTE_NVG, QUOTE_SCUBA, QUOTE_BOOTS };
if (dainv >= 1 && dainv < 8) FTA(invquotes[dainv - 1], p);
2020-05-17 11:25:39 +00:00
}
int weap = PlayerNewWeapon(plnum);
if (weap > 1 && p->kickback_pic > 0)
p->wantweaponfire = weap - 1;
2020-05-17 11:25:39 +00:00
// Here we have to be extra careful that the weapons do not get mixed up, so let's keep the code for Duke and RR completely separate.
fi.selectweapon(plnum, weap);
2020-05-17 11:25:39 +00:00
if (PlayerInput(plnum, SB_HOLSTER))
2020-05-17 11:25:39 +00:00
{
if (p->curr_weapon > KNEE_WEAPON)
{
if (p->holster_weapon == 0 && p->weapon_pos == 0)
{
p->holster_weapon = 1;
p->weapon_pos = -1;
FTA(QUOTE_WEAPON_LOWERED, p);
}
else if (p->holster_weapon == 1 && p->weapon_pos == -9)
{
p->holster_weapon = 0;
p->weapon_pos = 10;
FTA(QUOTE_WEAPON_RAISED, p);
}
}
}
2020-11-02 23:20:51 +00:00
if (PlayerUseItem(plnum, ICON_HOLODUKE) && (isRR() || p->newOwner == nullptr))
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_HOLODUKEON, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
if (!isRR())
2020-05-17 11:25:39 +00:00
{
if (p->holoduke_on == nullptr)
2020-05-17 11:25:39 +00:00
{
if (p->holoduke_amount > 0)
{
p->inven_icon = 3;
auto pactor =
CreateActor(p->cursector, p->GetActor()->getPosWithOffsetZ().plusZ(30), TILE_APLAYER, -64, DVector2(0, 0), p->GetActor()->spr.Angles.Yaw, 0., 0., nullptr, 10);
pactor->temp_data[3] = pactor->temp_data[4] = 0;
p->holoduke_on = pactor;
pactor->spr.yint = plnum;
pactor->spr.extra = 0;
FTA(QUOTE_HOLODUKE_ON, p);
S_PlayActorSound(TELEPORTER, p->holoduke_on);
}
else FTA(QUOTE_HOLODUKE_NOT_FOUND, p);
2020-05-17 11:25:39 +00:00
}
else
{
S_PlayActorSound(TELEPORTER, p->holoduke_on);
p->holoduke_on = nullptr;
FTA(QUOTE_HOLODUKE_OFF, p);
}
2020-05-17 11:25:39 +00:00
}
else // In RR this means drinking whiskey.
2020-05-17 11:25:39 +00:00
{
if (p->holoduke_amount > 0 && p->GetActor()->spr.extra < gs.max_player_health)
{
p->holoduke_amount -= 400;
p->GetActor()->spr.extra += 5;
if (p->GetActor()->spr.extra > gs.max_player_health)
p->GetActor()->spr.extra = gs.max_player_health;
p->drink_amt += 5;
p->inven_icon = 3;
if (p->holoduke_amount == 0)
checkavailinven(p);
2020-11-02 19:23:48 +00:00
if (p->drink_amt < 99 && !S_CheckActorSoundPlaying(pact, 425))
S_PlayActorSound(425, pact);
}
2020-05-17 11:25:39 +00:00
}
}
}
2020-11-02 23:20:51 +00:00
if (isRR() && PlayerUseItem(plnum, ICON_HEATS) && p->newOwner == nullptr)
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_USENIGHTVISION, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
if (p->yehaa_timer == 0)
2020-05-17 11:25:39 +00:00
{
p->yehaa_timer = 126;
2020-11-02 19:23:48 +00:00
S_PlayActorSound(390, pact);
2022-09-14 15:35:44 +00:00
p->noise_radius = 1024;
madenoise(plnum);
if (p->cursector->lotag == 857)
2020-05-17 11:25:39 +00:00
{
if (p->GetActor()->spr.extra <= gs.max_player_health)
{
p->GetActor()->spr.extra += 10;
if (p->GetActor()->spr.extra >= gs.max_player_health)
p->GetActor()->spr.extra = gs.max_player_health;
}
2020-05-17 11:25:39 +00:00
}
else
2020-05-17 11:25:39 +00:00
{
if (p->GetActor()->spr.extra + 1 <= gs.max_player_health)
{
p->GetActor()->spr.extra++;
}
2020-05-17 11:25:39 +00:00
}
}
}
}
if (PlayerUseItem(plnum, ICON_FIRSTAID))
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_USEMEDKIT, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
if (p->firstaid_amount > 0 && p->GetActor()->spr.extra < gs.max_player_health)
2020-05-17 11:25:39 +00:00
{
if (!isRR())
2020-05-17 11:25:39 +00:00
{
int j = gs.max_player_health - p->GetActor()->spr.extra;
2021-05-12 15:57:36 +00:00
if (p->firstaid_amount > j)
{
p->firstaid_amount -= j;
p->GetActor()->spr.extra = gs.max_player_health;
p->inven_icon = 1;
}
else
{
p->GetActor()->spr.extra += p->firstaid_amount;
p->firstaid_amount = 0;
checkavailinven(p);
}
2020-11-02 19:23:48 +00:00
S_PlayActorSound(DUKE_USEMEDKIT, pact);
2020-05-17 11:25:39 +00:00
}
else
{
int j = 10;
if (p->firstaid_amount > j)
{
p->firstaid_amount -= j;
p->GetActor()->spr.extra += j;
if (p->GetActor()->spr.extra > gs.max_player_health)
p->GetActor()->spr.extra = gs.max_player_health;
p->inven_icon = 1;
}
else
{
p->GetActor()->spr.extra += p->firstaid_amount;
p->firstaid_amount = 0;
checkavailinven(p);
}
if (p->GetActor()->spr.extra > gs.max_player_health)
p->GetActor()->spr.extra = gs.max_player_health;
p->drink_amt += 10;
2020-11-02 19:23:48 +00:00
if (p->drink_amt <= 100 && !S_CheckActorSoundPlaying(pact, DUKE_USEMEDKIT))
S_PlayActorSound(DUKE_USEMEDKIT, pact);
2020-05-17 11:25:39 +00:00
}
}
}
}
2020-11-02 23:20:51 +00:00
if (PlayerUseItem(plnum, ICON_JETPACK) && (isRR() || p->newOwner == nullptr))
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_USEJETPACK, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() == 0)
2020-05-17 11:25:39 +00:00
{
if (!isRR())
2020-05-17 11:25:39 +00:00
{
if (p->jetpack_amount > 0)
2020-05-17 11:25:39 +00:00
{
p->jetpack_on = !p->jetpack_on;
if (p->jetpack_on)
{
p->inven_icon = 4;
S_StopSound(DUKE_SCREAM, pact); // this will stop the falling scream
2020-11-02 19:23:48 +00:00
S_PlayActorSound(DUKE_JETPACK_ON, pact);
FTA(QUOTE_JETPACK_ON, p);
}
else
{
p->hard_landing = 0;
p->vel.Z = 0;
2020-11-02 19:23:48 +00:00
S_PlayActorSound(DUKE_JETPACK_OFF, pact);
S_StopSound(DUKE_JETPACK_IDLE, pact);
S_StopSound(DUKE_JETPACK_ON, pact);
FTA(QUOTE_JETPACK_OFF, p);
}
2020-05-17 11:25:39 +00:00
}
else FTA(QUOTE_JETPACK_NOT_FOUND, p);
2020-05-17 11:25:39 +00:00
}
else
2020-05-17 11:25:39 +00:00
{
// eat cow pie
if (p->jetpack_amount > 0 && p->GetActor()->spr.extra < gs.max_player_health)
2020-05-17 11:25:39 +00:00
{
2020-11-02 19:23:48 +00:00
if (!S_CheckActorSoundPlaying(pact, 429))
S_PlayActorSound(429, pact);
p->jetpack_amount -= 100;
if (p->drink_amt > 0)
{
p->drink_amt -= 5;
if (p->drink_amt < 0)
p->drink_amt = 0;
}
if (p->eat < 100)
{
p->eat += 5;
if (p->eat > 100)
p->eat = 100;
}
p->GetActor()->spr.extra += 5;
2020-05-17 11:25:39 +00:00
p->inven_icon = 4;
2020-05-17 11:25:39 +00:00
if (p->GetActor()->spr.extra > gs.max_player_health)
p->GetActor()->spr.extra = gs.max_player_health;
2020-05-17 11:25:39 +00:00
if (p->jetpack_amount <= 0)
checkavailinven(p);
}
2020-05-17 11:25:39 +00:00
}
}
}
if (PlayerInput(plnum, SB_TURNAROUND) && p->Angles.YawSpin == nullAngle && p->on_crane == nullptr)
2020-05-17 11:25:39 +00:00
{
SetGameVarID(g_iReturnVarID, 0, nullptr, plnum);
OnEvent(EVENT_TURNAROUND, plnum, nullptr, -1);
if (GetGameVarID(g_iReturnVarID, nullptr, plnum).value() != 0)
{
p->sync.actions &= ~SB_TURNAROUND;
}
2020-05-17 11:25:39 +00:00
}
}
}
//---------------------------------------------------------------------------
//
// Main input routine.
// This includes several input improvements from EDuke32, but this code
// has been mostly rewritten completely to make it clearer and reduce redundancy.
//
//---------------------------------------------------------------------------
#if 0
enum
{
TURBOTURNTIME = (TICRATE/8), // 7
NORMALTURN = 15,
PREAMBLETURN = 5,
NORMALKEYMOVE = 40,
MAXVEL = ((NORMALKEYMOVE*2)+10),
MAXSVEL = ((NORMALKEYMOVE*2)+10),
MAXANGVEL = 1024, // 127
MAXHORIZVEL = 256, // 127
};
#endif
enum
{
MAXVELMOTO = 120,
};
static constexpr float VEHICLETURN = (20.f * 360.f / 2048.f);
//---------------------------------------------------------------------------
//
// split out for readability
//
//---------------------------------------------------------------------------
static void doVehicleTilting(player_struct* const p, const float factor, const bool allowed = true)
{
if (const auto turndir = (p->vehTurnRight - p->vehTurnLeft) * allowed)
{
p->TiltStatus += factor * turndir;
if (abs(p->TiltStatus) > 10)
p->TiltStatus = 10.f * turndir;
}
}
static float doVehicleTurning(player_struct* const p, HIDInput* const hidInput, const int kbdDir, const float factor, const float baseVel, const float velScale)
{
// process only if we're turning
if (const auto turning = (p->vehTurnRight || p->moto_drink > 0) - (p->vehTurnLeft || p->moto_drink < 0))
{
const bool noattenuate = (isTurboTurnTime() || hidInput->mouseturnx || hidInput->joyaxes[JOYAXIS_Yaw]) && (!p->OnMotorcycle || p->MotoSpeed > 0);
const auto vel = (noattenuate) ? (baseVel) : (baseVel * velScale);
float turnvel = vel * hidInput->joyaxes[JOYAXIS_Yaw];
if (kbdDir)
{
turnvel += vel * kbdDir;
updateTurnHeldAmt(factor);
}
if (hidInput->mouseturnx)
turnvel += sqrtf(abs(vel * hidInput->mouseturnx / factor) * (7.f / 20.f)) * Sgn(vel) * Sgn(hidInput->mouseturnx);
return turnvel * factor;
}
return 0;
}
static float motoApplyTurn(player_struct* p, HIDInput* const hidInput, const int kbdDir, const float factor)
{
float turnvel = 0;
p->oTiltStatus = p->TiltStatus;
if (p->MotoSpeed == 0 || !p->on_ground)
{
resetTurnHeldAmt();
doVehicleTilting(p, factor);
}
else if (p->vehTurnLeft || p->vehTurnRight || p->moto_drink)
{
constexpr float velScale = (3.f / 10.f);
const float baseVel = VEHICLETURN * Sgn(p->MotoSpeed);
doVehicleTilting(p, factor);
turnvel = doVehicleTurning(p, hidInput, kbdDir, factor, baseVel, velScale);
}
else
{
resetTurnHeldAmt();
p->TiltStatus -= (float)factor * Sgn(p->TiltStatus);
}
if (fabs(p->TiltStatus) < factor)
p->TiltStatus = 0;
return turnvel;
}
//---------------------------------------------------------------------------
//
// same for the boat
//
//---------------------------------------------------------------------------
static float boatApplyTurn(player_struct *p, HIDInput* const hidInput, const int kbdDir, const float factor)
{
float turnvel = 0;
p->oTiltStatus = p->TiltStatus;
if (p->MotoSpeed && (p->vehTurnLeft || p->vehTurnRight || p->moto_drink))
{
const float velScale = !p->NotOnWater? 1.f : (6.f / 19.f);
const float baseVel = VEHICLETURN * velScale;
doVehicleTilting(p, factor, !p->NotOnWater);
turnvel = doVehicleTurning(p, hidInput, kbdDir, factor, baseVel, velScale);
}
else if (!p->NotOnWater)
{
resetTurnHeldAmt();
p->TiltStatus -= (float)factor * Sgn(p->TiltStatus);
}
if (fabs(p->TiltStatus) < factor)
p->TiltStatus = 0;
return turnvel;
}
//---------------------------------------------------------------------------
//
// much of this was rewritten from scratch to make the logic easier to follow.
//
//---------------------------------------------------------------------------
static void processVehicleInput(player_struct *p, HIDInput* const hidInput, InputPacket* const inputBuffer, InputPacket* const currInput, const double scaleAdjust)
{
bool const kbdLeft = buttonMap.ButtonDown(gamefunc_Turn_Left) || buttonMap.ButtonDown(gamefunc_Strafe_Left);
bool const kbdRight = buttonMap.ButtonDown(gamefunc_Turn_Right) || buttonMap.ButtonDown(gamefunc_Strafe_Right);
// Cancel out micro-movement
if (fabs(hidInput->mouseturnx) < (m_sensitivity_x * m_yaw * backendinputscale() * 2.f)) hidInput->mouseturnx = 0;
p->vehTurnLeft = kbdLeft || hidInput->mouseturnx < 0 || hidInput->joyaxes[JOYAXIS_Yaw] > 0;
p->vehTurnRight = kbdRight || hidInput->mouseturnx > 0 || hidInput->joyaxes[JOYAXIS_Yaw] < 0;
if (p->OnBoat || !p->moto_underwater)
{
p->vehForwardScale = min((buttonMap.ButtonDown(gamefunc_Move_Forward) || buttonMap.ButtonDown(gamefunc_Strafe)) + hidInput->joyaxes[JOYAXIS_Forward], 1.f);
p->vehReverseScale = min(buttonMap.ButtonDown(gamefunc_Move_Backward) + -hidInput->joyaxes[JOYAXIS_Forward], 1.f);
p->vehBraking = buttonMap.ButtonDown(gamefunc_Run);
}
if (p->OnMotorcycle)
{
currInput->avel = motoApplyTurn(p, hidInput, kbdRight - kbdLeft, (float)scaleAdjust);
if (p->moto_underwater) p->MotoSpeed = 0;
}
else
{
currInput->avel = boatApplyTurn(p, hidInput, kbdRight - kbdLeft, (float)scaleAdjust);
}
inputBuffer->fvel = clamp<float>((float)p->MotoSpeed, -(MAXVELMOTO >> 3), MAXVELMOTO) * (1.f / 40.f);
inputBuffer->avel += currInput->avel;
}
//---------------------------------------------------------------------------
//
// External entry point
//
//---------------------------------------------------------------------------
void GameInterface::GetInput(HIDInput* const hidInput, InputPacket* const inputBuffer, InputPacket* const currInput, const double scaleAdjust)
{
auto const p = &ps[myconnectindex];
if (isRRRA() && (p->OnMotorcycle || p->OnBoat))
{
processVehicleInput(p, hidInput, inputBuffer, currInput, scaleAdjust);
}
else
{
processMovement(hidInput, inputBuffer, currInput, scaleAdjust, p->drink_amt);
}
}
2020-05-17 11:25:39 +00:00
END_DUKE_NS