mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-09 02:20:49 +00:00
6952 lines
188 KiB
C++
6952 lines
188 KiB
C++
//-------------------------------------------------------------------------
|
|
/*
|
|
Copyright (C) 1997, 2005 - 3D Realms Entertainment
|
|
|
|
This file is part of Shadow Warrior version 1.2
|
|
|
|
Shadow Warrior 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
Original Source: 1997 - Frank Maddin and Jim Norwood
|
|
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
|
|
*/
|
|
//-------------------------------------------------------------------------
|
|
|
|
#include "ns.h"
|
|
|
|
#undef MAIN
|
|
#include "build.h"
|
|
|
|
#include "names2.h"
|
|
#include "panel.h"
|
|
#include "lists.h"
|
|
#include "misc.h"
|
|
#include "network.h"
|
|
#include "pal.h"
|
|
#include "player.h"
|
|
#include "v_2ddrawer.h"
|
|
#include "gamehud.h"
|
|
|
|
#include "weapon.h"
|
|
#include "razemenu.h"
|
|
#include "raze_sound.h"
|
|
|
|
BEGIN_SW_NS
|
|
|
|
|
|
int InitSwordAttack(PLAYER* pp);
|
|
PANEL_SPRITE* InitWeaponUziSecondaryReload(PANEL_SPRITE*);
|
|
PANEL_SPRITE* InitWeaponUzi2(PANEL_SPRITE*);
|
|
int InitShotgun(PLAYER* pp);
|
|
int InitRail(PLAYER* pp);
|
|
int InitMicro(PLAYER* pp);
|
|
int InitRocket(PLAYER* pp);
|
|
int InitNuke(PLAYER* pp);
|
|
int InitGrenade(PLAYER* pp);
|
|
int InitMine(PLAYER* pp);
|
|
int InitFistAttack(PLAYER* pp);
|
|
|
|
struct PANEL_SHRAP
|
|
{
|
|
short xoff, yoff, skip;
|
|
int lo_jump_speed, hi_jump_speed, lo_xspeed, hi_xspeed;
|
|
PANEL_STATE* state[2];
|
|
};
|
|
|
|
void PanelInvTestSuicide(PANEL_SPRITE* psp);
|
|
|
|
void InsertPanelSprite(PLAYER* pp, PANEL_SPRITE* psp);
|
|
void pKillSprite(PANEL_SPRITE* psp);
|
|
void pWeaponBob(PANEL_SPRITE* psp, short condition);
|
|
void pSuicide(PANEL_SPRITE* psp);
|
|
void pNextState(PANEL_SPRITE* psp);
|
|
void pStatePlusOne(PANEL_SPRITE* psp);
|
|
void pSetState(PANEL_SPRITE* psp, PANEL_STATE* panel_state);
|
|
void pStateControl(PANEL_SPRITE* psp);
|
|
|
|
int DoPanelFall(PANEL_SPRITE* psp);
|
|
int DoBeginPanelFall(PANEL_SPRITE* psp);
|
|
int DoPanelJump(PANEL_SPRITE* psp);
|
|
int DoBeginPanelJump(PANEL_SPRITE* psp);
|
|
void SpawnHeartBlood(PANEL_SPRITE* psp);
|
|
void SpawnUziShell(PANEL_SPRITE* psp);
|
|
|
|
bool pWeaponUnHideKeys(PANEL_SPRITE* psp, PANEL_STATE* state);
|
|
bool pWeaponHideKeys(PANEL_SPRITE* psp, PANEL_STATE* state);
|
|
void pHotHeadOverlays(PANEL_SPRITE* psp, short mode);
|
|
|
|
uint8_t UziRecoilYadj = 0;
|
|
|
|
extern short screenpeek;
|
|
|
|
pANIMATOR pNullAnimator;
|
|
int InitStar(PLAYER*);
|
|
int ChangeWeapon(PLAYER*);
|
|
|
|
ANIMATOR InitFire;
|
|
|
|
int NullAnimator(DSWActor*)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void pNullAnimator(PANEL_SPRITE*)
|
|
{
|
|
return;
|
|
}
|
|
|
|
PANEL_SPRITE* pFindMatchingSprite(PLAYER* pp, int x, int y, short pri)
|
|
{
|
|
PANEL_SPRITE* next;
|
|
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (auto psp = list->Next; next = psp->Next, psp != list; psp = next)
|
|
{
|
|
// early out
|
|
if (psp->priority > pri)
|
|
return nullptr;
|
|
|
|
if (psp->pos.X == x && psp->pos.Y == y && psp->priority == pri)
|
|
{
|
|
return psp;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
PANEL_SPRITE* pFindMatchingSpriteID(PLAYER* pp, short id, int x, int y, short pri)
|
|
{
|
|
PANEL_SPRITE* next;
|
|
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (auto psp = list->Next; next = psp->Next, psp != list; psp = next)
|
|
{
|
|
// early out
|
|
if (psp->priority > pri)
|
|
return nullptr;
|
|
|
|
if (psp->ID == id && psp->pos.X == x && psp->pos.Y == y && psp->priority == pri)
|
|
{
|
|
return psp;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
bool pKillScreenSpiteIDs(PLAYER* pp, short id)
|
|
{
|
|
PANEL_SPRITE* next;
|
|
bool found = false;
|
|
|
|
// Kill ALL sprites with the correct id
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (auto psp = list->Next; next = psp->Next, psp != list; psp = next)
|
|
{
|
|
if (psp->ID == id)
|
|
{
|
|
pKillSprite(psp);
|
|
found = true;
|
|
}
|
|
}
|
|
|
|
return found;
|
|
}
|
|
|
|
|
|
// Used to sprites in the view at correct aspect ratio and x,y location.
|
|
|
|
void pSetSuicide(PANEL_SPRITE* psp)
|
|
{
|
|
//psp->flags |= (PANF_SUICIDE);
|
|
//psp->State = nullptr;
|
|
psp->PanelSpriteFunc = pSuicide;
|
|
}
|
|
|
|
void pToggleCrosshair(void)
|
|
{
|
|
cl_crosshair = !cl_crosshair;
|
|
}
|
|
|
|
// Player has a chance of yelling out during combat, when firing a weapon.
|
|
void DoPlayerChooseYell(PLAYER* pp)
|
|
{
|
|
int choose_snd = 0;
|
|
|
|
if (RandomRange(1000) < 990) return;
|
|
|
|
choose_snd = StdRandomRange(MAX_YELLSOUNDS);
|
|
|
|
if (pp == Player+myconnectindex)
|
|
PlayerSound(PlayerYellVocs[choose_snd], v3df_follow|v3df_dontpan,pp);
|
|
}
|
|
|
|
void ArmorCalc(int damage_amt, int *armor_damage, int *player_damage)
|
|
{
|
|
int damage_percent;
|
|
|
|
if (damage_amt == 1)
|
|
{
|
|
*player_damage = RANDOM_P2(2<<8)>>8;
|
|
*armor_damage = 1;
|
|
return;
|
|
}
|
|
|
|
// note: this could easily be converted to a mulscale and save a
|
|
// bit of processing for floats
|
|
damage_percent = int((0.6 * damage_amt)+0.5);
|
|
|
|
*player_damage = damage_amt - damage_percent;
|
|
*armor_damage = damage_percent;
|
|
}
|
|
|
|
|
|
void PlayerUpdateHealth(PLAYER* pp, short value)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
short x,y;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (GodMode)
|
|
{
|
|
if (value < 0)
|
|
return;
|
|
}
|
|
|
|
if (pp->Flags & (PF_DEAD))
|
|
return;
|
|
|
|
if (value < 0)
|
|
{
|
|
bool IsChem = false;
|
|
bool NoArmor = false;
|
|
|
|
if (value <= -2000)
|
|
{
|
|
value += 2000;
|
|
NoArmor = true;
|
|
}
|
|
else if (value <= -1000)
|
|
{
|
|
value += 1000;
|
|
IsChem = true;
|
|
}
|
|
|
|
// TAKE SOME DAMAGE
|
|
plActor->user.LastDamage = -value;
|
|
|
|
// adjust for armor
|
|
if (pp->Armor && !NoArmor)
|
|
{
|
|
int armor_damage, player_damage;
|
|
ArmorCalc(labs(value), &armor_damage, &player_damage);
|
|
PlayerUpdateArmor(pp, -armor_damage);
|
|
value = -player_damage;
|
|
}
|
|
|
|
plActor->user.Health += value;
|
|
|
|
if (value < 0)
|
|
{
|
|
int choosesnd = 0;
|
|
|
|
choosesnd = RandomRange(MAX_PAIN);
|
|
|
|
if (plActor->user.Health > 50)
|
|
{
|
|
PlayerSound(PlayerPainVocs[choosesnd],v3df_dontpan|v3df_doppler|v3df_follow,pp);
|
|
}
|
|
else
|
|
{
|
|
PlayerSound(PlayerLowHealthPainVocs[choosesnd],v3df_dontpan|v3df_doppler|v3df_follow,pp);
|
|
}
|
|
|
|
}
|
|
// Do redness based on damage taken.
|
|
if (pp == Player + screenpeek)
|
|
{
|
|
if (IsChem)
|
|
SetFadeAmt(pp,-40,144); // ChemBomb green color
|
|
else
|
|
{
|
|
if (value <= -10)
|
|
SetFadeAmt(pp,value,112);
|
|
else if (value > -10 && value < 0)
|
|
SetFadeAmt(pp,-20,112);
|
|
}
|
|
}
|
|
if (plActor->user.Health <= 100)
|
|
pp->MaxHealth = 100; // Reset max health if sank below 100
|
|
}
|
|
else
|
|
{
|
|
// ADD SOME HEALTH
|
|
if (value > 1000)
|
|
plActor->user.Health += (value-1000);
|
|
else
|
|
{
|
|
if (plActor->user.Health < pp->MaxHealth)
|
|
{
|
|
plActor->user.Health+=value;
|
|
}
|
|
}
|
|
|
|
|
|
if (value >= 1000)
|
|
value -= 1000; // Strip out high value
|
|
}
|
|
|
|
if (plActor->user.Health < 0)
|
|
plActor->user.Health = 0;
|
|
|
|
if (plActor->user.Health > pp->MaxHealth)
|
|
plActor->user.Health = pp->MaxHealth;
|
|
}
|
|
|
|
void PlayerUpdateAmmo(PLAYER* pp, short UpdateWeaponNum, short value)
|
|
{
|
|
short x,y;
|
|
short WeaponNum;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (DamageData[UpdateWeaponNum].max_ammo == -1)
|
|
{
|
|
return;
|
|
}
|
|
|
|
WeaponNum = UpdateWeaponNum;
|
|
|
|
// get the WeaponNum of the ammo
|
|
if (DamageData[UpdateWeaponNum].with_weapon != -1)
|
|
{
|
|
WeaponNum = DamageData[UpdateWeaponNum].with_weapon;
|
|
}
|
|
|
|
pp->WpnAmmo[WeaponNum] += value;
|
|
|
|
if (pp->WpnAmmo[WeaponNum] <= 0)
|
|
{
|
|
// star and mine
|
|
if (WeaponIsAmmo & BIT(WeaponNum))
|
|
pp->WpnFlags &= ~BIT(WeaponNum);
|
|
|
|
pp->WpnAmmo[WeaponNum] = 0;
|
|
}
|
|
|
|
if (pp->WpnAmmo[WeaponNum] > DamageData[WeaponNum].max_ammo)
|
|
{
|
|
pp->WpnAmmo[WeaponNum] = DamageData[WeaponNum].max_ammo;
|
|
}
|
|
}
|
|
|
|
void PlayerUpdateWeapon(PLAYER* pp, short WeaponNum)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
|
|
// weapon Change
|
|
if (Prediction)
|
|
return;
|
|
|
|
plActor->user.WeaponNum = int8_t(WeaponNum);
|
|
}
|
|
|
|
void PlayerUpdateKills(PLAYER* pp, short value)
|
|
{
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (gNet.MultiGameType == MULTI_GAME_COOPERATIVE)
|
|
return;
|
|
|
|
// Team play
|
|
if (gNet.MultiGameType == MULTI_GAME_COMMBAT && gNet.TeamPlay)
|
|
{
|
|
short pnum;
|
|
PLAYER* opp;
|
|
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
opp = Player + pnum;
|
|
|
|
// for everyone on the same team
|
|
if (opp != pp && opp->actor->user.spal == pp->actor->user.spal)
|
|
{
|
|
opp->Kills += value;
|
|
if (opp->Kills > 999)
|
|
opp->Kills = 0;
|
|
if (opp->Kills < -99)
|
|
opp->Kills = -99;
|
|
}
|
|
}
|
|
}
|
|
|
|
pp->Kills += value;
|
|
if (pp->Kills > 999)
|
|
pp->Kills = 0;
|
|
if (pp->Kills < -99)
|
|
pp->Kills = -99;
|
|
}
|
|
|
|
void PlayerUpdateArmor(PLAYER* pp, short value)
|
|
{
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (value >= 1000)
|
|
pp->Armor = value-1000;
|
|
else
|
|
pp->Armor += value;
|
|
|
|
if (pp->Armor > 100)
|
|
pp->Armor = 100;
|
|
if (pp->Armor < 0)
|
|
pp->Armor = 0;
|
|
}
|
|
|
|
|
|
int WeaponOperate(PLAYER* pp)
|
|
{
|
|
short weapon;
|
|
DSWActor* plActor = pp->actor;
|
|
|
|
InventoryKeys(pp);
|
|
|
|
// UziType >= 3 means they are reloading
|
|
if (pp->WpnUziType >= 3) return true;
|
|
|
|
//if (CheatInputMode)
|
|
// return (0);
|
|
|
|
if (pp->FirePause != 0)
|
|
{
|
|
pp->FirePause -= synctics;
|
|
if (pp->FirePause <= 0)
|
|
pp->FirePause = 0;
|
|
}
|
|
|
|
if (pp->sop)
|
|
{
|
|
switch (pp->sop->track)
|
|
{
|
|
case SO_VEHICLE:
|
|
case SO_TURRET:
|
|
case SO_TURRET_MGUN:
|
|
// case SO_SPEED_BOAT:
|
|
|
|
if (!(pp->sop->flags & SOBJ_HAS_WEAPON))
|
|
break;
|
|
|
|
if (pp->input.actions & SB_FIRE)
|
|
{
|
|
if (pp->KeyPressBits & SB_FIRE)
|
|
{
|
|
if (!pp->FirePause)
|
|
{
|
|
InitSobjGun(pp);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
weapon = pp->input.getNewWeapon();
|
|
|
|
if (weapon)
|
|
{
|
|
if (pp->KeyPressBits & SB_FIRST_WEAPON_BIT)
|
|
{
|
|
pp->KeyPressBits &= ~SB_FIRST_WEAPON_BIT;
|
|
|
|
weapon -= 1;
|
|
|
|
// Special uzi crap
|
|
if (weapon != WPN_UZI) pp->WpnUziType = 2; // This means we aren't on uzi
|
|
|
|
switch (weapon)
|
|
{
|
|
case WPN_FIST:
|
|
//case WPN_SWORD:
|
|
if (plActor->user.WeaponNum == WPN_FIST
|
|
|| plActor->user.WeaponNum == WPN_SWORD)
|
|
{
|
|
// toggle
|
|
if (pp->WpnFirstType == WPN_FIST)
|
|
pp->WpnFirstType = WPN_SWORD;
|
|
else if (pp->WpnFirstType == WPN_SWORD)
|
|
pp->WpnFirstType = WPN_FIST;
|
|
}
|
|
|
|
switch (pp->WpnFirstType)
|
|
{
|
|
case WPN_SWORD:
|
|
InitWeaponSword(pp);
|
|
break;
|
|
case WPN_FIST:
|
|
InitWeaponFist(pp);
|
|
break;
|
|
}
|
|
break;
|
|
case WPN_STAR:
|
|
InitWeaponStar(pp);
|
|
break;
|
|
case WPN_UZI:
|
|
if (plActor->user.WeaponNum == WPN_UZI)
|
|
{
|
|
if (pp->Flags & (PF_TWO_UZI))
|
|
{
|
|
pp->WpnUziType++;
|
|
PlaySound(DIGI_UZI_UP, pp, v3df_follow);
|
|
if (pp->WpnUziType > 1)
|
|
pp->WpnUziType = 0;
|
|
}
|
|
else
|
|
pp->WpnUziType = 1; // Use retracted state for single uzi
|
|
}
|
|
InitWeaponUzi(pp);
|
|
break;
|
|
case WPN_MICRO:
|
|
if (plActor->user.WeaponNum == WPN_MICRO)
|
|
{
|
|
pp->WpnRocketType++;
|
|
PlaySound(DIGI_ROCKET_UP, pp, v3df_follow);
|
|
if (pp->WpnRocketType > 2)
|
|
pp->WpnRocketType = 0;
|
|
if (pp->WpnRocketType == 2 && pp->WpnRocketNuke == 0)
|
|
pp->WpnRocketType = 0;
|
|
if (pp->WpnRocketType == 2)
|
|
pp->TestNukeInit = true; // Init the nuke
|
|
else
|
|
pp->TestNukeInit = false;
|
|
}
|
|
InitWeaponMicro(pp);
|
|
break;
|
|
case WPN_SHOTGUN:
|
|
if (plActor->user.WeaponNum == WPN_SHOTGUN)
|
|
{
|
|
pp->WpnShotgunType++;
|
|
if (pp->WpnShotgunType > 1)
|
|
pp->WpnShotgunType = 0;
|
|
PlaySound(DIGI_SHOTGUN_UP, pp, v3df_follow);
|
|
}
|
|
InitWeaponShotgun(pp);
|
|
break;
|
|
case WPN_RAIL:
|
|
if (!SW_SHAREWARE)
|
|
{
|
|
#if 0
|
|
if (plActor->user.WeaponNum == WPN_RAIL)
|
|
{
|
|
pp->WpnRailType++;
|
|
if (pp->WpnRailType > 1)
|
|
pp->WpnRailType = 0;
|
|
}
|
|
if (pp->WpnRailType == 1)
|
|
PlaySound(DIGI_RAIL_UP, pp, v3df_follow);
|
|
#endif
|
|
InitWeaponRail(pp);
|
|
}
|
|
else
|
|
{
|
|
PutStringInfo(pp,"Order the full version");
|
|
}
|
|
break;
|
|
case WPN_HOTHEAD:
|
|
if (!SW_SHAREWARE)
|
|
{
|
|
if (plActor->user.WeaponNum == WPN_HOTHEAD
|
|
|| plActor->user.WeaponNum == WPN_RING
|
|
|| plActor->user.WeaponNum == WPN_NAPALM)
|
|
{
|
|
pp->WpnFlameType++;
|
|
if (pp->WpnFlameType > 2)
|
|
pp->WpnFlameType = 0;
|
|
// if(pp->Wpn[WPN_HOTHEAD])
|
|
pHotHeadOverlays(pp->Wpn[WPN_HOTHEAD], pp->WpnFlameType);
|
|
PlaySound(DIGI_HOTHEADSWITCH, pp, v3df_dontpan|v3df_follow);
|
|
}
|
|
|
|
InitWeaponHothead(pp);
|
|
}
|
|
else
|
|
{
|
|
PutStringInfo(pp,"Order the full version");
|
|
}
|
|
break;
|
|
case WPN_HEART:
|
|
if (!SW_SHAREWARE)
|
|
{
|
|
InitWeaponHeart(pp);
|
|
}
|
|
else
|
|
{
|
|
PutStringInfo(pp,"Order the full version");
|
|
}
|
|
break;
|
|
case WPN_GRENADE:
|
|
InitWeaponGrenade(pp);
|
|
break;
|
|
case WPN_MINE:
|
|
//InitChops(pp);
|
|
InitWeaponMine(pp);
|
|
break;
|
|
case 13:
|
|
pp->WpnFirstType = WPN_FIST;
|
|
InitWeaponFist(pp);
|
|
break;
|
|
case 14:
|
|
pp->WpnFirstType = WPN_SWORD;
|
|
InitWeaponSword(pp);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pp->KeyPressBits |= SB_FIRST_WEAPON_BIT;
|
|
}
|
|
|
|
// Shut that computer chick up if weapon has changed!
|
|
// This really should be handled better, but since there's no usable tracking state for the sounds, the easiest way to handle them is to play on otherwise unused channels.
|
|
if (pp->WpnRocketType != 2 || pp->CurWpn != pp->Wpn[WPN_MICRO])
|
|
{
|
|
pp->InitingNuke = false;
|
|
soundEngine->StopSound(SOURCE_Player, pp, CHAN_WEAPON);
|
|
}
|
|
if (pp->CurWpn != pp->Wpn[WPN_RAIL])
|
|
{
|
|
soundEngine->StopSound(SOURCE_Player, pp, CHAN_ITEM);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool WeaponOK(PLAYER* pp)
|
|
{
|
|
short min_ammo, WeaponNum, FindWeaponNum;
|
|
static const uint8_t wpn_order[] = {2,3,4,5,6,7,8,9,1,0};
|
|
unsigned wpn_ndx=0;
|
|
|
|
DSWActor* plActor = pp->actor;
|
|
|
|
if (!plActor || !plActor->hasU())
|
|
return(false);
|
|
|
|
// sword
|
|
if (DamageData[plActor->user.WeaponNum].max_ammo == -1)
|
|
return true;
|
|
|
|
WeaponNum = plActor->user.WeaponNum;
|
|
FindWeaponNum = plActor->user.WeaponNum;
|
|
|
|
min_ammo = DamageData[WeaponNum].min_ammo;
|
|
|
|
// if ran out of ammo switch to something else
|
|
if (pp->WpnAmmo[WeaponNum] < min_ammo)
|
|
{
|
|
if (plActor->user.WeaponNum == WPN_UZI) pp->WpnUziType = 2; // Set it for retract
|
|
|
|
// Still got a nuke, it's ok.
|
|
if (WeaponNum == WPN_MICRO && pp->WpnRocketNuke)
|
|
{
|
|
//pp->WpnRocketType = 2; // Set it to Nuke
|
|
if (!pp->NukeInitialized) pp->TestNukeInit = true;
|
|
|
|
plActor->user.WeaponNum = WPN_MICRO;
|
|
(*DamageData[plActor->user.WeaponNum].Init)(pp);
|
|
|
|
return true;
|
|
}
|
|
|
|
pp->KeyPressBits &= ~SB_FIRE;
|
|
|
|
FindWeaponNum = WPN_SHOTGUN; // Start at the top
|
|
|
|
while (true)
|
|
{
|
|
// ran out of weapons - choose SWORD
|
|
if (wpn_ndx > sizeof(wpn_order))
|
|
{
|
|
FindWeaponNum = WPN_SWORD;
|
|
break;
|
|
}
|
|
|
|
// if you have the weapon and the ammo is greater than 0
|
|
if (pp->WpnFlags & (BIT(FindWeaponNum)) && pp->WpnAmmo[FindWeaponNum] >= min_ammo)
|
|
break;
|
|
|
|
wpn_ndx++;
|
|
FindWeaponNum = wpn_order[wpn_ndx];
|
|
}
|
|
|
|
plActor->user.WeaponNum = int8_t(FindWeaponNum);
|
|
|
|
if (plActor->user.WeaponNum == WPN_HOTHEAD)
|
|
{
|
|
pp->WeaponType = WPN_HOTHEAD;
|
|
pp->WpnFlameType = 0;
|
|
}
|
|
|
|
(*DamageData[plActor->user.WeaponNum].Init)(pp);
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// X/Y POSITION INLINES
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
inline double pspSinVel(PANEL_SPRITE* const psp, int const ang = INT_MAX)
|
|
{
|
|
return psp->vel * synctics * bsinf(ang == INT_MAX ? psp->ang : ang, -22);
|
|
}
|
|
|
|
inline double pspCosVel(PANEL_SPRITE* const psp, int const ang = INT_MAX)
|
|
{
|
|
return psp->vel * synctics * bcosf(ang == INT_MAX ? psp->ang : ang, -22);
|
|
}
|
|
|
|
inline double pspPresentRetractScale(int const picnum, double const defaultheight)
|
|
{
|
|
double const picheight = tileHeight(picnum);
|
|
return picheight == defaultheight ? 1 : picheight / defaultheight;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SWORD
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
static short SwordAngTable[] =
|
|
{
|
|
82,
|
|
168,
|
|
256+64
|
|
};
|
|
|
|
short SwordAng = 0;
|
|
|
|
void SwordBlur(PANEL_SPRITE* psp)
|
|
{
|
|
psp->kill_tics -= synctics;
|
|
if (psp->kill_tics <= 0)
|
|
{
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
else if (psp->kill_tics <= 6)
|
|
{
|
|
psp->flags |= (PANF_TRANS_FLIP);
|
|
}
|
|
|
|
psp->shade += 10;
|
|
// change sprites priority
|
|
REMOVE(psp);
|
|
psp->priority--;
|
|
InsertPanelSprite(psp->PlayerP, psp);
|
|
}
|
|
|
|
void SpawnSwordBlur(PANEL_SPRITE* psp)
|
|
{
|
|
if (cl_nomeleeblur) return;
|
|
|
|
PANEL_SPRITE* nsp;
|
|
//PICITEMp pip;
|
|
|
|
if (psp->PlayerP->SwordAng > 200)
|
|
return;
|
|
|
|
nsp = pSpawnSprite(psp->PlayerP, nullptr, PRI_BACK, psp->pos.X, psp->pos.Y);
|
|
|
|
nsp->flags |= (PANF_WEAPON_SPRITE);
|
|
nsp->ang = psp->ang;
|
|
nsp->vel = psp->vel;
|
|
nsp->PanelSpriteFunc = SwordBlur;
|
|
nsp->kill_tics = 9;
|
|
nsp->shade = psp->shade + 10;
|
|
|
|
nsp->picndx = -1;
|
|
nsp->picnum = psp->picndx;
|
|
|
|
if ((psp->State->flags & psf_Xflip))
|
|
nsp->flags |= (PANF_XFLIP);
|
|
|
|
nsp->rotate_ang = psp->rotate_ang;
|
|
nsp->scale = psp->scale;
|
|
|
|
nsp->flags |= (PANF_TRANSLUCENT);
|
|
}
|
|
|
|
void pSwordPresent(PANEL_SPRITE* psp);
|
|
void pSwordRetract(PANEL_SPRITE* psp);
|
|
void pSwordAction(PANEL_SPRITE* psp);
|
|
void pSwordRest(PANEL_SPRITE* psp);
|
|
void pSwordAttack(PANEL_SPRITE* psp);
|
|
void pSwordSlide(PANEL_SPRITE* psp);
|
|
void pSwordSlideDown(PANEL_SPRITE* psp);
|
|
void pSwordHide(PANEL_SPRITE* psp);
|
|
void pSwordUnHide(PANEL_SPRITE* psp);
|
|
|
|
void pSwordSlideR(PANEL_SPRITE* psp);
|
|
void pSwordSlideDownR(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_SwordSwing[];
|
|
extern PANEL_STATE ps_ReloadSword[];
|
|
|
|
#define Sword_BEAT_RATE 24
|
|
#define Sword_ACTION_RATE 10
|
|
|
|
PANEL_STATE ps_PresentSword[] =
|
|
{
|
|
{ID_SwordPresent0, Sword_BEAT_RATE, pSwordPresent, &ps_PresentSword[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_SwordRest[] =
|
|
{
|
|
{ID_SwordPresent0, Sword_BEAT_RATE, pSwordRest, &ps_SwordRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_SwordHide[] =
|
|
{
|
|
{ID_SwordPresent0, Sword_BEAT_RATE, pSwordHide, &ps_SwordHide[0], 0,0,0}
|
|
};
|
|
|
|
#define SWORD_PAUSE_TICS 10
|
|
//#define SWORD_SLIDE_TICS 14
|
|
#define SWORD_SLIDE_TICS 10
|
|
#define SWORD_MID_SLIDE_TICS 14
|
|
|
|
PANEL_STATE ps_SwordSwing[] =
|
|
{
|
|
{ID_SwordSwing0, SWORD_PAUSE_TICS, pNullAnimator, &ps_SwordSwing[1], 0,0,0},
|
|
{ID_SwordSwing1, SWORD_SLIDE_TICS, /* start slide */ pSwordSlide, &ps_SwordSwing[2], 0,0,0},
|
|
{ID_SwordSwing1, 0, /* damage */ pSwordAttack, &ps_SwordSwing[3], psf_QuickCall, 0,0},
|
|
{ID_SwordSwing2, SWORD_MID_SLIDE_TICS, /* mid slide */ pSwordSlideDown, &ps_SwordSwing[4], 0,0,0},
|
|
|
|
{ID_SwordSwing2, 99, /* end slide */ pSwordSlideDown, &ps_SwordSwing[4], 0,0,0},
|
|
|
|
{ID_SwordSwingR1, SWORD_SLIDE_TICS, /* start slide */ pSwordSlideR, &ps_SwordSwing[6], psf_Xflip, 0,0},
|
|
{ID_SwordSwingR2, 0, /* damage */ pSwordAttack, &ps_SwordSwing[7], psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_SwordSwingR2, SWORD_MID_SLIDE_TICS, /* mid slide */ pSwordSlideDownR, &ps_SwordSwing[8], psf_Xflip, 0,0},
|
|
|
|
{ID_SwordSwingR2, 99, /* end slide */ pSwordSlideDownR, &ps_SwordSwing[8], psf_Xflip, 0,0},
|
|
{ID_SwordSwingR2, 2, /* end slide */ pNullAnimator, &ps_SwordSwing[1], psf_Xflip, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RetractSword[] =
|
|
{
|
|
{ID_SwordPresent0, Sword_BEAT_RATE, pSwordRetract, &ps_RetractSword[0], 0,0,0}
|
|
};
|
|
|
|
#define SWORD_SWAY_AMT 12
|
|
|
|
// left swing
|
|
#define SWORD_XOFF (320 + SWORD_SWAY_AMT)
|
|
#define SWORD_YOFF 200
|
|
|
|
// right swing
|
|
#define SWORDR_XOFF (0 - 80)
|
|
|
|
#define SWORD_VEL 1700
|
|
#define SWORD_POWER_VEL 2500
|
|
|
|
|
|
void SpecialUziRetractFunc(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupy();
|
|
psp->pos.Y += 4 * synctics;
|
|
|
|
if (psp->pos.Y >= 200 + tileHeight(psp->picnum))
|
|
{
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
void RetractCurWpn(PLAYER* pp)
|
|
{
|
|
// Retract old weapon
|
|
if (pp->CurWpn)
|
|
{
|
|
PANEL_SPRITE* cur,* nxt;
|
|
|
|
if ((pp->CurWpn == pp->Wpn[WPN_UZI] && pp->WpnUziType == 2) || pp->CurWpn != pp->Wpn[WPN_UZI])
|
|
{
|
|
pSetState(pp->CurWpn, pp->CurWpn->RetractState);
|
|
pp->Flags |= (PF_WEAPON_RETRACT);
|
|
}
|
|
|
|
if (pp->CurWpn->sibling)
|
|
{
|
|
// primarily for double uzi to take down the second uzi
|
|
pSetState(pp->CurWpn->sibling, pp->CurWpn->sibling->RetractState);
|
|
}
|
|
else
|
|
{
|
|
// check for any outstanding siblings that need to go away also
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (cur = list->Next; nxt = cur->Next, cur != list; cur = nxt)
|
|
{
|
|
if (cur->sibling && cur->sibling == pp->CurWpn)
|
|
{
|
|
// special case for uzi reload pieces
|
|
cur->picnum = cur->picndx;
|
|
cur->State = nullptr;
|
|
cur->PanelSpriteFunc = SpecialUziRetractFunc;
|
|
cur->sibling = nullptr;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void InitWeaponSword(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
short rnd_num;
|
|
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_SWORD)) ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
// needed for death sequence when the SWORD was your weapon when you died
|
|
if (pp->Wpn[WPN_SWORD] && (pp->Wpn[WPN_SWORD]->flags & PANF_DEATH_HIDE))
|
|
{
|
|
pp->Wpn[WPN_SWORD]->flags &= ~(PANF_DEATH_HIDE);
|
|
pp->Flags &= ~(PF_WEAPON_RETRACT|PF_WEAPON_DOWN);
|
|
pSetState(pp->CurWpn, pp->CurWpn->PresentState);
|
|
return;
|
|
}
|
|
|
|
|
|
if (!pp->Wpn[WPN_SWORD])
|
|
{
|
|
psp = pp->Wpn[WPN_SWORD] = pSpawnSprite(pp, ps_PresentSword, PRI_MID, SWORD_XOFF, SWORD_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_SWORD])
|
|
{
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_SWORD);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_SWORD];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_SwordSwing;
|
|
psp->RetractState = ps_RetractSword;
|
|
psp->PresentState = ps_PresentSword;
|
|
psp->RestState = ps_SwordRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
PlaySound(DIGI_SWORD_UP, pp, v3df_follow|v3df_dontpan);
|
|
|
|
if (pp == Player+myconnectindex && PlayClock > 0)
|
|
{
|
|
rnd_num = StdRandomRange(1024);
|
|
if (rnd_num > 900)
|
|
PlaySound(DIGI_TAUNTAI2, pp, v3df_follow|v3df_dontpan);
|
|
else if (rnd_num > 800)
|
|
PlaySound(DIGI_PLAYERYELL1, pp, v3df_follow|v3df_dontpan);
|
|
else if (rnd_num > 700)
|
|
PlaySound(DIGI_PLAYERYELL2, pp, v3df_follow|v3df_dontpan);
|
|
else if (rnd_num > 600)
|
|
PlayerSound(DIGI_ILIKESWORD, v3df_follow|v3df_dontpan,pp);
|
|
}
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
|
|
void pSwordPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics * pspPresentRetractScale(psp->picndx, 136);
|
|
|
|
if (psp->pos.Y < SWORD_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = SWORD_YOFF;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
//
|
|
// LEFT SWING
|
|
//
|
|
|
|
void pSwordSlide(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnSwordBlur(psp);
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel += 24 * synctics;
|
|
}
|
|
|
|
void pSwordSlideDown(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnSwordBlur(psp);
|
|
|
|
auto ang = SwordAng + psp->ang + psp->PlayerP->SwordAng;
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp, ang);
|
|
psp->pos.Y -= pspSinVel(psp, ang);
|
|
|
|
psp->vel += 20 * synctics;
|
|
|
|
if (psp->pos.X < -40)
|
|
{
|
|
// if still holding down the fire key - continue swinging
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
// continue to next state to swing right
|
|
pStatePlusOne(psp);
|
|
psp->opos.X = psp->pos.X = SWORDR_XOFF;
|
|
psp->opos.Y = psp->pos.Y = SWORD_YOFF;
|
|
psp->backupboby();
|
|
psp->ang = 1024;
|
|
psp->PlayerP->SwordAng = SwordAngTable[RandomRange(SIZ(SwordAngTable))];
|
|
psp->vel = 2500;
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// NOT still holding down the fire key - stop swinging
|
|
pSetState(psp, psp->PresentState);
|
|
psp->opos.X = psp->pos.X = SWORD_XOFF;
|
|
psp->opos.Y = psp->pos.Y = SWORD_YOFF + tileHeight(psp->picndx);
|
|
psp->backupboby();
|
|
}
|
|
}
|
|
|
|
//
|
|
// RIGHT SWING
|
|
//
|
|
|
|
void pSwordSlideR(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnSwordBlur(psp);
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X -= pspCosVel(psp);
|
|
psp->pos.Y += pspSinVel(psp);
|
|
|
|
psp->vel += 24 * synctics;
|
|
}
|
|
|
|
void pSwordSlideDownR(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnSwordBlur(psp);
|
|
|
|
auto ang = SwordAng + psp->ang - psp->PlayerP->SwordAng;
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X -= pspCosVel(psp, ang);
|
|
psp->pos.Y += pspSinVel(psp, ang);
|
|
|
|
psp->vel += 24 * synctics;
|
|
|
|
if (psp->pos.X > 350)
|
|
{
|
|
// if still holding down the fire key - continue swinging
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
// back to action state
|
|
pStatePlusOne(psp);
|
|
psp->opos.X = psp->pos.X = SWORD_XOFF + 80;
|
|
psp->opos.Y = psp->pos.Y = SWORD_YOFF;
|
|
psp->backupboby();
|
|
psp->PlayerP->SwordAng = SwordAngTable[RandomRange(SIZ(SwordAngTable))];
|
|
psp->ang = 1024;
|
|
psp->vel = 2500;
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// NOT still holding down the fire key - stop swinging
|
|
pSetState(psp, psp->PresentState);
|
|
psp->opos.X = psp->pos.X = SWORD_XOFF;
|
|
psp->opos.Y = psp->pos.Y = SWORD_YOFF + tileHeight(psp->picndx);
|
|
psp->backupboby();
|
|
}
|
|
}
|
|
|
|
void pSwordBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = SWORD_SWAY_AMT;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pSwordHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= SWORD_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = SWORD_YOFF + tileHeight(picnum);
|
|
psp->opos.X = psp->pos.X = SWORD_XOFF;
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pSwordRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_SwordHide))
|
|
return;
|
|
|
|
psp->bobpos.Y += synctics;
|
|
|
|
if (psp->bobpos.Y > SWORD_YOFF)
|
|
{
|
|
psp->bobpos.Y = SWORD_YOFF;
|
|
}
|
|
|
|
pSwordBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
pSetState(psp, psp->ActionState);
|
|
|
|
psp->vel = 2500;
|
|
|
|
psp->ang = 1024;
|
|
psp->PlayerP->SwordAng = SwordAngTable[RandomRange(SIZ(SwordAngTable))];
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
}
|
|
}
|
|
}
|
|
|
|
void pSwordAction(PANEL_SPRITE* psp)
|
|
{
|
|
pSwordBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pSwordAttack(PANEL_SPRITE* psp)
|
|
{
|
|
|
|
InitSwordAttack(psp->PlayerP);
|
|
}
|
|
|
|
void pSwordRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics * pspPresentRetractScale(picnum, 136);
|
|
|
|
if (psp->pos.Y >= SWORD_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_SWORD] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// STAR
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pStarPresent(PANEL_SPRITE* psp);
|
|
void pStarRetract(PANEL_SPRITE* psp);
|
|
void pStarAction(PANEL_SPRITE* psp);
|
|
void pStarRest(PANEL_SPRITE* psp);
|
|
void pStarThrow(PANEL_SPRITE* psp);
|
|
|
|
void pStarGrab(PANEL_SPRITE* psp);
|
|
void pStarDone(PANEL_SPRITE* psp);
|
|
void pStarFollowThru(PANEL_SPRITE* psp);
|
|
void pStarFollowThru2(PANEL_SPRITE* psp);
|
|
void pStarStartThrow(PANEL_SPRITE* psp);
|
|
|
|
void pStarFollowUp(PANEL_SPRITE* psp);
|
|
void pStarFollowDown(PANEL_SPRITE* psp);
|
|
|
|
void pStarHide(PANEL_SPRITE* psp);
|
|
void pStarRestTest(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_StarThrow[];
|
|
|
|
#define PRESENT_STAR_RATE 5
|
|
|
|
PANEL_STATE ps_PresentStar[] =
|
|
{
|
|
{ID_StarPresent0, PRESENT_STAR_RATE, pStarPresent, &ps_PresentStar[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_StarHide[] =
|
|
{
|
|
{ID_StarPresent0, PRESENT_STAR_RATE, pStarHide, &ps_StarHide[0], 0,0,0}
|
|
};
|
|
|
|
#define Star_RATE 2 // was 5
|
|
|
|
PANEL_STATE ps_StarRest[] =
|
|
{
|
|
{ID_StarPresent0, Star_RATE, pStarRest, &ps_StarRest[0], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_ThrowStar[] =
|
|
{
|
|
{ID_StarDown0, Star_RATE+3, pNullAnimator, &ps_ThrowStar[1], 0,0,0},
|
|
{ID_StarDown1, Star_RATE+3, pNullAnimator, &ps_ThrowStar[2], 0,0,0},
|
|
{ID_StarDown1, Star_RATE*2, pNullAnimator, &ps_ThrowStar[3], psf_Invisible, 0,0},
|
|
{ID_StarDown1, Star_RATE, pNullAnimator, &ps_ThrowStar[4], 0,0,0},
|
|
{ID_StarDown0, Star_RATE, pNullAnimator, &ps_ThrowStar[5], 0,0,0},
|
|
{ID_ThrowStar0, 1, pNullAnimator, &ps_ThrowStar[6], 0,0,0},
|
|
{ID_ThrowStar0, Star_RATE, pStarThrow, &ps_ThrowStar[7], psf_QuickCall, 0,0},
|
|
{ID_ThrowStar0, Star_RATE, pNullAnimator, &ps_ThrowStar[8], 0,0,0},
|
|
{ID_ThrowStar1, Star_RATE, pNullAnimator, &ps_ThrowStar[9], 0,0,0},
|
|
{ID_ThrowStar2, Star_RATE*2, pNullAnimator, &ps_ThrowStar[10], 0,0,0},
|
|
{ID_ThrowStar3, Star_RATE*2, pNullAnimator, &ps_ThrowStar[11], 0,0,0},
|
|
{ID_ThrowStar4, Star_RATE*2, pNullAnimator, &ps_ThrowStar[12], 0,0,0},
|
|
// start up
|
|
{ID_StarDown1, Star_RATE+3, pNullAnimator, &ps_ThrowStar[13], 0,0,0},
|
|
{ID_StarDown0, Star_RATE+3, pNullAnimator, &ps_ThrowStar[14], 0,0,0},
|
|
{ID_StarPresent0, Star_RATE+3, pNullAnimator, &ps_ThrowStar[15], 0,0,0},
|
|
// maybe to directly to rest state
|
|
{ID_StarDown0, 3, pStarRestTest, &ps_ThrowStar[16], psf_QuickCall, 0,0},
|
|
// if holding the fire key we get to here
|
|
{ID_ThrowStar4, 3, pNullAnimator, &ps_ThrowStar[5], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RetractStar[] =
|
|
{
|
|
{ID_StarPresent0, PRESENT_STAR_RATE, pStarRetract, &ps_RetractStar[0], 0,0,0}
|
|
};
|
|
|
|
//
|
|
// Right hand star routines
|
|
//
|
|
|
|
//#define STAR_YOFF 220
|
|
//#define STAR_XOFF (160+25)
|
|
#define STAR_YOFF 208
|
|
#define STAR_XOFF (160+80)
|
|
|
|
void pStarRestTest(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
// continue to next state to swing right
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
pStatePlusOne(psp);
|
|
return;
|
|
}
|
|
}
|
|
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
|
|
void InitWeaponStar(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp = nullptr;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_STAR)) ||
|
|
pp->WpnAmmo[WPN_STAR] < 3 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
{
|
|
//pp->WpnFirstType = WPN_SWORD;
|
|
//InitWeaponSword(pp);
|
|
return;
|
|
}
|
|
|
|
// needed for death sequence when the STAR was your weapon when you died
|
|
if (pp->Wpn[WPN_STAR] && (pp->Wpn[WPN_STAR]->flags & PANF_DEATH_HIDE))
|
|
{
|
|
pp->Wpn[WPN_STAR]->flags &= ~(PANF_DEATH_HIDE);
|
|
pp->Flags &= ~(PF_WEAPON_RETRACT);
|
|
pSetState(pp->CurWpn, pp->CurWpn->PresentState);
|
|
return;
|
|
}
|
|
|
|
if (!pp->Wpn[WPN_STAR])
|
|
{
|
|
psp = pp->Wpn[WPN_STAR] = pSpawnSprite(pp, ps_PresentStar, PRI_MID, STAR_XOFF, STAR_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_STAR])
|
|
{
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_STAR);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
pp->CurWpn = pp->Wpn[WPN_STAR];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_ThrowStar;
|
|
//psp->ActionState = &ps_ThrowStar[1];
|
|
psp->RetractState = ps_RetractStar;
|
|
psp->PresentState = ps_PresentStar;
|
|
psp->RestState = ps_StarRest;
|
|
//psp->RestState = ps_ThrowStar;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
PlaySound(DIGI_PULL, pp, v3df_follow|v3df_dontpan);
|
|
if (StdRandomRange(1000) > 900 && pp == Player+myconnectindex)
|
|
{
|
|
if (!sw_darts)
|
|
PlayerSound(DIGI_ILIKESHURIKEN, v3df_follow|v3df_dontpan,pp);
|
|
}
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
void pStarPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < STAR_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = STAR_YOFF;
|
|
}
|
|
|
|
if (psp->pos.Y <= STAR_YOFF)
|
|
{
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pStarBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 10;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pLStarBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 6;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 16;
|
|
}
|
|
|
|
void pStarHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= STAR_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = STAR_YOFF + tileHeight(picnum);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pStarRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_StarHide))
|
|
return;
|
|
|
|
pStarBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
|
|
pSetState(psp, psp->ActionState);
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
}
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
|
|
void pStarAction(PANEL_SPRITE* psp)
|
|
{
|
|
pStarBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pStarThrow(PANEL_SPRITE* psp)
|
|
{
|
|
InitStar(psp->PlayerP);
|
|
}
|
|
|
|
void pStarRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= STAR_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
|
|
// kill only in its own routine
|
|
psp->PlayerP->Wpn[WPN_STAR] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// UZI
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pUziPresent(PANEL_SPRITE*);
|
|
void pUziFire(PANEL_SPRITE*);
|
|
void pUziRetract(PANEL_SPRITE*);
|
|
void pUziAction(PANEL_SPRITE*);
|
|
void pUziRest(PANEL_SPRITE*);
|
|
void pUziHide(PANEL_SPRITE*);
|
|
void pUziPresentReload(PANEL_SPRITE*);
|
|
|
|
void pSpawnUziClip(PANEL_SPRITE*);
|
|
void pSpawnUziReload(PANEL_SPRITE*);
|
|
void pUziReload(PANEL_SPRITE*);
|
|
void pUziReloadRetract(PANEL_SPRITE*);
|
|
void pUziClip(PANEL_SPRITE*);
|
|
void pUziDoneReload(PANEL_SPRITE*);
|
|
|
|
void pUziEjectDown(PANEL_SPRITE*);
|
|
void pUziEjectUp(PANEL_SPRITE*);
|
|
|
|
// CTW MODIFICATION
|
|
//void SetVisNorm(void);
|
|
int SetVisNorm(void);
|
|
// CTW MODIFICATION END
|
|
pANIMATOR pSetVisNorm;
|
|
|
|
// Right Uzi
|
|
PANEL_STATE ps_FireUzi[] =
|
|
{
|
|
{ID_UziPresent0, 3, pUziRest, &ps_FireUzi[0], 0,0,0},
|
|
{ID_UziFire0, 1, pUziAction, &ps_FireUzi[2], psf_ShadeHalf, 0,0},
|
|
|
|
{ID_UziFire1, 0, pUziFire, &ps_FireUzi[3], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
{ID_UziFire1, 4, pUziAction, &ps_FireUzi[4], psf_ShadeNone, 0,0},
|
|
{ID_UziFire1, 0, pSetVisNorm, &ps_FireUzi[5], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
{ID_UziFire1, 4, pUziAction, &ps_FireUzi[6], psf_ShadeNone, 0,0},
|
|
{ID_UziFire1, 0, pUziFire, &ps_FireUzi[7], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
{ID_UziFire1, 4, pUziAction, &ps_FireUzi[8], psf_ShadeNone, 0,0},
|
|
{ID_UziFire1, 0, pSetVisNorm, &ps_FireUzi[9], psf_ShadeNone, 0,0},
|
|
|
|
{ID_UziFire0, 4, pUziAction, &ps_FireUzi[10], psf_ShadeHalf, 0,0},
|
|
{ID_UziFire0, 0, pUziFire, &ps_FireUzi[11], psf_QuickCall, 0,0},
|
|
{ID_UziFire0, 4, pUziAction, &ps_FireUzi[12], psf_ShadeHalf, 0,0},
|
|
{ID_UziFire0, 4, pUziAction, &ps_FireUzi[13], psf_ShadeHalf, 0,0},
|
|
|
|
{ID_UziFire1, 5, pUziRest, &ps_FireUzi[0], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
};
|
|
|
|
#define PRESENT_UZI_RATE 6
|
|
#define RELOAD_UZI_RATE 1
|
|
|
|
PANEL_STATE ps_UziNull[] =
|
|
{
|
|
{ID_UziPresent0, PRESENT_UZI_RATE, pNullAnimator, &ps_UziNull[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_UziHide[] =
|
|
{
|
|
{ID_UziPresent0, PRESENT_UZI_RATE, pUziHide, &ps_UziHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_PresentUzi[] =
|
|
{
|
|
{ID_UziPresent0, PRESENT_UZI_RATE, pUziPresent, &ps_PresentUzi[0], 0,0,0},
|
|
};
|
|
|
|
// present of secondary uzi for reload needs to be faster
|
|
PANEL_STATE ps_PresentUziReload[] =
|
|
{
|
|
{ID_UziPresent0, RELOAD_UZI_RATE, pUziPresentReload, &ps_PresentUziReload[0], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RetractUzi[] =
|
|
{
|
|
{ID_UziPresent0, PRESENT_UZI_RATE, pUziRetract, &ps_RetractUzi[0], 0,0,0},
|
|
};
|
|
|
|
// Left Uzi
|
|
|
|
PANEL_STATE ps_FireUzi2[] =
|
|
{
|
|
{ID_Uzi2Present0, 3, pUziRest, &ps_FireUzi2[0], psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire0, 1, pUziAction, &ps_FireUzi2[2], psf_ShadeHalf|psf_Xflip, 0,0},
|
|
|
|
{ID_Uzi2Fire1, 0, pUziFire, &ps_FireUzi2[3], psf_ShadeNone|psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire1, 4, pUziAction, &ps_FireUzi2[4], psf_ShadeNone|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire1, 4, pUziAction, &ps_FireUzi2[5], psf_ShadeNone|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire1, 0, pUziFire, &ps_FireUzi2[6], psf_ShadeNone|psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire1, 4, pUziAction, &ps_FireUzi2[7], psf_ShadeNone|psf_Xflip, 0,0},
|
|
|
|
{ID_Uzi2Fire0, 4, pUziAction, &ps_FireUzi2[8], psf_ShadeHalf|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire0, 0, pUziFire, &ps_FireUzi2[9], psf_ShadeHalf|psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire0, 4, pUziAction, &ps_FireUzi2[10], psf_ShadeHalf|psf_Xflip, 0,0},
|
|
{ID_Uzi2Fire0, 4, pUziAction, &ps_FireUzi2[11], psf_ShadeHalf|psf_Xflip, 0,0},
|
|
|
|
{ID_Uzi2Fire1, 5, pUziRest, &ps_FireUzi2[0], psf_QuickCall, 0,0},
|
|
};
|
|
|
|
|
|
PANEL_STATE ps_PresentUzi2[] =
|
|
{
|
|
{ID_Uzi2Present0, PRESENT_UZI_RATE, pUziPresent, &ps_PresentUzi2[0], psf_Xflip, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_Uzi2Hide[] =
|
|
{
|
|
{ID_Uzi2Present0, PRESENT_UZI_RATE, pUziHide, &ps_Uzi2Hide[0], psf_Xflip, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RetractUzi2[] =
|
|
{
|
|
{ID_Uzi2Present0, PRESENT_UZI_RATE, pUziRetract, &ps_RetractUzi2[0], psf_Xflip, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_Uzi2Suicide[] =
|
|
{
|
|
{ID_Uzi2Present0, PRESENT_UZI_RATE, pSuicide, &ps_Uzi2Suicide[0], psf_Xflip, 0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_Uzi2Null[] =
|
|
{
|
|
{ID_Uzi2Present0, PRESENT_UZI_RATE, pNullAnimator, &ps_Uzi2Null[0], psf_Xflip, 0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_UziEject[] =
|
|
{
|
|
{ID_UziPresent0, 1, pNullAnimator, &ps_UziEject[1], 0,0,0},
|
|
{ID_UziPresent0, RELOAD_UZI_RATE, pUziEjectDown, &ps_UziEject[1], 0,0,0},
|
|
{ID_UziEject0, RELOAD_UZI_RATE, pUziEjectUp, &ps_UziEject[2], 0,0,0},
|
|
{ID_UziEject0, 1, pNullAnimator, &ps_UziEject[4], 0,0,0},
|
|
{ID_UziEject0, RELOAD_UZI_RATE, pSpawnUziClip, &ps_UziEject[5], psf_QuickCall, 0,0},
|
|
{ID_UziEject0, RELOAD_UZI_RATE, pNullAnimator, &ps_UziEject[5], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_UziClip[] =
|
|
{
|
|
{ID_UziClip0, RELOAD_UZI_RATE, pUziClip, &ps_UziClip[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_UziReload[] =
|
|
{
|
|
{ID_UziReload0, RELOAD_UZI_RATE, pUziReload, &ps_UziReload[0], 0,0,0},
|
|
{ID_UziReload0, RELOAD_UZI_RATE, pUziReloadRetract, &ps_UziReload[1], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_UziDoneReload[] =
|
|
{
|
|
{ID_UziEject0, RELOAD_UZI_RATE, pUziDoneReload, &ps_UziDoneReload[0], 0,0,0}
|
|
};
|
|
|
|
#define CHAMBER_REST 0
|
|
#define CHAMBER_FIRE 1
|
|
#define CHAMBER_RELOAD 2
|
|
void pUziOverlays(PANEL_SPRITE* psp, short mode)
|
|
{
|
|
#define UZI_CHAMBER_XOFF 32
|
|
#define UZI_CHAMBER_YOFF -73
|
|
|
|
#define UZI_CHAMBERRELOAD_XOFF 14
|
|
#define UZI_CHAMBERRELOAD_YOFF -100
|
|
|
|
if (!(psp->flags & PANF_SECONDARY)) return;
|
|
|
|
if (psp->over[0].xoff == -1)
|
|
{
|
|
psp->over[0].xoff = UZI_CHAMBER_XOFF;
|
|
psp->over[0].yoff = UZI_CHAMBER_YOFF;
|
|
}
|
|
|
|
switch (mode)
|
|
{
|
|
case 0: // At rest
|
|
psp->over[0].pic = UZI_COPEN;
|
|
break;
|
|
case 1: // Firing
|
|
psp->over[0].pic = UZI_CLIT;
|
|
break;
|
|
case 2: // Reloading
|
|
psp->over[0].pic = UZI_CRELOAD;
|
|
psp->over[0].xoff = UZI_CHAMBERRELOAD_XOFF;
|
|
psp->over[0].yoff = UZI_CHAMBERRELOAD_YOFF;
|
|
break;
|
|
}
|
|
}
|
|
|
|
#define UZI_CLIP_XOFF 16
|
|
#define UZI_CLIP_YOFF (-84)
|
|
|
|
//#define UZI_XOFF (80)
|
|
#define UZI_XOFF (100)
|
|
#define UZI_YOFF 208
|
|
|
|
#define UZI_RELOAD_YOFF 200
|
|
|
|
|
|
//
|
|
// Uzi Reload
|
|
//
|
|
|
|
void pUziEjectDown(PANEL_SPRITE* gun)
|
|
{
|
|
gun->opos.Y = gun->pos.Y;
|
|
gun->pos.Y += 5 * synctics;
|
|
|
|
if (gun->pos.Y > 260)
|
|
{
|
|
gun->opos.Y = gun->pos.Y = 260;
|
|
pStatePlusOne(gun);
|
|
}
|
|
}
|
|
|
|
void pUziEjectUp(PANEL_SPRITE* gun)
|
|
{
|
|
|
|
pUziOverlays(gun, CHAMBER_RELOAD);
|
|
|
|
gun->opos.Y = gun->pos.Y;
|
|
gun->pos.Y -= 5 * synctics;
|
|
|
|
if (gun->pos.Y < UZI_RELOAD_YOFF)
|
|
{
|
|
gun->opos.Y = gun->pos.Y = UZI_RELOAD_YOFF;
|
|
pStatePlusOne(gun);
|
|
}
|
|
}
|
|
|
|
|
|
void pSpawnUziClip(PANEL_SPRITE* gun)
|
|
{
|
|
PANEL_SPRITE* New;
|
|
|
|
PlaySound(DIGI_REMOVECLIP, gun->PlayerP,v3df_follow|v3df_dontpan|v3df_doppler|v3df_follow);
|
|
|
|
if ((gun->flags & PANF_XFLIP))
|
|
{
|
|
New = pSpawnSprite(gun->PlayerP, ps_UziClip, PRI_BACK, gun->pos.X - UZI_CLIP_XOFF, gun->pos.Y + UZI_CLIP_YOFF);
|
|
New->flags |= (PANF_XFLIP);
|
|
New->ang = NORM_ANGLE(1024 + 256 + 22);
|
|
New->ang = NORM_ANGLE(New->ang + 512);
|
|
}
|
|
else
|
|
{
|
|
New = pSpawnSprite(gun->PlayerP, ps_UziClip, PRI_BACK, gun->pos.X + UZI_CLIP_XOFF, gun->pos.Y + UZI_CLIP_YOFF);
|
|
New->ang = NORM_ANGLE(1024 + 256 - 22);
|
|
}
|
|
|
|
New->vel = 1050;
|
|
New->flags |= (PANF_WEAPON_SPRITE);
|
|
|
|
|
|
// carry Eject sprite with clip
|
|
New->sibling = gun;
|
|
}
|
|
|
|
void pSpawnUziReload(PANEL_SPRITE* oclip)
|
|
{
|
|
PANEL_SPRITE* nclip;
|
|
|
|
nclip = pSpawnSprite(oclip->PlayerP, ps_UziReload, PRI_BACK, oclip->pos.X, UZI_RELOAD_YOFF);
|
|
nclip->flags |= PANF_WEAPON_SPRITE;
|
|
|
|
if ((oclip->flags & PANF_XFLIP))
|
|
nclip->flags |= PANF_XFLIP;
|
|
|
|
// move Reload in oposite direction of clip
|
|
nclip->ang = NORM_ANGLE(oclip->ang + 1024);
|
|
nclip->vel = 900;
|
|
|
|
// move gun sprite from clip to reload
|
|
nclip->sibling = oclip->sibling;
|
|
}
|
|
|
|
void pUziReload(PANEL_SPRITE* nclip)
|
|
{
|
|
PANEL_SPRITE* gun = nclip->sibling;
|
|
|
|
nclip->vel += 14 * synctics;
|
|
|
|
nclip->opos.X = nclip->pos.X;
|
|
nclip->opos.Y = nclip->pos.Y;
|
|
|
|
nclip->pos.X += pspCosVel(nclip);
|
|
nclip->pos.Y -= pspSinVel(nclip);
|
|
|
|
gun->opos.X = gun->pos.X;
|
|
gun->opos.Y = gun->pos.Y;
|
|
|
|
gun->pos.X -= pspCosVel(gun);
|
|
gun->pos.Y += pspSinVel(gun);
|
|
|
|
if ((nclip->flags & PANF_XFLIP))
|
|
{
|
|
if (nclip->pos.X < gun->pos.X)
|
|
{
|
|
PlaySound(DIGI_REPLACECLIP, nclip->PlayerP,v3df_follow|v3df_dontpan|v3df_doppler);
|
|
|
|
nclip->opos.X = nclip->pos.X = gun->pos.X - UZI_CLIP_XOFF;
|
|
nclip->opos.Y = nclip->pos.Y = gun->pos.Y + UZI_CLIP_YOFF;
|
|
nclip->vel = 680;
|
|
nclip->ang = NORM_ANGLE(nclip->ang - 128 - 64);
|
|
// go to retract phase
|
|
pSetState(nclip, &ps_UziReload[1]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (nclip->pos.X > gun->pos.X)
|
|
{
|
|
PlaySound(DIGI_REPLACECLIP, nclip->PlayerP,v3df_follow|v3df_dontpan|v3df_doppler);
|
|
|
|
nclip->opos.X = nclip->pos.X = gun->pos.X + UZI_CLIP_XOFF;
|
|
nclip->opos.Y = nclip->pos.Y = gun->pos.Y + UZI_CLIP_YOFF;
|
|
nclip->vel = 680;
|
|
nclip->ang = NORM_ANGLE(nclip->ang + 128 + 64);
|
|
// go to retract phase
|
|
pSetState(nclip, &ps_UziReload[1]);
|
|
}
|
|
}
|
|
}
|
|
|
|
void pUziReloadRetract(PANEL_SPRITE* nclip)
|
|
{
|
|
PANEL_SPRITE* gun = nclip->sibling;
|
|
|
|
double xadj = pspCosVel(nclip);
|
|
double yadj = pspSinVel(nclip);
|
|
|
|
nclip->vel += 18 * synctics;
|
|
|
|
nclip->backupcoords();
|
|
|
|
nclip->pos.X -= xadj;
|
|
nclip->pos.Y += yadj;
|
|
|
|
gun->backupcoords();
|
|
|
|
gun->pos.X -= xadj;
|
|
gun->pos.Y += yadj;
|
|
|
|
if (gun->pos.Y > UZI_RELOAD_YOFF + tileHeight(gun->picndx))
|
|
{
|
|
pSetState(gun, ps_UziDoneReload);
|
|
pKillSprite(nclip);
|
|
}
|
|
}
|
|
|
|
void pUziDoneReload(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
|
|
if (psp->flags & (PANF_PRIMARY) && pp->WpnUziType == 3)
|
|
{
|
|
// if 2 uzi's and the first one has been reloaded
|
|
// kill the first one and make the second one the CurWeapon
|
|
// Set uzi's back to previous state
|
|
PANEL_SPRITE* New;
|
|
|
|
|
|
if (pp->WpnUziType > 2)
|
|
pp->WpnUziType -= 3;
|
|
|
|
New = InitWeaponUziSecondaryReload(psp);
|
|
pp->Wpn[WPN_UZI] = New;
|
|
pp->CurWpn = New;
|
|
pp->CurWpn->sibling = nullptr;
|
|
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// Reset everything
|
|
|
|
// Set uzi's back to previous state
|
|
if (pp->WpnUziType > 2)
|
|
pp->WpnUziType -= 3;
|
|
|
|
// reset uzi variable
|
|
pp->Wpn[WPN_UZI] = nullptr;
|
|
pp->CurWpn = nullptr;
|
|
|
|
// kill uzi eject sequence for good
|
|
pKillSprite(psp);
|
|
|
|
// give the uzi back
|
|
InitWeaponUzi(pp);
|
|
}
|
|
}
|
|
|
|
void pUziClip(PANEL_SPRITE* oclip)
|
|
{
|
|
|
|
oclip->opos.X = oclip->pos.X;
|
|
oclip->opos.Y = oclip->pos.Y;
|
|
|
|
oclip->pos.X += pspCosVel(oclip);
|
|
oclip->pos.Y -= pspSinVel(oclip);
|
|
|
|
if (oclip->pos.Y > UZI_RELOAD_YOFF)
|
|
{
|
|
PANEL_SPRITE* gun = oclip->sibling;
|
|
|
|
// as synctics gets bigger, oclip->x can be way off
|
|
// when clip goes off the screen - recalc oclip->x from scratch
|
|
// so it will end up the same for all synctic values
|
|
for (oclip->pos.X = oclip->opos.X, oclip->pos.Y = oclip->opos.Y; oclip->pos.Y < UZI_RELOAD_YOFF; )
|
|
{
|
|
oclip->pos.X += oclip->vel * bcosf(oclip->ang, -22);
|
|
oclip->pos.Y -= oclip->vel * bsinf(oclip->ang, -22);
|
|
}
|
|
|
|
oclip->opos.X = oclip->pos.X;
|
|
oclip->opos.Y = oclip->pos.Y = UZI_RELOAD_YOFF;
|
|
|
|
gun->vel = 800;
|
|
gun->ang = NORM_ANGLE(oclip->ang + 1024);
|
|
|
|
pSpawnUziReload(oclip);
|
|
pKillSprite(oclip);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Uzi Basic Stuff
|
|
//
|
|
|
|
void InitWeaponUzi(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp = nullptr;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
pp->WeaponType = WPN_UZI;
|
|
|
|
// make sure you have the uzi, uzi ammo, and not retracting another
|
|
// weapon
|
|
if (!(pp->WpnFlags &BIT(WPN_UZI)) ||
|
|
// pp->WpnAmmo[WPN_UZI] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
// if players uzi is null
|
|
if (!pp->Wpn[WPN_UZI])
|
|
{
|
|
psp = pp->Wpn[WPN_UZI] = pSpawnSprite(pp, ps_PresentUzi, PRI_MID, 160 + UZI_XOFF, UZI_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
// if Current weapon is uzi
|
|
if (pp->CurWpn == pp->Wpn[WPN_UZI])
|
|
{
|
|
// Retracting other uzi?
|
|
if (pp->CurWpn->sibling && pp->WpnUziType == 1)
|
|
{
|
|
RetractCurWpn(pp);
|
|
}
|
|
else
|
|
// Is player toggling between one and two uzi's?
|
|
if (pp->CurWpn->sibling && (pp->Wpn[WPN_UZI]->flags & PANF_PRIMARY) && pp->WpnUziType == 0)
|
|
{
|
|
if (!(pp->CurWpn->flags & PANF_RELOAD))
|
|
InitWeaponUzi2(pp->Wpn[WPN_UZI]);
|
|
}
|
|
|
|
// if actually picked an uzi up and don't currently have double uzi
|
|
if (pp->Flags & (PF_PICKED_UP_AN_UZI) && !(pp->Wpn[WPN_UZI]->flags & PANF_PRIMARY))
|
|
{
|
|
pp->Flags &= ~(PF_PICKED_UP_AN_UZI);
|
|
|
|
if (!(pp->CurWpn->flags & PANF_RELOAD))
|
|
InitWeaponUzi2(pp->Wpn[WPN_UZI]);
|
|
}
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
pp->Flags &= ~(PF_PICKED_UP_AN_UZI);
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_UZI);
|
|
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
pp->CurWpn = pp->Wpn[WPN_UZI];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = &ps_FireUzi[1];
|
|
psp->RetractState = ps_RetractUzi;
|
|
psp->PresentState = ps_PresentUzi;
|
|
psp->RestState = ps_FireUzi;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
// power up
|
|
// NOTE: PRIMARY is ONLY set when there is a powerup
|
|
if (pp->Flags & (PF_TWO_UZI))
|
|
{
|
|
InitWeaponUzi2(psp);
|
|
}
|
|
|
|
PlaySound(DIGI_UZI_UP, pp, v3df_follow);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
PANEL_SPRITE* InitWeaponUzi2(PANEL_SPRITE* uzi_orig)
|
|
{
|
|
PANEL_SPRITE* New;
|
|
PLAYER* pp = uzi_orig->PlayerP;
|
|
|
|
|
|
// There is already a second uzi, or it's retracting
|
|
if (pp->WpnUziType == 1 || pp->CurWpn->sibling || (pp->Flags & PF_WEAPON_RETRACT)) return nullptr;
|
|
|
|
// NOTE: PRIMARY is ONLY set when there is a powerup
|
|
uzi_orig->flags |= PANF_PRIMARY;
|
|
|
|
// Spawning a 2nd uzi, set weapon mode
|
|
pp->WpnUziType = 0; // 0 is up, 1 is retract
|
|
|
|
New = pSpawnSprite(pp, ps_PresentUzi2, PRI_MID, 160 - UZI_XOFF, UZI_YOFF);
|
|
New->pos.Y += tileHeight(New->picndx);
|
|
New->opos.Y = New->pos.Y;
|
|
uzi_orig->sibling = New;
|
|
|
|
// Set up the New weapon variables
|
|
New->flags |= (PANF_WEAPON_SPRITE);
|
|
New->ActionState = &ps_FireUzi2[1];
|
|
New->RetractState = ps_RetractUzi2;
|
|
New->PresentState = ps_PresentUzi2;
|
|
New->RestState = ps_FireUzi2;
|
|
pSetState(New, New->PresentState);
|
|
|
|
New->sibling = uzi_orig;
|
|
New->flags |= (PANF_SECONDARY);
|
|
pUziOverlays(New, CHAMBER_REST);
|
|
|
|
return New;
|
|
}
|
|
|
|
PANEL_SPRITE* InitWeaponUziSecondaryReload(PANEL_SPRITE* uzi_orig)
|
|
{
|
|
PANEL_SPRITE* New;
|
|
PLAYER* pp = uzi_orig->PlayerP;
|
|
|
|
New = pSpawnSprite(pp, ps_PresentUzi, PRI_MID, 160 - UZI_XOFF, UZI_YOFF);
|
|
New->pos.Y += tileHeight(New->picndx);
|
|
New->opos.Y = New->pos.Y;
|
|
|
|
New->flags |= (PANF_XFLIP);
|
|
|
|
// Set up the New weapon variables
|
|
New->flags |= (PANF_WEAPON_SPRITE);
|
|
New->ActionState = ps_UziEject;
|
|
New->RetractState = ps_RetractUzi;
|
|
New->PresentState = ps_PresentUzi;
|
|
New->RestState = ps_UziEject;
|
|
// pSetState(New, New->PresentState);
|
|
pSetState(New, ps_PresentUziReload);
|
|
|
|
New->sibling = uzi_orig;
|
|
New->flags |= (PANF_SECONDARY|PANF_RELOAD);
|
|
|
|
return New;
|
|
}
|
|
|
|
void pUziPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->PlayerP->WpnUziType)
|
|
{
|
|
psp->PlayerP->WpnReloadState = 2;
|
|
}
|
|
|
|
if (psp->pos.Y < UZI_YOFF)
|
|
{
|
|
psp->flags &= ~(PANF_RELOAD);
|
|
|
|
psp->opos.Y = psp->pos.Y = UZI_YOFF;
|
|
psp->backupx();
|
|
psp->backupbobcoords();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
// same as pUziPresent only faster for reload sequence
|
|
void pUziPresentReload(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 5 * synctics;
|
|
|
|
psp->PlayerP->WpnReloadState = 2;
|
|
|
|
if (psp->pos.Y < UZI_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = UZI_YOFF;
|
|
psp->backupx();
|
|
psp->backupbobcoords();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pUziBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 12;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pUziStartReload(PANEL_SPRITE* psp)
|
|
{
|
|
SetVisNorm();
|
|
|
|
psp->backupx();
|
|
|
|
// Set uzi's to reload state
|
|
if (psp->PlayerP->WpnUziType < 3)
|
|
psp->PlayerP->WpnUziType += 3;
|
|
|
|
// Uzi #1 reload - starting from a full up position
|
|
pSetState(psp, ps_UziEject);
|
|
|
|
psp->flags |= (PANF_RELOAD);
|
|
|
|
if (psp->flags & (PANF_PRIMARY) && psp->sibling)
|
|
{
|
|
// this is going to KILL Uzi #2 !!!
|
|
pSetState(psp->sibling, psp->sibling->RetractState);
|
|
}
|
|
}
|
|
|
|
void pUziHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= 200 + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = 200 + tileHeight(picnum);
|
|
|
|
if (psp->flags & (PANF_PRIMARY) && psp->PlayerP->WpnUziType != 1)
|
|
{
|
|
if (pWeaponUnHideKeys(psp, psp->PresentState))
|
|
pSetState(psp->sibling, psp->sibling->PresentState);
|
|
}
|
|
else if (!(psp->flags & PANF_SECONDARY))
|
|
{
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
}
|
|
|
|
void pUziRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool shooting;
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
|
|
// If you have two uzi's, but one didn't come up, spawn it
|
|
if (psp->PlayerP->Flags & (PF_TWO_UZI) && psp->sibling == nullptr)
|
|
{
|
|
InitWeaponUzi2(psp);
|
|
}
|
|
|
|
if (psp->flags & (PANF_PRIMARY) && psp->sibling)
|
|
{
|
|
if (pWeaponHideKeys(psp, ps_UziHide))
|
|
{
|
|
if (psp->sibling != nullptr) // !JIM! Without this line, will ASSERT if reloading here
|
|
pSetState(psp->sibling, ps_Uzi2Hide);
|
|
return;
|
|
}
|
|
}
|
|
else if (!(psp->flags & PANF_SECONDARY))
|
|
{
|
|
if (pWeaponHideKeys(psp, ps_UziHide))
|
|
return;
|
|
}
|
|
|
|
if (psp->flags & (PANF_SECONDARY))
|
|
pUziOverlays(psp, CHAMBER_REST);
|
|
|
|
SetVisNorm();
|
|
|
|
shooting = (psp->PlayerP->input.actions & SB_FIRE) && (psp->PlayerP->KeyPressBits & SB_FIRE);
|
|
shooting |= force;
|
|
|
|
pUziBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP) || shooting);
|
|
|
|
if (shooting)
|
|
{
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
pSetState(psp, psp->ActionState);
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pUziAction(PANEL_SPRITE* psp)
|
|
{
|
|
static int alternate = 0;
|
|
|
|
bool shooting = (psp->PlayerP->input.actions & SB_FIRE) && (psp->PlayerP->KeyPressBits & SB_FIRE);
|
|
|
|
if (shooting)
|
|
{
|
|
if (psp->flags & (PANF_SECONDARY))
|
|
{
|
|
alternate++;
|
|
if (alternate > 6) alternate = 0;
|
|
if (alternate <= 3)
|
|
pUziOverlays(psp, CHAMBER_FIRE);
|
|
else
|
|
pUziOverlays(psp, CHAMBER_REST);
|
|
}
|
|
// Only Recoil if shooting
|
|
pUziBobSetup(psp);
|
|
UziRecoilYadj = (RANDOM_P2(1024)) >> 8; // global hack for
|
|
// weapon Bob
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP) || shooting);
|
|
UziRecoilYadj = 0; // reset my global hack
|
|
if (RANDOM_P2(1024) > 990)
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
}
|
|
else
|
|
{
|
|
if (psp->flags & (PANF_SECONDARY))
|
|
pUziOverlays(psp, CHAMBER_REST);
|
|
pUziBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP) || shooting);
|
|
}
|
|
}
|
|
|
|
void pUziFire(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
if (psp->flags & (PANF_SECONDARY) && pp->WpnUziType > 0) return;
|
|
|
|
InitUzi(psp->PlayerP);
|
|
SpawnUziShell(psp);
|
|
|
|
// If its the second Uzi, give the shell back only if it's a reload count to keep #'s even
|
|
if (psp->flags & (PANF_SECONDARY))
|
|
{
|
|
if (pp->Flags & (PF_TWO_UZI) && psp->sibling)
|
|
{
|
|
if ((pp->WpnAmmo[WPN_UZI] % 100) == 0)
|
|
pp->WpnAmmo[WPN_UZI]++;
|
|
}
|
|
else if ((pp->WpnAmmo[WPN_UZI] % 50) == 0)
|
|
pp->WpnAmmo[WPN_UZI]++;
|
|
}
|
|
else
|
|
{
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 32);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
// Reload if done with clip
|
|
if (pp->Flags & (PF_TWO_UZI) && psp->sibling)
|
|
{
|
|
if ((pp->WpnAmmo[WPN_UZI] % 100) == 0)
|
|
pUziStartReload(psp);
|
|
}
|
|
else if ((pp->WpnAmmo[WPN_UZI] % 50) == 0)
|
|
{
|
|
// clip has run out
|
|
pUziStartReload(psp);
|
|
}
|
|
}
|
|
}
|
|
|
|
void pUziRetract(PANEL_SPRITE* psp)
|
|
{
|
|
// PANEL_SPRITE* sib = psp->sibling;
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= 200 + tileHeight(picnum))
|
|
{
|
|
// if in the reload phase and its retracting then get rid of uzi
|
|
// no matter whether it is PRIMARY/SECONDARY/neither.
|
|
if (psp->flags & (PANF_RELOAD))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_UZI] = nullptr;
|
|
}
|
|
else
|
|
{
|
|
// NOT reloading here
|
|
if (psp->flags & (PANF_PRIMARY))
|
|
{
|
|
// only reset when primary goes off the screen
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_UZI] = nullptr;
|
|
}
|
|
else if (psp->flags & (PANF_SECONDARY))
|
|
{
|
|
// primarily for beginning of reload sequence where seconary
|
|
// is taken off of the screen. Lets the primary know that
|
|
// he is alone.
|
|
if (psp->sibling && psp->sibling->sibling == psp)
|
|
psp->sibling->sibling = nullptr;
|
|
}
|
|
else
|
|
{
|
|
// only one uzi here is retracting
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_UZI] = nullptr;
|
|
}
|
|
}
|
|
|
|
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// UZI SHELL
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pUziShell(PANEL_SPRITE*);
|
|
|
|
#define UZI_SHELL_RATE 10
|
|
|
|
PANEL_STATE ps_UziShell[] =
|
|
{
|
|
{ID_UziShell0, UZI_SHELL_RATE, pUziShell, &ps_UziShell[1], 0,0,0},
|
|
{ID_UziShell1, UZI_SHELL_RATE, pUziShell, &ps_UziShell[2], 0,0,0},
|
|
{ID_UziShell2, UZI_SHELL_RATE, pUziShell, &ps_UziShell[3], 0,0,0},
|
|
{ID_UziShell3, UZI_SHELL_RATE, pUziShell, &ps_UziShell[4], 0,0,0},
|
|
{ID_UziShell4, UZI_SHELL_RATE, pUziShell, &ps_UziShell[5], 0,0,0},
|
|
{ID_UziShell5, UZI_SHELL_RATE, pUziShell, &ps_UziShell[0], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_Uzi2Shell[] =
|
|
{
|
|
{ID_Uzi2Shell0, UZI_SHELL_RATE, pUziShell, &ps_Uzi2Shell[1], psf_Xflip, 0,0},
|
|
{ID_Uzi2Shell1, UZI_SHELL_RATE, pUziShell, &ps_Uzi2Shell[2], psf_Xflip, 0,0},
|
|
{ID_Uzi2Shell2, UZI_SHELL_RATE, pUziShell, &ps_Uzi2Shell[3], psf_Xflip, 0,0},
|
|
{ID_Uzi2Shell3, UZI_SHELL_RATE, pUziShell, &ps_Uzi2Shell[4], psf_Xflip, 0,0},
|
|
{ID_Uzi2Shell4, UZI_SHELL_RATE, pUziShell, &ps_Uzi2Shell[5], psf_Xflip, 0,0},
|
|
{ID_Uzi2Shell5, UZI_SHELL_RATE, pUziShell, &ps_Uzi2Shell[0], psf_Xflip, 0,0},
|
|
};
|
|
|
|
void SpawnUziShell(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
if (psp->State && (psp->State->flags & psf_Xflip))
|
|
{
|
|
// LEFT side
|
|
pp->UziShellLeftAlt = !pp->UziShellLeftAlt;
|
|
if (pp->UziShellLeftAlt)
|
|
SpawnShell(pp->actor,-3);
|
|
}
|
|
else
|
|
{
|
|
// RIGHT side
|
|
pp->UziShellRightAlt = !pp->UziShellRightAlt;
|
|
if (pp->UziShellRightAlt)
|
|
SpawnShell(pp->actor,-2);
|
|
}
|
|
}
|
|
|
|
void pUziShell(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupx();
|
|
|
|
if (psp->State && (psp->State->flags & psf_Xflip))
|
|
{
|
|
psp->pos.X -= 3 * synctics;
|
|
}
|
|
else
|
|
{
|
|
psp->pos.X += 3 * synctics;
|
|
}
|
|
|
|
// increment the ndx into the sin table and wrap at 1024
|
|
psp->sin_ndx = (psp->sin_ndx + (synctics << psp->sin_arc_speed) + 1024) & 1023;
|
|
|
|
// get height
|
|
psp->opos.Y = psp->pos.Y = psp->bobpos.Y;
|
|
psp->pos.Y += psp->sin_amt * -bsinf(psp->sin_ndx, -14);
|
|
|
|
// if off of the screen kill them
|
|
if (psp->pos.X > 330 || psp->pos.X < -10)
|
|
{
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SHOTGUN SHELL
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pShotgunShell(PANEL_SPRITE*);
|
|
|
|
#define SHOTGUN_SHELL_RATE 7
|
|
|
|
PANEL_STATE ps_ShotgunShell[] =
|
|
{
|
|
{ID_ShotgunShell0, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[1], 0,0,0},
|
|
{ID_ShotgunShell1, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[2], 0,0,0},
|
|
{ID_ShotgunShell2, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[3], 0,0,0},
|
|
{ID_ShotgunShell3, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[4], 0,0,0},
|
|
{ID_ShotgunShell4, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[5], 0,0,0},
|
|
{ID_ShotgunShell5, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[6], 0,0,0},
|
|
{ID_ShotgunShell6, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[7], 0,0,0},
|
|
{ID_ShotgunShell6, SHOTGUN_SHELL_RATE, pShotgunShell, &ps_ShotgunShell[0], 0,0,0},
|
|
};
|
|
|
|
void SpawnShotgunShell(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
SpawnShell(pp->actor,-4);
|
|
}
|
|
|
|
void pShotgunShell(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_JUMPING))
|
|
{
|
|
DoPanelJump(psp);
|
|
}
|
|
else if (psp->flags & (PANF_FALLING))
|
|
{
|
|
DoPanelFall(psp);
|
|
}
|
|
|
|
psp->backupx();
|
|
|
|
psp->pos.X += psp->xspeed * (1. / FRACUNIT);
|
|
|
|
if (psp->pos.X > 320 || psp->pos.X < 0 || psp->pos.Y > 200)
|
|
{
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SHOTGUN
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pShotgunPresent(PANEL_SPRITE* psp);
|
|
void pShotgunRetract(PANEL_SPRITE* psp);
|
|
void pShotgunAction(PANEL_SPRITE* psp);
|
|
void pShotgunRest(PANEL_SPRITE* psp);
|
|
void pShotgunFire(PANEL_SPRITE* psp);
|
|
void pShotgunHide(PANEL_SPRITE* psp);
|
|
void pShotgunRestTest(PANEL_SPRITE* psp);
|
|
|
|
void pShotgunReloadDown(PANEL_SPRITE* psp);
|
|
void pShotgunReloadUp(PANEL_SPRITE* psp);
|
|
void pShotgunBobSetup(PANEL_SPRITE* psp);
|
|
void pShotgunRecoilUp(PANEL_SPRITE* psp);
|
|
void pShotgunRecoilDown(PANEL_SPRITE* psp);
|
|
|
|
bool pShotgunReloadTest(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_ShotgunReload[];
|
|
|
|
#define Shotgun_BEAT_RATE 24
|
|
#define Shotgun_ACTION_RATE 4
|
|
|
|
PANEL_STATE ps_PresentShotgun[] =
|
|
{
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pShotgunPresent, &ps_PresentShotgun[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_ShotgunRest[] =
|
|
{
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pShotgunRest, &ps_ShotgunRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_ShotgunHide[] =
|
|
{
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pShotgunHide, &ps_ShotgunHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_ShotgunRecoil[] =
|
|
{
|
|
// recoil
|
|
{ID_ShotgunReload0, Shotgun_ACTION_RATE, pShotgunRecoilDown, &ps_ShotgunRecoil[0], 0,0,0},
|
|
{ID_ShotgunReload0, Shotgun_ACTION_RATE, pShotgunRecoilUp, &ps_ShotgunRecoil[1], 0,0,0},
|
|
// reload
|
|
{ID_ShotgunReload0, Shotgun_ACTION_RATE*5, pNullAnimator, &ps_ShotgunRecoil[3], 0,0,0},
|
|
{ID_ShotgunReload1, Shotgun_ACTION_RATE, pNullAnimator, &ps_ShotgunRecoil[4], 0,0,0},
|
|
{ID_ShotgunReload2, Shotgun_ACTION_RATE*5, pNullAnimator, &ps_ShotgunRecoil[5], 0,0,0},
|
|
{ID_ShotgunPresent0,Shotgun_ACTION_RATE, pShotgunRestTest, &ps_ShotgunRecoil[6], 0,0,0},
|
|
{ID_ShotgunPresent0,Shotgun_ACTION_RATE/2, pShotgunAction, &ps_ShotgunRecoil[7], 0,0,0},
|
|
{ID_ShotgunPresent0,Shotgun_ACTION_RATE/2, pShotgunAction, &ps_ShotgunRecoil[8], 0,0,0},
|
|
// ready to fire again
|
|
{ID_ShotgunPresent0, 3, pNullAnimator, &ps_ShotgunRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_ShotgunRecoilAuto[] =
|
|
{
|
|
// recoil
|
|
{ID_ShotgunReload0, 1, pShotgunRecoilDown, &ps_ShotgunRecoilAuto[0], 0,0,0},
|
|
{ID_ShotgunReload0, 1, pShotgunRecoilUp, &ps_ShotgunRecoilAuto[1], 0,0,0},
|
|
// Reload
|
|
{ID_ShotgunReload0, 1, pNullAnimator, &ps_ShotgunRecoilAuto[3], 0,0,0},
|
|
{ID_ShotgunReload0, 1, pNullAnimator, &ps_ShotgunRecoilAuto[4], 0,0,0},
|
|
{ID_ShotgunReload0, 1, pNullAnimator, &ps_ShotgunRecoilAuto[5], 0,0,0},
|
|
|
|
{ID_ShotgunPresent0,1, pShotgunRestTest, &ps_ShotgunRecoilAuto[6], 0,0,0},
|
|
{ID_ShotgunPresent0,1, pShotgunAction, &ps_ShotgunRecoilAuto[7], 0,0,0},
|
|
{ID_ShotgunPresent0,1, pShotgunRest, &ps_ShotgunRest[0],psf_QuickCall, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_ShotgunFire[] =
|
|
{
|
|
{ID_ShotgunFire0, Shotgun_ACTION_RATE, pShotgunAction, &ps_ShotgunFire[1], psf_ShadeHalf, 0,0},
|
|
{ID_ShotgunFire1, Shotgun_ACTION_RATE, pShotgunFire, &ps_ShotgunFire[2], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
{ID_ShotgunFire1, Shotgun_ACTION_RATE, pShotgunAction, &ps_ShotgunFire[3], psf_ShadeNone, 0,0},
|
|
{ID_ShotgunReload0, 0, SpawnShotgunShell, &ps_ShotgunFire[4], psf_QuickCall, 0,0},
|
|
{ID_ShotgunReload0, Shotgun_ACTION_RATE, pShotgunAction, &ps_ShotgunRecoil[0], 0,0,0}
|
|
};
|
|
|
|
|
|
#if 1
|
|
PANEL_STATE ps_ShotgunAutoFire[] =
|
|
{
|
|
{ID_ShotgunFire1, 2, pShotgunAction, &ps_ShotgunAutoFire[1], psf_ShadeHalf, 0,0},
|
|
{ID_ShotgunFire1, 2, pShotgunFire, &ps_ShotgunAutoFire[2], psf_ShadeNone, 0,0},
|
|
{ID_ShotgunFire1, 2, pShotgunAction, &ps_ShotgunAutoFire[3], psf_ShadeNone, 0,0},
|
|
{ID_ShotgunReload0, 0, SpawnShotgunShell, &ps_ShotgunAutoFire[4], psf_QuickCall, 0,0},
|
|
{ID_ShotgunReload0, 1, pShotgunAction, &ps_ShotgunRecoilAuto[0], 0,0,0}
|
|
};
|
|
#endif
|
|
|
|
#if 1
|
|
PANEL_STATE ps_ShotgunReload[] =
|
|
{
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pShotgunReloadDown, &ps_ShotgunReload[0], 0,0,0},
|
|
{ID_ShotgunPresent0, 30, pNullAnimator, &ps_ShotgunReload[2], 0,0,0},
|
|
// make reload sound here
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pNullAnimator, &ps_ShotgunReload[3], psf_QuickCall, 0,0},
|
|
{ID_ShotgunPresent0, 30, pNullAnimator, &ps_ShotgunReload[4], 0,0,0},
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pShotgunReloadUp, &ps_ShotgunReload[4], 0,0,0},
|
|
{ID_ShotgunPresent0, 3, pNullAnimator, &ps_ShotgunRest[0], 0,0,0}
|
|
};
|
|
#endif
|
|
|
|
PANEL_STATE ps_RetractShotgun[] =
|
|
{
|
|
{ID_ShotgunPresent0, Shotgun_BEAT_RATE, pShotgunRetract, &ps_RetractShotgun[0], 0,0,0}
|
|
};
|
|
|
|
#define SHOTGUN_YOFF 200
|
|
#define SHOTGUN_XOFF (160+42)
|
|
|
|
void InitWeaponShotgun(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp = nullptr;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
pp->WeaponType = WPN_SHOTGUN;
|
|
|
|
if (!(pp->WpnFlags &BIT(pp->WeaponType)) ||
|
|
// pp->WpnAmmo[pp->WeaponType] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[pp->WeaponType])
|
|
{
|
|
psp = pp->Wpn[pp->WeaponType] = pSpawnSprite(pp, ps_PresentShotgun, PRI_MID, SHOTGUN_XOFF, SHOTGUN_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[pp->WeaponType])
|
|
{
|
|
return;
|
|
}
|
|
|
|
psp->WeaponType = pp->WeaponType;
|
|
PlayerUpdateWeapon(pp, pp->WeaponType);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[pp->WeaponType];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_ShotgunFire;
|
|
//psp->ActionState = ps_ShotgunAutoFire;
|
|
psp->RetractState = ps_RetractShotgun;
|
|
psp->PresentState = ps_PresentShotgun;
|
|
psp->RestState = ps_ShotgunRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
PlaySound(DIGI_SHOTGUN_UP, pp, v3df_follow);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
void pShotgunSetRecoil(PANEL_SPRITE* psp)
|
|
{
|
|
psp->vel = 900;
|
|
psp->ang = NORM_ANGLE(-256);
|
|
}
|
|
|
|
void pShotgunRecoilDown(PANEL_SPRITE* psp)
|
|
{
|
|
int targetvel = psp->PlayerP->WpnShotgunType == 1 ? 890 : 780;
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel -= 24 * synctics;
|
|
|
|
if (psp->vel < targetvel)
|
|
{
|
|
psp->vel = targetvel;
|
|
psp->ang = NORM_ANGLE(psp->ang + 1024);
|
|
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pShotgunRecoilUp(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel += 15 * synctics;
|
|
|
|
if (psp->pos.Y < SHOTGUN_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = SHOTGUN_YOFF;
|
|
psp->opos.X = psp->pos.X = SHOTGUN_XOFF;
|
|
|
|
pShotgunSetRecoil(psp);
|
|
|
|
pStatePlusOne(psp);
|
|
psp->flags &= ~(PANF_BOB);
|
|
}
|
|
}
|
|
|
|
#if 1
|
|
void pShotgunReloadDown(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= SHOTGUN_YOFF + (tileHeight(picnum)/2))
|
|
{
|
|
PlaySound(DIGI_ROCKET_UP, psp->PlayerP,v3df_follow|v3df_dontpan|v3df_doppler);
|
|
|
|
psp->opos.Y = psp->pos.Y = SHOTGUN_YOFF + (tileHeight(picnum)/2);
|
|
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pShotgunReloadUp(PANEL_SPRITE* psp)
|
|
{
|
|
psp->opos.X = psp->pos.X = SHOTGUN_XOFF;
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
psp->PlayerP->WpnReloadState = 2;
|
|
|
|
if (psp->pos.Y < SHOTGUN_YOFF)
|
|
{
|
|
PlaySound(DIGI_SHOTGUN_UP, psp->PlayerP,v3df_follow|v3df_dontpan|v3df_doppler);
|
|
|
|
psp->opos.Y = psp->pos.Y = SHOTGUN_YOFF;
|
|
|
|
pStatePlusOne(psp);
|
|
psp->flags &= ~(PANF_BOB);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void pShotgunPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
// Needed for recoil
|
|
psp->ang = NORM_ANGLE(256 + 128);
|
|
pShotgunSetRecoil(psp);
|
|
///
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < SHOTGUN_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = SHOTGUN_YOFF;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pShotgunBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 12;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
bool pShotgunOverlays(PANEL_SPRITE* psp)
|
|
{
|
|
#define SHOTGUN_AUTO_XOFF 28
|
|
#define SHOTGUN_AUTO_YOFF -17
|
|
|
|
if (psp->over[SHOTGUN_AUTO_NUM].xoff == -1)
|
|
{
|
|
psp->over[SHOTGUN_AUTO_NUM].xoff = SHOTGUN_AUTO_XOFF;
|
|
psp->over[SHOTGUN_AUTO_NUM].yoff = SHOTGUN_AUTO_YOFF;
|
|
}
|
|
|
|
//if(psp->PlayerP->WpnShotgunAuto == 0 && psp->PlayerP->WpnRocketType == 1)
|
|
//psp->PlayerP->WpnShotgunType--;
|
|
|
|
switch (psp->PlayerP->WpnShotgunType)
|
|
{
|
|
case 0:
|
|
psp->over[SHOTGUN_AUTO_NUM].pic = -1;
|
|
psp->over[SHOTGUN_AUTO_NUM].flags |= (psf_ShadeNone);
|
|
return false;
|
|
case 1:
|
|
psp->over[SHOTGUN_AUTO_NUM].pic = SHOTGUN_AUTO;
|
|
psp->over[SHOTGUN_AUTO_NUM].flags |= (psf_ShadeNone);
|
|
return false;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
PANEL_STATE ps_ShotgunFlash[] =
|
|
{
|
|
{SHOTGUN_AUTO, 30, nullptr, &ps_ShotgunFlash[1], 0,0,0},
|
|
{0, 30, nullptr, &ps_ShotgunFlash[2], 0,0,0},
|
|
{SHOTGUN_AUTO, 30, nullptr, &ps_ShotgunFlash[3], 0,0,0},
|
|
{0, 30, nullptr, &ps_ShotgunFlash[4], 0,0,0},
|
|
{SHOTGUN_AUTO, 30, nullptr, &ps_ShotgunFlash[5], 0,0,0},
|
|
{0, 30, nullptr, &ps_ShotgunFlash[6], 0,0,0},
|
|
{SHOTGUN_AUTO, 30, nullptr, &ps_ShotgunFlash[7], 0,0,0},
|
|
{0, 30, nullptr, &ps_ShotgunFlash[8], 0,0,0},
|
|
{SHOTGUN_AUTO, 30, nullptr, &ps_ShotgunFlash[9], 0,0,0},
|
|
{0, 0, nullptr, nullptr, 0,0,0}
|
|
};
|
|
|
|
|
|
void pShotgunHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= SHOTGUN_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = SHOTGUN_YOFF + tileHeight(picnum);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
#if 1
|
|
bool pShotgunReloadTest(PANEL_SPRITE* psp)
|
|
{
|
|
//short ammo = psp->PlayerP->WpnAmmo[psp->PlayerP->WeaponType];
|
|
short ammo = psp->PlayerP->WpnAmmo[WPN_SHOTGUN];
|
|
|
|
// Reload if done with clip
|
|
if (ammo > 0 && (ammo % 4) == 0)
|
|
{
|
|
// clip has run out
|
|
psp->flags &= ~(PANF_REST_POS);
|
|
pSetState(psp, ps_ShotgunReload);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
void pShotgunRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
//short ammo = psp->PlayerP->WpnAmmo[psp->PlayerP->WeaponType];
|
|
int ammo = psp->PlayerP->WpnAmmo[WPN_SHOTGUN];
|
|
int lastammo = psp->PlayerP->WpnShotgunLastShell;
|
|
|
|
if (pWeaponHideKeys(psp, ps_ShotgunHide))
|
|
return;
|
|
|
|
if (psp->PlayerP->WpnShotgunType == 1 && ammo > 0 && ((ammo % 4) != 0) && lastammo != ammo && (psp->flags & PANF_REST_POS))
|
|
{
|
|
force = true;
|
|
}
|
|
|
|
pShotgunOverlays(psp);
|
|
|
|
pShotgunBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
psp->flags |= (PANF_REST_POS); // Used for reload checking in autofire
|
|
|
|
if (psp->PlayerP->WpnShotgunType == 0)
|
|
psp->PlayerP->WpnShotgunLastShell = ammo-1;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
if (psp->PlayerP->WpnShotgunType==0)
|
|
pSetState(psp, ps_ShotgunFire);
|
|
else
|
|
pSetState(psp, ps_ShotgunAutoFire);
|
|
}
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pShotgunRestTest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (psp->PlayerP->WpnShotgunType == 1 && !pShotgunReloadTest(psp))
|
|
force = true;
|
|
|
|
if (pShotgunReloadTest(psp))
|
|
return;
|
|
|
|
if (pWeaponHideKeys(psp, ps_ShotgunHide))
|
|
return;
|
|
|
|
pShotgunBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
return;
|
|
}
|
|
}
|
|
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
|
|
void pShotgunAction(PANEL_SPRITE* psp)
|
|
{
|
|
pShotgunBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pShotgunFire(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 32);
|
|
InitShotgun(psp->PlayerP);
|
|
//SpawnShotgunShell(psp);
|
|
}
|
|
|
|
void pShotgunRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= SHOTGUN_YOFF + tileHeight(picnum) + 50)
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[psp->WeaponType] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// RAIL
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pRailPresent(PANEL_SPRITE* psp);
|
|
void pRailRetract(PANEL_SPRITE* psp);
|
|
void pRailAction(PANEL_SPRITE* psp);
|
|
void pRailRest(PANEL_SPRITE* psp);
|
|
void pRailFire(PANEL_SPRITE* psp);
|
|
void pRailHide(PANEL_SPRITE* psp);
|
|
void pRailRestTest(PANEL_SPRITE* psp);
|
|
void pRailOkTest(PANEL_SPRITE* psp);
|
|
void pRailRecoilUp(PANEL_SPRITE* psp);
|
|
void pRailRecoilDown(PANEL_SPRITE* psp);
|
|
|
|
void pRailBobSetup(PANEL_SPRITE* psp);
|
|
|
|
bool pRailReloadTest(PANEL_SPRITE* psp);
|
|
|
|
#define Rail_BEAT_RATE 24
|
|
#define Rail_ACTION_RATE 3 // !JIM! Was 10
|
|
#define Rail_CHARGE_RATE 3
|
|
|
|
PANEL_STATE ps_PresentRail[] =
|
|
{
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailPresent, &ps_PresentRail[0], psf_ShadeNone, 0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RailRest[] =
|
|
{
|
|
{ID_RailRest0, Rail_BEAT_RATE, pRailRest, &ps_RailRest[1], psf_ShadeNone, 0,0},
|
|
{ID_RailRest1, Rail_BEAT_RATE, pRailRest, &ps_RailRest[2], psf_ShadeNone, 0,0},
|
|
{ID_RailRest2, Rail_BEAT_RATE, pRailRest, &ps_RailRest[3], psf_ShadeNone, 0,0},
|
|
{ID_RailRest3, Rail_BEAT_RATE, pRailRest, &ps_RailRest[4], psf_ShadeNone, 0,0},
|
|
{ID_RailRest4, Rail_BEAT_RATE, pRailRest, &ps_RailRest[0], psf_ShadeNone, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RailHide[] =
|
|
{
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailHide, &ps_RailHide[0], psf_ShadeNone, 0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RailRecoil[] =
|
|
{
|
|
// recoil
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailRecoilDown, &ps_RailRecoil[0], 0,0,0},
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailRecoilUp, &ps_RailRecoil[1], 0,0,0},
|
|
// ready to fire again
|
|
{ID_RailPresent0, 3, pNullAnimator, &ps_RailRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RailFire[] =
|
|
{
|
|
{ID_RailCharge0, Rail_CHARGE_RATE, pRailAction, &ps_RailFire[1], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE, pRailAction, &ps_RailFire[2], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE, pRailAction, &ps_RailFire[3], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE, pRailAction, &ps_RailFire[4], psf_ShadeNone, 0,0},
|
|
|
|
{ID_RailFire0, Rail_ACTION_RATE, pRailAction, &ps_RailFire[5], psf_ShadeNone, 0,0},
|
|
{ID_RailFire1, Rail_ACTION_RATE, pRailAction, &ps_RailFire[6], psf_ShadeNone, 0,0},
|
|
{ID_RailFire1, Rail_ACTION_RATE, pRailAction, &ps_RailFire[7], psf_ShadeNone, 0,0},
|
|
{ID_RailFire1, 0, pRailFire, &ps_RailFire[8], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
|
|
// recoil
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailRecoilDown, &ps_RailFire[8], psf_ShadeNone, 0,0},
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailRecoilUp, &ps_RailFire[9], psf_ShadeNone, 0,0},
|
|
// !JIM! I added these to introduce firing delay, that looks like a charge down.
|
|
{ID_RailCharge0, Rail_CHARGE_RATE, pRailOkTest, &ps_RailFire[11], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE, pRailAction, &ps_RailFire[12], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE, pRailOkTest, &ps_RailFire[13], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE, pRailAction, &ps_RailFire[14], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge0, Rail_CHARGE_RATE+1, pRailAction, &ps_RailFire[15], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE+1, pRailAction, &ps_RailFire[16], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE+1, pRailAction, &ps_RailFire[17], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE+2, pRailAction, &ps_RailFire[18], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge0, Rail_CHARGE_RATE+2, pRailAction, &ps_RailFire[19], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE+2, pRailAction, &ps_RailFire[20], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE+3, pRailAction, &ps_RailFire[21], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE+3, pRailAction, &ps_RailFire[22], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge0, Rail_CHARGE_RATE+4, pRailAction, &ps_RailFire[23], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE+4, pRailAction, &ps_RailFire[24], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE+4, pRailAction, &ps_RailFire[25], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge0, Rail_CHARGE_RATE+5, pRailAction, &ps_RailFire[26], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE+5, pRailAction, &ps_RailFire[27], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE+5, pRailAction, &ps_RailFire[28], psf_ShadeNone, 0,0},
|
|
|
|
{ID_RailCharge0, Rail_ACTION_RATE, pRailRestTest, &ps_RailFire[29], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_ACTION_RATE, pRailRest, &ps_RailRest[0], psf_ShadeNone, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RailFireEMP[] =
|
|
{
|
|
{ID_RailCharge0, Rail_CHARGE_RATE, pRailAction, &ps_RailFireEMP[1], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE, pRailAction, &ps_RailFireEMP[2], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge2, Rail_CHARGE_RATE, pRailAction, &ps_RailFireEMP[3], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_CHARGE_RATE, pRailAction, &ps_RailFireEMP[4], psf_ShadeNone, 0,0},
|
|
|
|
{ID_RailFire0, Rail_ACTION_RATE, pRailAction, &ps_RailFireEMP[5], psf_ShadeNone, 0,0},
|
|
{ID_RailFire1, Rail_ACTION_RATE, pRailAction, &ps_RailFireEMP[6], psf_ShadeNone, 0,0},
|
|
{ID_RailFire1, Rail_ACTION_RATE, pRailAction, &ps_RailFireEMP[7], psf_ShadeNone, 0,0},
|
|
{ID_RailFire1, 0, pRailFire, &ps_RailFireEMP[8], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
|
|
{ID_RailCharge0, Rail_ACTION_RATE, pRailRestTest, &ps_RailFireEMP[9], psf_ShadeNone, 0,0},
|
|
{ID_RailCharge1, Rail_ACTION_RATE, pRailRest, &ps_RailRest[0], psf_ShadeNone, 0,0},
|
|
};
|
|
|
|
|
|
PANEL_STATE ps_RetractRail[] =
|
|
{
|
|
{ID_RailPresent0, Rail_BEAT_RATE, pRailRetract, &ps_RetractRail[0], psf_ShadeNone, 0,0}
|
|
};
|
|
|
|
#define RAIL_YOFF 200
|
|
//#define RAIL_XOFF (160+60)
|
|
#define RAIL_XOFF (160+6)
|
|
|
|
void InitWeaponRail(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp = nullptr;
|
|
|
|
if (SW_SHAREWARE) return;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
pp->WeaponType = WPN_RAIL;
|
|
|
|
if (!(pp->WpnFlags &BIT(pp->WeaponType)) ||
|
|
// pp->WpnAmmo[pp->WeaponType] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[pp->WeaponType])
|
|
{
|
|
psp = pp->Wpn[pp->WeaponType] = pSpawnSprite(pp, ps_PresentRail, PRI_MID, RAIL_XOFF, RAIL_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[pp->WeaponType])
|
|
{
|
|
return;
|
|
}
|
|
|
|
psp->WeaponType = pp->WeaponType;
|
|
PlayerUpdateWeapon(pp, pp->WeaponType);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[pp->WeaponType];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_RailFire;
|
|
psp->RetractState = ps_RetractRail;
|
|
psp->PresentState = ps_PresentRail;
|
|
psp->RestState = ps_RailRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
PlaySound(DIGI_RAIL_UP, pp, v3df_follow);
|
|
PlaySound(DIGI_RAILREADY, pp, v3df_follow | v3df_dontpan, CHAN_ITEM); // this one needs to be on a dedicated channel to allow switching it off without too many checks.
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
void pRailSetRecoil(PANEL_SPRITE* psp)
|
|
{
|
|
psp->vel = 900;
|
|
psp->ang = NORM_ANGLE(-256);
|
|
}
|
|
|
|
void pRailRecoilDown(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel -= 24 * synctics;
|
|
|
|
if (psp->vel < 800)
|
|
{
|
|
psp->vel = 800;
|
|
psp->ang = NORM_ANGLE(psp->ang + 1024);
|
|
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pRailRecoilUp(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel += 15 * synctics;
|
|
|
|
if (psp->pos.Y < RAIL_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = RAIL_YOFF;
|
|
psp->opos.X = psp->pos.X = RAIL_XOFF;
|
|
|
|
pRailSetRecoil(psp);
|
|
|
|
pStatePlusOne(psp);
|
|
psp->flags &= ~(PANF_BOB);
|
|
}
|
|
}
|
|
|
|
void pRailPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
// Needed for recoil
|
|
psp->ang = NORM_ANGLE(256 + 128);
|
|
pRailSetRecoil(psp);
|
|
///
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < RAIL_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = RAIL_YOFF;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pRailBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 12;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pRailHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= RAIL_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = RAIL_YOFF + tileHeight(picnum);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pRailOkTest(PANEL_SPRITE* psp)
|
|
{
|
|
if (pWeaponHideKeys(psp, ps_RailHide))
|
|
return;
|
|
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pRailRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (SW_SHAREWARE) return;
|
|
|
|
if (pWeaponHideKeys(psp, ps_RailHide))
|
|
return;
|
|
|
|
pRailBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
//PlaySound(DIGI_RAILPWRUP, &psp->PlayerP->posx, &psp->PlayerP->posy, &psp->PlayerP->posz, v3df_follow);
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
if (psp->PlayerP->WpnRailType == 0)
|
|
pSetState(psp, ps_RailFire);
|
|
else
|
|
pSetState(psp, ps_RailFireEMP);
|
|
}
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pRailRestTest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_RailHide))
|
|
return;
|
|
|
|
pRailBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
return;
|
|
}
|
|
}
|
|
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
|
|
void pRailAction(PANEL_SPRITE* psp)
|
|
{
|
|
pRailBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pRailFire(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 16);
|
|
InitRail(psp->PlayerP);
|
|
}
|
|
|
|
void pRailRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= RAIL_YOFF + tileHeight(picnum) + 50)
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[psp->WeaponType] = nullptr;
|
|
DeleteNoSoundOwner(psp->PlayerP->actor);
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// HOTHEAD
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pHotheadPresent(PANEL_SPRITE* psp);
|
|
void pHotheadRetract(PANEL_SPRITE* psp);
|
|
void pHotheadAction(PANEL_SPRITE* psp);
|
|
void pHotheadActionCenter(PANEL_SPRITE* psp);
|
|
void pHotheadRest(PANEL_SPRITE* psp);
|
|
void pHotheadRestCenter(PANEL_SPRITE* psp);
|
|
void pHotheadAttack(PANEL_SPRITE* psp);
|
|
void pHotheadRing(PANEL_SPRITE* psp);
|
|
void pHotheadNapalm(PANEL_SPRITE* psp);
|
|
void pHotheadHide(PANEL_SPRITE* psp);
|
|
void pHotheadRestTest(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_HotheadAttack[];
|
|
extern PANEL_STATE ps_ReloadHothead[];
|
|
extern PANEL_STATE ps_HotheadTurn[];
|
|
|
|
|
|
#define Hothead_BEAT_RATE 24
|
|
#define Hothead_ACTION_RATE_PRE 2 // !JIM! Was 1
|
|
#define Hothead_ACTION_RATE_POST 7
|
|
|
|
PANEL_STATE ps_PresentHothead[] =
|
|
{
|
|
{ID_HotheadPresent0, Hothead_BEAT_RATE, pHotheadPresent, &ps_PresentHothead[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadHide[] =
|
|
{
|
|
{ID_HotheadRest0, Hothead_BEAT_RATE, pHotheadHide, &ps_HotheadHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RetractHothead[] =
|
|
{
|
|
{ID_HotheadPresent0, Hothead_BEAT_RATE, pHotheadRetract, &ps_RetractHothead[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadRest[] =
|
|
{
|
|
{ID_HotheadRest0, Hothead_BEAT_RATE, pHotheadRest, &ps_HotheadRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadRestRing[] =
|
|
{
|
|
{ID_HotheadRest0, Hothead_BEAT_RATE, pHotheadRest, &ps_HotheadRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadRestNapalm[] =
|
|
{
|
|
{ID_HotheadRest0, Hothead_BEAT_RATE, pHotheadRest, &ps_HotheadRest[0], 0,0,0}
|
|
};
|
|
// Turns - attacks
|
|
|
|
PANEL_STATE ps_HotheadAttack[] =
|
|
{
|
|
{ID_HotheadAttack0, Hothead_ACTION_RATE_PRE, pHotheadAction, &ps_HotheadAttack[1], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 3, pHotheadAction, &ps_HotheadAttack[2], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadAttack, &ps_HotheadAttack[3], psf_QuickCall, 0,0},
|
|
{ID_HotheadAttack0, 3, pHotheadAction, &ps_HotheadAttack[4], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadRestTest, &ps_HotheadAttack[4], psf_QuickCall, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadAction, &ps_HotheadAttack[0], psf_ShadeHalf, 0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadRing[] =
|
|
{
|
|
{ID_HotheadAttack0, Hothead_ACTION_RATE_PRE, pHotheadAction, &ps_HotheadRing[1], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 10, pHotheadAction, &ps_HotheadRing[2], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadAttack, &ps_HotheadRing[3], psf_QuickCall, 0,0},
|
|
{ID_HotheadAttack0, 40, pHotheadAction, &ps_HotheadRing[4], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadRestTest, &ps_HotheadRing[4], psf_QuickCall, 0,0},
|
|
{ID_HotheadAttack0, 3, pHotheadAction, &ps_HotheadRing[0], psf_ShadeHalf, 0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadNapalm[] =
|
|
{
|
|
{ID_HotheadAttack0, Hothead_ACTION_RATE_PRE, pHotheadAction, &ps_HotheadNapalm[1], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 3, pHotheadAction, &ps_HotheadNapalm[2], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadAttack, &ps_HotheadNapalm[3], psf_QuickCall, 0,0},
|
|
{ID_HotheadAttack0, 50, pHotheadAction, &ps_HotheadNapalm[4], psf_ShadeHalf, 0,0},
|
|
{ID_HotheadAttack0, 0, pHotheadRestTest, &ps_HotheadNapalm[4], psf_QuickCall, 0,0},
|
|
{ID_HotheadAttack0, 3, pHotheadAction, &ps_HotheadNapalm[0], psf_ShadeHalf, 0,0}
|
|
};
|
|
|
|
// Turns - can do three different turns
|
|
|
|
PANEL_STATE ps_HotheadTurn[] =
|
|
{
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[1], 0,0,0},
|
|
{ID_HotheadTurn1, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[2], 0,0,0},
|
|
{ID_HotheadTurn2, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[3], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[4], 0,0,0},
|
|
{ID_HotheadChomp0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[5], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[6], 0,0,0},
|
|
{ID_HotheadChomp0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[7], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[8], 0,0,0},
|
|
{ID_HotheadTurn2, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[9], 0,0,0},
|
|
{ID_HotheadTurn1, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[10], 0,0,0},
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurn[11], 0,0,0},
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadTurnRing[] =
|
|
{
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[1], 0,0,0},
|
|
{ID_HotheadTurn1, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[2], 0,0,0},
|
|
{ID_HotheadTurn2, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[3], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[4], 0,0,0},
|
|
{ID_HotheadChomp0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[5], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[6], 0,0,0},
|
|
{ID_HotheadTurn2, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[7], 0,0,0},
|
|
{ID_HotheadTurn1, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[8], 0,0,0},
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnRing[9], 0,0,0},
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadRestRing[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HotheadTurnNapalm[] =
|
|
{
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[1], 0,0,0},
|
|
{ID_HotheadTurn1, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[2], 0,0,0},
|
|
{ID_HotheadTurn2, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[3], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[4], 0,0,0},
|
|
{ID_HotheadTurn3, Hothead_BEAT_RATE*2, pNullAnimator, &ps_HotheadTurnNapalm[5], 0,0,0},
|
|
{ID_HotheadTurn2, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[6], 0,0,0},
|
|
{ID_HotheadTurn1, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[7], 0,0,0},
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadTurnNapalm[8], 0,0,0},
|
|
{ID_HotheadTurn0, Hothead_BEAT_RATE, pNullAnimator, &ps_HotheadRestNapalm[0], 0,0,0}
|
|
};
|
|
|
|
|
|
PANEL_STATE* HotheadAttackStates[] =
|
|
{
|
|
ps_HotheadAttack,
|
|
ps_HotheadRing,
|
|
ps_HotheadNapalm
|
|
};
|
|
|
|
PANEL_STATE* HotheadRestStates[] =
|
|
{
|
|
ps_HotheadRest,
|
|
ps_HotheadRestRing,
|
|
ps_HotheadRestNapalm
|
|
};
|
|
|
|
PANEL_STATE* HotheadTurnStates[] =
|
|
{
|
|
ps_HotheadTurn,
|
|
ps_HotheadTurnRing,
|
|
ps_HotheadTurnNapalm
|
|
};
|
|
|
|
#define FIREBALL_MODE 0
|
|
#define RING_MODE 1
|
|
#define NAPALM_MODE 2
|
|
void pHotHeadOverlays(PANEL_SPRITE* psp, short mode)
|
|
{
|
|
#define HOTHEAD_FINGER_XOFF 0
|
|
#define HOTHEAD_FINGER_YOFF 0
|
|
|
|
switch (mode)
|
|
{
|
|
case 0: // Great balls o' fire
|
|
psp->over[0].pic = HEAD_MODE1;
|
|
break;
|
|
case 1: // Ring of fire
|
|
psp->over[0].pic = HEAD_MODE2;
|
|
break;
|
|
case 2: // I love the smell of napalm in the morning
|
|
psp->over[0].pic = HEAD_MODE3;
|
|
break;
|
|
}
|
|
}
|
|
|
|
#define HOTHEAD_BOB_X_AMT 10
|
|
|
|
#define HOTHEAD_XOFF (200 + HOTHEAD_BOB_X_AMT + 6)
|
|
#define HOTHEAD_YOFF 200
|
|
|
|
void InitWeaponHothead(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp = nullptr;
|
|
|
|
if (SW_SHAREWARE) return;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_HOTHEAD)) ||
|
|
// pp->WpnAmmo[WPN_HOTHEAD] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[WPN_HOTHEAD])
|
|
{
|
|
psp = pp->Wpn[WPN_HOTHEAD] = pSpawnSprite(pp, ps_PresentHothead, PRI_MID, HOTHEAD_XOFF, HOTHEAD_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_HOTHEAD])
|
|
{
|
|
return;
|
|
}
|
|
|
|
psp->WeaponType = WPN_HOTHEAD;
|
|
PlayerUpdateWeapon(pp, WPN_HOTHEAD);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_HOTHEAD];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_HotheadAttack;
|
|
psp->PresentState = ps_PresentHothead;
|
|
psp->RestState = HotheadRestStates[psp->PlayerP->WpnFlameType];
|
|
psp->RetractState = ps_RetractHothead;
|
|
pSetState(psp, psp->PresentState);
|
|
psp->ang = 768;
|
|
psp->vel = 512;
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
pHotHeadOverlays(psp, pp->WpnFlameType);
|
|
psp->over[0].xoff = HOTHEAD_FINGER_XOFF;
|
|
psp->over[0].yoff = HOTHEAD_FINGER_YOFF;
|
|
|
|
PlaySound(DIGI_GRDALERT, pp, v3df_follow|v3df_dontpan);
|
|
}
|
|
|
|
void pHotheadRestTest(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
//if (!(psp->PlayerP->Flags & PF_DIVING))
|
|
{
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
if (psp->PlayerP->WpnAmmo[WPN_HOTHEAD] < 10)
|
|
{
|
|
psp->PlayerP->WpnFlameType = 0;
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
}
|
|
|
|
pStatePlusOne(psp);
|
|
return;
|
|
}
|
|
}
|
|
|
|
pSetState(psp, HotheadRestStates[psp->PlayerP->WpnFlameType]);
|
|
psp->over[0].xoff = HOTHEAD_FINGER_XOFF;
|
|
psp->over[0].yoff = HOTHEAD_FINGER_YOFF;
|
|
}
|
|
|
|
void pHotheadPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < HOTHEAD_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = HOTHEAD_YOFF;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
//pSetState(psp, HotheadTurnStates[psp->PlayerP->WpnFlameType]);
|
|
}
|
|
}
|
|
|
|
void pHotheadBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = HOTHEAD_BOB_X_AMT;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 4;
|
|
}
|
|
|
|
void pHotheadHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupx();
|
|
psp->pos.X += 3 * synctics;
|
|
|
|
if (psp->pos.X >= HOTHEAD_XOFF + tileWidth(picnum) || psp->pos.Y >= HOTHEAD_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.X = psp->pos.X = HOTHEAD_XOFF;
|
|
psp->opos.Y = psp->pos.Y = HOTHEAD_YOFF + tileHeight(picnum);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
|
|
void pHotheadRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (SW_SHAREWARE) return;
|
|
|
|
if (pWeaponHideKeys(psp, ps_HotheadHide))
|
|
return;
|
|
|
|
if (HotheadRestStates[psp->PlayerP->WpnFlameType] != psp->RestState)
|
|
{
|
|
psp->RestState = HotheadRestStates[psp->PlayerP->WpnFlameType];
|
|
pSetState(psp, HotheadRestStates[psp->PlayerP->WpnFlameType]);
|
|
}
|
|
|
|
// in rest position - only bob when in rest pos
|
|
pHotheadBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
//if ((psp->PlayerP->Flags & PF_DIVING))
|
|
// return;
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
if (psp->PlayerP->WpnAmmo[WPN_HOTHEAD] < 10)
|
|
{
|
|
psp->PlayerP->WpnFlameType = 0;
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
//pSetState(psp, psp->ActionState);
|
|
pSetState(psp, HotheadAttackStates[psp->PlayerP->WpnFlameType]);
|
|
psp->over[0].xoff = HOTHEAD_FINGER_XOFF-1;
|
|
psp->over[0].yoff = HOTHEAD_FINGER_YOFF-10;
|
|
}
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pHotheadAction(PANEL_SPRITE* psp)
|
|
{
|
|
bool shooting = (psp->PlayerP->input.actions & SB_FIRE) && (psp->PlayerP->KeyPressBits & SB_FIRE);
|
|
|
|
if (shooting)
|
|
{
|
|
pUziBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP) || shooting);
|
|
}
|
|
}
|
|
|
|
void pHotheadAttack(PANEL_SPRITE* psp)
|
|
{
|
|
switch (psp->PlayerP->WpnFlameType)
|
|
{
|
|
case 0:
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 32);
|
|
InitFireball(psp->PlayerP);
|
|
break;
|
|
case 1:
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 20);
|
|
InitSpellRing(psp->PlayerP);
|
|
break;
|
|
case 2:
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 16);
|
|
InitSpellNapalm(psp->PlayerP);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void pHotheadRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= HOTHEAD_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_HOTHEAD] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// HOTHEAD ON_FIRE
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pOnFire(PANEL_SPRITE*);
|
|
|
|
#define ON_FIRE_RATE 10
|
|
|
|
PANEL_STATE ps_OnFire[] =
|
|
{
|
|
{ID_OnFire0, ON_FIRE_RATE, pOnFire, &ps_OnFire[1], 0,0,0},
|
|
{ID_OnFire1, ON_FIRE_RATE, pOnFire, &ps_OnFire[2], 0,0,0},
|
|
{ID_OnFire2, ON_FIRE_RATE, pOnFire, &ps_OnFire[3], 0,0,0},
|
|
{ID_OnFire3, ON_FIRE_RATE, pOnFire, &ps_OnFire[4], 0,0,0},
|
|
{ID_OnFire4, ON_FIRE_RATE, pOnFire, &ps_OnFire[5], 0,0,0},
|
|
{ID_OnFire5, ON_FIRE_RATE, pOnFire, &ps_OnFire[6], 0,0,0},
|
|
{ID_OnFire6, ON_FIRE_RATE, pOnFire, &ps_OnFire[7], 0,0,0},
|
|
{ID_OnFire7, ON_FIRE_RATE, pOnFire, &ps_OnFire[8], 0,0,0},
|
|
{ID_OnFire8, ON_FIRE_RATE, pOnFire, &ps_OnFire[9], 0,0,0},
|
|
{ID_OnFire9, ON_FIRE_RATE, pOnFire, &ps_OnFire[10], 0,0,0},
|
|
{ID_OnFire10, ON_FIRE_RATE, pOnFire, &ps_OnFire[11], 0,0,0},
|
|
{ID_OnFire11, ON_FIRE_RATE, pOnFire, &ps_OnFire[12], 0,0,0},
|
|
{ID_OnFire12, ON_FIRE_RATE, pOnFire, &ps_OnFire[0], 0,0,0},
|
|
};
|
|
|
|
#define ON_FIRE_Y_TOP 190
|
|
#define ON_FIRE_Y_BOT 230
|
|
|
|
void SpawnOnFire(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* fire;
|
|
short x = 50;
|
|
|
|
while (x < 320)
|
|
{
|
|
fire = pSpawnSprite(pp, &ps_OnFire[RANDOM_P2(8<<8)>>8], PRI_FRONT, x, ON_FIRE_Y_BOT);
|
|
fire->flags |= PANF_WEAPON_SPRITE;
|
|
x += tileWidth(fire->picndx);
|
|
}
|
|
}
|
|
|
|
void pOnFire(PANEL_SPRITE* psp)
|
|
{
|
|
DSWActor* plActor = psp->PlayerP->actor;
|
|
|
|
// Kill immediately - in case of death/water
|
|
if (plActor->user.flameActor == nullptr && plActor->user.Flags2 & SPR2_FLAMEDIE)
|
|
{
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
|
|
psp->backupy();
|
|
|
|
if (plActor->user.flameActor == nullptr)
|
|
{
|
|
// take flames down and kill them
|
|
psp->pos.Y += 1;
|
|
if (psp->pos.Y > ON_FIRE_Y_BOT)
|
|
{
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// bring flames up
|
|
psp->pos.Y -= 2;
|
|
if (psp->pos.Y < ON_FIRE_Y_TOP)
|
|
psp->opos.Y = psp->pos.Y = ON_FIRE_Y_TOP;
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MICRO
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pMicroPresent(PANEL_SPRITE* psp);
|
|
void pMicroRetract(PANEL_SPRITE* psp);
|
|
void pMicroAction(PANEL_SPRITE* psp);
|
|
void pMicroRest(PANEL_SPRITE* psp);
|
|
void pMicroFire(PANEL_SPRITE* psp);
|
|
void pMicroHide(PANEL_SPRITE* psp);
|
|
void pMicroUnHide(PANEL_SPRITE* psp);
|
|
void pMicroRecoilDown(PANEL_SPRITE* psp);
|
|
void pMicroRecoilUp(PANEL_SPRITE* psp);
|
|
void pMicroBobSetup(PANEL_SPRITE* psp);
|
|
void pMicroReloadUp(PANEL_SPRITE* psp);
|
|
void pMicroReloadDown(PANEL_SPRITE* psp);
|
|
void pMicroStandBy(PANEL_SPRITE* psp);
|
|
void pMicroCount(PANEL_SPRITE* psp);
|
|
void pMicroReady(PANEL_SPRITE* psp);
|
|
void pNukeAction(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_MicroReload[];
|
|
|
|
#define Micro_REST_RATE 24
|
|
#define Micro_ACTION_RATE 6 // !JIM! was 9
|
|
|
|
PANEL_STATE ps_PresentMicro[] =
|
|
{
|
|
{ID_MicroPresent0, Micro_REST_RATE, pMicroPresent, &ps_PresentMicro[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MicroRest[] =
|
|
{
|
|
{ID_MicroPresent0, Micro_REST_RATE, pMicroRest, &ps_MicroRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MicroHide[] =
|
|
{
|
|
{ID_MicroPresent0, Micro_REST_RATE, pMicroHide, &ps_MicroHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_InitNuke[] =
|
|
{
|
|
{ID_MicroPresent0, Micro_ACTION_RATE,pNukeAction, &ps_InitNuke[1], 0,0,0},
|
|
{ID_MicroPresent0, 0, pMicroStandBy, &ps_InitNuke[2], psf_QuickCall, 0,0},
|
|
{ID_MicroPresent0, 120*2, pNukeAction, &ps_InitNuke[3], 0,0,0},
|
|
{ID_MicroPresent0, 0, pMicroCount, &ps_InitNuke[4], psf_QuickCall, 0,0},
|
|
{ID_MicroPresent0, 120*3, pNukeAction, &ps_InitNuke[5], 0,0,0},
|
|
{ID_MicroPresent0, 0, pMicroReady, &ps_InitNuke[6], psf_QuickCall, 0,0},
|
|
{ID_MicroPresent0, 120*2, pNukeAction, &ps_InitNuke[7], 0,0,0},
|
|
{ID_MicroPresent0, 3, pNukeAction, &ps_MicroRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MicroRecoil[] =
|
|
{
|
|
// recoil
|
|
{ID_MicroPresent0, Micro_ACTION_RATE, pMicroRecoilDown, &ps_MicroRecoil[0], 0,0,0},
|
|
{ID_MicroPresent0, Micro_ACTION_RATE, pMicroRecoilUp, &ps_MicroRecoil[1], 0,0,0},
|
|
|
|
// Firing delay.
|
|
{ID_MicroPresent0, 30, pNullAnimator, &ps_MicroRecoil[3], 0,0,0},
|
|
// ready to fire again
|
|
{ID_MicroPresent0, 3, pNullAnimator, &ps_MicroRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MicroFire[] =
|
|
{
|
|
{ID_MicroFire0, Micro_ACTION_RATE, pMicroAction, &ps_MicroFire[1], psf_ShadeNone, 0,0},
|
|
{ID_MicroFire1, Micro_ACTION_RATE, pMicroAction, &ps_MicroFire[2], psf_ShadeNone, 0,0},
|
|
{ID_MicroFire2, Micro_ACTION_RATE, pMicroAction, &ps_MicroFire[3], psf_ShadeHalf, 0,0},
|
|
{ID_MicroFire3, Micro_ACTION_RATE, pMicroAction, &ps_MicroFire[4], psf_ShadeHalf, 0,0},
|
|
{ID_MicroPresent0, 0, pMicroFire, &ps_MicroFire[5], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
|
|
|
|
// !JIM! After firing delay so rockets can't fire so fast!
|
|
// Putting a BIG blast radius for rockets, this is better than small and fast for this weap.
|
|
{ID_MicroPresent0, 120, pMicroAction, &ps_MicroFire[6], 0,0,0},
|
|
|
|
{ID_MicroPresent0, 3, pMicroAction, &ps_MicroRecoil[0], 0,0,0}
|
|
};
|
|
|
|
#define Micro_SINGLE_RATE 8
|
|
#define Micro_DISSIPATE_RATE 6
|
|
|
|
PANEL_STATE ps_MicroSingleFire[] =
|
|
{
|
|
{ID_MicroSingleFire0, Micro_SINGLE_RATE, pMicroAction, &ps_MicroSingleFire[1], psf_ShadeHalf, 0,0},
|
|
{ID_MicroSingleFire1, Micro_SINGLE_RATE, pMicroAction, &ps_MicroSingleFire[2], psf_ShadeNone, 0,0},
|
|
{ID_MicroSingleFire1, 0, pMicroFire, &ps_MicroSingleFire[3], psf_ShadeNone|psf_QuickCall, 0,0},
|
|
{ID_MicroSingleFire2, Micro_DISSIPATE_RATE, pMicroAction, &ps_MicroSingleFire[4], psf_ShadeNone, 0,0},
|
|
{ID_MicroSingleFire3, Micro_DISSIPATE_RATE, pMicroAction, &ps_MicroSingleFire[5], psf_ShadeHalf, 0,0},
|
|
|
|
// !JIM! Put in firing delay.
|
|
//{ID_MicroPresent0, 60, pMicroAction, &ps_MicroSingleFire[6]},
|
|
|
|
{ID_MicroPresent0, 3, pMicroAction, &ps_MicroRecoil[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RetractMicro[] =
|
|
{
|
|
{ID_MicroPresent0, Micro_REST_RATE, pMicroRetract, &ps_RetractMicro[0], 0,0,0}
|
|
};
|
|
|
|
#define MICRO_BOB_X_AMT 10
|
|
#define MICRO_YOFF 205
|
|
#define MICRO_XOFF (150+MICRO_BOB_X_AMT)
|
|
|
|
void pMicroSetRecoil(PANEL_SPRITE* psp)
|
|
{
|
|
psp->vel = 900;
|
|
psp->ang = NORM_ANGLE(-256);
|
|
}
|
|
|
|
void InitWeaponMicro(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_MICRO)) ||
|
|
// pp->WpnAmmo[WPN_MICRO] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[WPN_MICRO])
|
|
{
|
|
psp = pp->Wpn[WPN_MICRO] = pSpawnSprite(pp, ps_PresentMicro, PRI_MID, MICRO_XOFF, MICRO_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_MICRO])
|
|
{
|
|
if (pp->TestNukeInit && pp->WpnRocketType == 2 && !pp->InitingNuke && pp->WpnRocketNuke && !pp->NukeInitialized)
|
|
{
|
|
pp->TestNukeInit = false;
|
|
pp->InitingNuke = true;
|
|
psp = pp->Wpn[WPN_MICRO];
|
|
pSetState(psp, ps_InitNuke);
|
|
}
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_MICRO);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_MICRO];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_MicroFire;
|
|
psp->RetractState = ps_RetractMicro;
|
|
psp->RestState = ps_MicroRest;
|
|
psp->PresentState = ps_PresentMicro;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
if (pp->WpnRocketType == 2 && !pp->InitingNuke && !pp->NukeInitialized)
|
|
pp->TestNukeInit = pp->InitingNuke = true;
|
|
|
|
PlaySound(DIGI_ROCKET_UP, pp, v3df_follow);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
|
|
}
|
|
|
|
|
|
void pMicroRecoilDown(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel -= 24 * synctics;
|
|
|
|
if (psp->vel < 550)
|
|
{
|
|
psp->vel = 550;
|
|
psp->ang = NORM_ANGLE(psp->ang + 1024);
|
|
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pMicroRecoilUp(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel += 15 * synctics;
|
|
|
|
if (psp->pos.Y < MICRO_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = MICRO_YOFF;
|
|
psp->opos.X = psp->pos.X = MICRO_XOFF;
|
|
|
|
pMicroSetRecoil(psp);
|
|
|
|
pStatePlusOne(psp);
|
|
psp->flags &= ~(PANF_BOB);
|
|
}
|
|
}
|
|
|
|
void pMicroPresent(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
// Needed for recoil
|
|
psp->ang = NORM_ANGLE(256 + 96);
|
|
pMicroSetRecoil(psp);
|
|
///
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < MICRO_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = MICRO_YOFF;
|
|
psp->backupboby();
|
|
if (pp->WpnRocketType == 2 && !pp->NukeInitialized)
|
|
{
|
|
pp->TestNukeInit = false;
|
|
pSetState(psp, ps_InitNuke);
|
|
}
|
|
else
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pMicroBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = MICRO_BOB_X_AMT;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pMicroHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= MICRO_YOFF + tileHeight(picnum) + 20)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = MICRO_YOFF + tileHeight(picnum) + 20;
|
|
psp->opos.X = psp->pos.X = MICRO_XOFF;
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
bool pMicroOverlays(PANEL_SPRITE* psp)
|
|
{
|
|
#define MICRO_SIGHT_XOFF 29
|
|
#define MICRO_SIGHT_YOFF -58
|
|
|
|
#define MICRO_SHOT_XOFF 65
|
|
#define MICRO_SHOT_YOFF -41
|
|
|
|
#define MICRO_HEAT_XOFF 78
|
|
#define MICRO_HEAT_YOFF -51
|
|
|
|
if (psp->over[MICRO_SIGHT_NUM].xoff == -1)
|
|
{
|
|
psp->over[MICRO_SIGHT_NUM].xoff = MICRO_SIGHT_XOFF;
|
|
psp->over[MICRO_SIGHT_NUM].yoff = MICRO_SIGHT_YOFF;
|
|
psp->over[MICRO_SHOT_NUM].xoff = MICRO_SHOT_XOFF;
|
|
psp->over[MICRO_SHOT_NUM].yoff = MICRO_SHOT_YOFF;
|
|
psp->over[MICRO_HEAT_NUM].xoff = MICRO_HEAT_XOFF;
|
|
psp->over[MICRO_HEAT_NUM].yoff = MICRO_HEAT_YOFF;
|
|
|
|
}
|
|
|
|
if (psp->PlayerP->WpnRocketNuke == 0 && psp->PlayerP->WpnRocketType == 2)
|
|
psp->PlayerP->WpnRocketType=0;
|
|
|
|
switch (psp->PlayerP->WpnRocketType)
|
|
{
|
|
case 0:
|
|
psp->over[MICRO_SIGHT_NUM].pic = MICRO_SIGHT;
|
|
psp->over[MICRO_SHOT_NUM].pic = MICRO_SHOT_1;
|
|
psp->over[MICRO_SHOT_NUM].flags |= (psf_ShadeNone);
|
|
psp->over[MICRO_HEAT_NUM].pic = -1;
|
|
return false;
|
|
case 1:
|
|
if (psp->PlayerP->WpnRocketHeat)
|
|
{
|
|
psp->over[MICRO_SIGHT_NUM].pic = MICRO_SIGHT;
|
|
psp->over[MICRO_SHOT_NUM].pic = MICRO_SHOT_1;
|
|
psp->over[MICRO_SHOT_NUM].flags |= (psf_ShadeNone);
|
|
|
|
ASSERT(psp->PlayerP->WpnRocketHeat < 6);
|
|
|
|
psp->over[MICRO_HEAT_NUM].pic = MICRO_HEAT + (5 - psp->PlayerP->WpnRocketHeat);
|
|
}
|
|
else
|
|
{
|
|
psp->over[MICRO_SIGHT_NUM].pic = MICRO_SIGHT;
|
|
psp->over[MICRO_SHOT_NUM].pic = MICRO_SHOT_1;
|
|
psp->over[MICRO_SHOT_NUM].flags |= (psf_ShadeNone);
|
|
psp->over[MICRO_HEAT_NUM].pic = -1;
|
|
}
|
|
|
|
return false;
|
|
case 2:
|
|
psp->over[MICRO_SIGHT_NUM].pic = -1;
|
|
psp->over[MICRO_HEAT_NUM].pic = -1;
|
|
|
|
psp->over[MICRO_SHOT_NUM].pic = MICRO_SHOT_20;
|
|
psp->over[MICRO_SHOT_NUM].flags |= (psf_ShadeNone);
|
|
psp->over[MICRO_HEAT_NUM].flags |= (psf_ShadeNone);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
PANEL_STATE ps_MicroHeatFlash[] =
|
|
{
|
|
{MICRO_HEAT, 30, nullptr, &ps_MicroHeatFlash[1], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroHeatFlash[2], 0,0,0},
|
|
{MICRO_HEAT, 30, nullptr, &ps_MicroHeatFlash[3], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroHeatFlash[4], 0,0,0},
|
|
{MICRO_HEAT, 30, nullptr, &ps_MicroHeatFlash[5], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroHeatFlash[6], 0,0,0},
|
|
{MICRO_HEAT, 30, nullptr, &ps_MicroHeatFlash[7], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroHeatFlash[8], 0,0,0},
|
|
{MICRO_HEAT, 30, nullptr, &ps_MicroHeatFlash[9], 0,0,0},
|
|
{0, 0, nullptr, nullptr, 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MicroNukeFlash[] =
|
|
{
|
|
{MICRO_SHOT_20, 30, nullptr, &ps_MicroNukeFlash[1], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroNukeFlash[2], 0,0,0},
|
|
{MICRO_SHOT_20, 30, nullptr, &ps_MicroNukeFlash[3], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroNukeFlash[4], 0,0,0},
|
|
{MICRO_SHOT_20, 30, nullptr, &ps_MicroNukeFlash[5], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroNukeFlash[6], 0,0,0},
|
|
{MICRO_SHOT_20, 30, nullptr, &ps_MicroNukeFlash[7], 0,0,0},
|
|
{0, 30, nullptr, &ps_MicroNukeFlash[8], 0,0,0},
|
|
{MICRO_SHOT_20, 30, nullptr, &ps_MicroNukeFlash[9], 0,0,0},
|
|
{0, 0, nullptr, nullptr, 0,0,0}
|
|
};
|
|
|
|
void pMicroRest(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_MicroHide))
|
|
return;
|
|
|
|
pMicroOverlays(psp);
|
|
|
|
pMicroBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if (pp->InitingNuke)
|
|
{
|
|
int choose_voc=0;
|
|
|
|
pp->InitingNuke = false;
|
|
pp->NukeInitialized = true;
|
|
|
|
if (pp == Player+myconnectindex)
|
|
{
|
|
choose_voc = StdRandomRange(1024);
|
|
if (choose_voc > 600)
|
|
PlayerSound(DIGI_TAUNTAI2,v3df_dontpan|v3df_follow,psp->PlayerP);
|
|
else if (choose_voc > 300)
|
|
PlayerSound(DIGI_TAUNTAI4,v3df_dontpan|v3df_follow,psp->PlayerP);
|
|
}
|
|
}
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
if (psp->PlayerP->WpnAmmo[WPN_MICRO] <= 0 && psp->PlayerP->WpnRocketType != 2)
|
|
{
|
|
psp->PlayerP->WpnRocketNuke = 0;
|
|
WeaponOK(psp->PlayerP);
|
|
psp->PlayerP->WpnRocketNuke = 1;
|
|
return;
|
|
}
|
|
|
|
switch (psp->PlayerP->WpnRocketType)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
pSetState(psp, ps_MicroSingleFire);
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
break;
|
|
case 2:
|
|
if (psp->PlayerP->WpnRocketNuke > 0)
|
|
pSetState(psp, ps_MicroSingleFire);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pMicroAction(PANEL_SPRITE* psp)
|
|
{
|
|
pMicroBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pMicroFire(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 20);
|
|
switch (psp->PlayerP->WpnRocketType)
|
|
{
|
|
case 0:
|
|
if (sw_bunnyrockets)
|
|
InitBunnyRocket(psp->PlayerP);
|
|
else
|
|
InitRocket(psp->PlayerP);
|
|
break;
|
|
case 1:
|
|
if (sw_bunnyrockets)
|
|
InitBunnyRocket(psp->PlayerP);
|
|
else
|
|
InitRocket(psp->PlayerP);
|
|
break;
|
|
case 2:
|
|
PlaySound(DIGI_WARNING,psp->PlayerP,v3df_dontpan|v3df_follow);
|
|
InitNuke(psp->PlayerP);
|
|
psp->PlayerP->NukeInitialized = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void pMicroRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= MICRO_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_MICRO] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
void pNukeAction(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
#if 0 // Code commented out as it's causing interpolation issues when initialising a nuke.
|
|
psp->backupy();
|
|
psp->y -= 3 * synctics;
|
|
|
|
if (psp->y < MICRO_YOFF)
|
|
{
|
|
psp->oy = psp->y = MICRO_YOFF;
|
|
psp->backupboby();
|
|
}
|
|
#endif
|
|
|
|
pMicroBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
if (!pp->InitingNuke)
|
|
pSetState(psp, psp->PresentState);
|
|
}
|
|
|
|
void pMicroStandBy(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
pMicroOverlays(psp);
|
|
PlaySound(DIGI_NUKESTDBY, pp, v3df_follow|v3df_dontpan, CHAN_WEAPON);
|
|
}
|
|
|
|
void pMicroCount(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
PlaySound(DIGI_NUKECDOWN, pp, v3df_follow|v3df_dontpan, CHAN_WEAPON);
|
|
}
|
|
|
|
void pMicroReady(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
PlaySound(DIGI_NUKEREADY, pp, v3df_follow|v3df_dontpan, CHAN_WEAPON);
|
|
pp->NukeInitialized = true;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// HEART
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pHeartPresent(PANEL_SPRITE* psp);
|
|
void pHeartRetract(PANEL_SPRITE* psp);
|
|
void pHeartAction(PANEL_SPRITE* psp);
|
|
void pHeartRest(PANEL_SPRITE* psp);
|
|
void pHeartAttack(PANEL_SPRITE* psp);
|
|
void pHeartHide(PANEL_SPRITE* psp);
|
|
void pHeartActionBlood(PANEL_SPRITE* psp);
|
|
void SpawnSmallHeartBlood(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_HeartAttack[];
|
|
extern PANEL_STATE ps_ReloadHeart[];
|
|
|
|
|
|
#define Heart_BEAT_RATE 60
|
|
#define Heart_ACTION_RATE 10
|
|
|
|
PANEL_STATE ps_PresentHeart[] =
|
|
{
|
|
{ID_HeartPresent0, Heart_BEAT_RATE, pHeartPresent, &ps_PresentHeart[1], 0,0,0},
|
|
{ID_HeartPresent1, Heart_BEAT_RATE, pHeartPresent, &ps_PresentHeart[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HeartRest[] =
|
|
{
|
|
{ID_HeartPresent0, Heart_BEAT_RATE, pHeartRest, &ps_HeartRest[1], 0,0,0},
|
|
{ID_HeartPresent1, Heart_BEAT_RATE, pHeartRest, &ps_HeartRest[2], 0,0,0},
|
|
{ID_HeartPresent1, Heart_BEAT_RATE, SpawnSmallHeartBlood, &ps_HeartRest[3], psf_QuickCall, 0,0},
|
|
{ID_HeartPresent1, 0, pHeartRest, &ps_HeartRest[0], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_HeartHide[] =
|
|
{
|
|
{ID_HeartPresent0, Heart_BEAT_RATE, pHeartHide, &ps_HeartHide[1], 0,0,0},
|
|
{ID_HeartPresent1, Heart_BEAT_RATE, pHeartHide, &ps_HeartHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_HeartAttack[] =
|
|
{
|
|
// squeeze
|
|
{ID_HeartAttack0, Heart_ACTION_RATE, pHeartActionBlood, &ps_HeartAttack[1], psf_ShadeHalf, 0,0},
|
|
{ID_HeartAttack1, Heart_ACTION_RATE, pHeartActionBlood, &ps_HeartAttack[2], psf_ShadeNone, 0,0},
|
|
{ID_HeartAttack1, Heart_ACTION_RATE, pHeartActionBlood, &ps_HeartAttack[3], psf_ShadeNone, 0,0},
|
|
// attack
|
|
{ID_HeartAttack1, Heart_ACTION_RATE, pHeartAttack, &ps_HeartAttack[4], psf_QuickCall, 0,0},
|
|
// unsqueeze
|
|
{ID_HeartAttack1, Heart_ACTION_RATE, pHeartAction, &ps_HeartAttack[5], psf_ShadeNone, 0,0},
|
|
{ID_HeartAttack1, Heart_ACTION_RATE, pHeartAction, &ps_HeartAttack[6], psf_ShadeNone, 0,0},
|
|
{ID_HeartAttack0, Heart_ACTION_RATE, pHeartAction, &ps_HeartAttack[7], psf_ShadeHalf, 0,0},
|
|
|
|
{ID_HeartAttack0, Heart_ACTION_RATE, pHeartAction, &ps_HeartRest[0], psf_ShadeHalf, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RetractHeart[] =
|
|
{
|
|
{ID_HeartPresent0, Heart_BEAT_RATE, pHeartRetract, &ps_RetractHeart[1], 0,0,0},
|
|
{ID_HeartPresent1, Heart_BEAT_RATE, pHeartRetract, &ps_RetractHeart[0], 0,0,0}
|
|
};
|
|
|
|
#define HEART_YOFF 212
|
|
|
|
void InitWeaponHeart(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
|
|
if (SW_SHAREWARE) return;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_HEART)) ||
|
|
// pp->WpnAmmo[WPN_HEART] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[WPN_HEART])
|
|
{
|
|
psp = pp->Wpn[WPN_HEART] = pSpawnSprite(pp, ps_PresentHeart, PRI_MID, 160 + 10, HEART_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_HEART])
|
|
{
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_HEART);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
PlaySound(DIGI_HEARTBEAT, pp, v3df_follow|v3df_dontpan|v3df_doppler);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_HEART];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_HeartAttack;
|
|
psp->RetractState = ps_RetractHeart;
|
|
psp->PresentState = ps_PresentHeart;
|
|
psp->RestState = ps_HeartRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
void pHeartPresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < HEART_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = HEART_YOFF;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pHeartBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 12;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pHeartHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= HEART_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = HEART_YOFF + tileHeight(picnum);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pHeartRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_HeartHide))
|
|
return;
|
|
|
|
psp->bobpos.Y += synctics;
|
|
|
|
if (psp->bobpos.Y > HEART_YOFF)
|
|
{
|
|
psp->bobpos.Y = HEART_YOFF;
|
|
}
|
|
|
|
pHeartBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->PlayerP->KeyPressBits &= ~SB_FIRE;
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
pSetState(psp, psp->ActionState);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
}
|
|
|
|
void pHeartAction(PANEL_SPRITE* psp)
|
|
{
|
|
psp->bobpos.Y -= synctics;
|
|
|
|
if (psp->bobpos.Y < 200)
|
|
{
|
|
psp->bobpos.Y = 200;
|
|
}
|
|
|
|
pHeartBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
void pHeartActionBlood(PANEL_SPRITE* psp)
|
|
{
|
|
psp->bobpos.Y -= synctics;
|
|
|
|
if (psp->bobpos.Y < 200)
|
|
{
|
|
psp->bobpos.Y = 200;
|
|
}
|
|
|
|
psp->backupy();
|
|
psp->pos.Y = psp->bobpos.Y;
|
|
|
|
pHeartBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
SpawnHeartBlood(psp);
|
|
}
|
|
|
|
void InitHeartAttack(PLAYER* pp);
|
|
|
|
void pHeartAttack(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
// CTW MODIFICATION
|
|
//int InitHeartAttack(PLAYER* pp);
|
|
// CTW MODIFICATION END
|
|
|
|
PlaySound(DIGI_HEARTFIRE, pp, v3df_follow|v3df_dontpan);
|
|
if (RandomRange(1000) > 800)
|
|
PlayerSound(DIGI_JG9009, v3df_follow|v3df_dontpan,pp);
|
|
InitHeartAttack(psp->PlayerP);
|
|
}
|
|
|
|
void pHeartRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= HEART_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_HEART] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// HEART BLOOD
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pHeartBlood(PANEL_SPRITE*);
|
|
|
|
#define HEART_BLOOD_RATE 10
|
|
|
|
PANEL_STATE ps_HeartBlood[] =
|
|
{
|
|
{ID_HeartBlood0, HEART_BLOOD_RATE, pHeartBlood, &ps_HeartBlood[1], 0,0,0},
|
|
{ID_HeartBlood1, HEART_BLOOD_RATE, pHeartBlood, &ps_HeartBlood[2], 0,0,0},
|
|
{ID_HeartBlood2, HEART_BLOOD_RATE, pHeartBlood, &ps_HeartBlood[3], 0,0,0},
|
|
{ID_HeartBlood3, HEART_BLOOD_RATE, pHeartBlood, &ps_HeartBlood[4], 0,0,0},
|
|
{ID_HeartBlood4, HEART_BLOOD_RATE, pHeartBlood, &ps_HeartBlood[5], 0,0,0},
|
|
{ID_HeartBlood5, HEART_BLOOD_RATE, pHeartBlood, &ps_HeartBlood[6], 0,0,0},
|
|
{ID_HeartBlood5, HEART_BLOOD_RATE, pSuicide, &ps_HeartBlood[6], 0,0,0},
|
|
};
|
|
|
|
#define HEART_BLOOD_SMALL_RATE 7
|
|
|
|
PANEL_STATE ps_HeartBloodSmall[] =
|
|
{
|
|
{ID_HeartBlood0, HEART_BLOOD_SMALL_RATE, pHeartBlood, &ps_HeartBlood[1], 0,0,0},
|
|
{ID_HeartBlood1, HEART_BLOOD_SMALL_RATE, pHeartBlood, &ps_HeartBlood[2], 0,0,0},
|
|
{ID_HeartBlood2, HEART_BLOOD_SMALL_RATE, pHeartBlood, &ps_HeartBlood[3], 0,0,0},
|
|
{ID_HeartBlood3, HEART_BLOOD_SMALL_RATE, pHeartBlood, &ps_HeartBlood[4], 0,0,0},
|
|
{ID_HeartBlood4, HEART_BLOOD_SMALL_RATE, pHeartBlood, &ps_HeartBlood[5], 0,0,0},
|
|
{ID_HeartBlood5, HEART_BLOOD_SMALL_RATE, pHeartBlood, &ps_HeartBlood[6], 0,0,0},
|
|
{ID_HeartBlood5, HEART_BLOOD_SMALL_RATE, pSuicide, &ps_HeartBlood[6], 0,0,0},
|
|
};
|
|
|
|
void SpawnHeartBlood(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
PANEL_SPRITE* blood;
|
|
PANEL_SHRAP* hsp;
|
|
|
|
static PANEL_SHRAP HeartShrap[] =
|
|
{
|
|
{-10, -80, 2, -FIXED(1,32000), -FIXED(2,32000), -FIXED(5,32000), -FIXED(3,32000), {ps_HeartBlood, ps_HeartBlood}},
|
|
{0, -85, 0, -FIXED(3,32000), -FIXED(8,32000), -FIXED(3,32000), -FIXED(1,32000), {ps_HeartBlood, ps_HeartBlood}},
|
|
{10, -85, 2, -FIXED(1,32000), -FIXED(2,32000), FIXED(2,32000), FIXED(3,32000), {ps_HeartBlood, ps_HeartBlood}},
|
|
{25, -80, 2, -FIXED(1,32000), -FIXED(2,32000), FIXED(5,32000), FIXED(6,32000), {ps_HeartBlood, ps_HeartBlood}},
|
|
{0, 0, 0, 0, 0, 0, 0, {0, 0}},
|
|
};
|
|
|
|
for (hsp = HeartShrap; hsp->lo_jump_speed; hsp++)
|
|
{
|
|
if (hsp->skip == 2)
|
|
{
|
|
if (MoveSkip2 != 0)
|
|
continue;
|
|
}
|
|
|
|
// RIGHT side
|
|
blood = pSpawnSprite(pp, hsp->state[RANDOM_P2(2<<8)>>8], PRI_BACK, 0, 0);
|
|
blood->pos.X = psp->pos.X + hsp->xoff;
|
|
blood->opos.X = blood->pos.X;
|
|
blood->pos.Y = psp->pos.Y + hsp->yoff;
|
|
blood->opos.Y = blood->pos.Y;
|
|
blood->xspeed = hsp->lo_xspeed + (RandomRange((hsp->hi_xspeed - hsp->lo_xspeed)>>4) << 4);
|
|
blood->flags |= (PANF_WEAPON_SPRITE);
|
|
|
|
blood->scale = 20000 + RandomRange(50000 - 20000);
|
|
|
|
blood->jump_speed = hsp->lo_jump_speed + (RandomRange((hsp->hi_jump_speed + hsp->lo_jump_speed)>>4) << 4);
|
|
DoBeginPanelJump(blood);
|
|
}
|
|
}
|
|
|
|
void SpawnSmallHeartBlood(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
PANEL_SPRITE* blood;
|
|
PANEL_SHRAP* hsp;
|
|
|
|
static PANEL_SHRAP HeartShrap[] =
|
|
{
|
|
{-10, -80, 0, -FIXED(1,0), -FIXED(2,0), -FIXED(1,0), -FIXED(3,0), {ps_HeartBloodSmall, ps_HeartBloodSmall}},
|
|
{0, -85, 0, -FIXED(1,0), -FIXED(5,0), -FIXED(1,0), -FIXED(1,0), {ps_HeartBloodSmall, ps_HeartBloodSmall}},
|
|
{10, -85, 0, -FIXED(1,0), -FIXED(2,0), FIXED(1,0), FIXED(2,0), {ps_HeartBloodSmall, ps_HeartBloodSmall}},
|
|
{25, -80, 0, -FIXED(1,0), -FIXED(2,0), FIXED(3,0), FIXED(4,0), {ps_HeartBloodSmall, ps_HeartBloodSmall}},
|
|
{0, 0, 0, 0, 0, 0, 0, {0,0}},
|
|
};
|
|
|
|
PlaySound(DIGI_HEARTBEAT, pp, v3df_follow|v3df_dontpan|v3df_doppler);
|
|
|
|
for (hsp = HeartShrap; hsp->lo_jump_speed; hsp++)
|
|
{
|
|
// RIGHT side
|
|
blood = pSpawnSprite(pp, hsp->state[RANDOM_P2(2<<8)>>8], PRI_BACK, 0, 0);
|
|
blood->pos.X = psp->pos.X + hsp->xoff;
|
|
blood->opos.X = blood->pos.X;
|
|
blood->pos.Y = psp->pos.Y + hsp->yoff;
|
|
blood->opos.Y = blood->pos.Y;
|
|
blood->xspeed = hsp->lo_xspeed + (RandomRange((hsp->hi_xspeed - hsp->lo_xspeed)>>4) << 4);
|
|
blood->flags |= (PANF_WEAPON_SPRITE);
|
|
|
|
blood->scale = 10000 + RandomRange(30000 - 10000);
|
|
|
|
blood->jump_speed = hsp->lo_jump_speed + (RandomRange((hsp->hi_jump_speed + hsp->lo_jump_speed)>>4) << 4);
|
|
DoBeginPanelJump(blood);
|
|
}
|
|
}
|
|
|
|
void pHeartBlood(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_JUMPING))
|
|
{
|
|
DoPanelJump(psp);
|
|
}
|
|
else if (psp->flags & (PANF_FALLING))
|
|
{
|
|
DoPanelFall(psp);
|
|
}
|
|
|
|
psp->backupx();
|
|
psp->pos.X += psp->xspeed * (1. / FRACUNIT);
|
|
|
|
if (psp->pos.X > 320 || psp->pos.X < 0 || psp->pos.Y > 200)
|
|
{
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
}
|
|
|
|
int DoBeginPanelJump(PANEL_SPRITE* psp)
|
|
{
|
|
#define PANEL_JUMP_GRAVITY FIXED(0,8000)
|
|
|
|
psp->flags |= (PANF_JUMPING);
|
|
psp->flags &= ~(PANF_FALLING);
|
|
|
|
// set up individual actor jump gravity
|
|
psp->jump_grav = PANEL_JUMP_GRAVITY;
|
|
|
|
DoPanelJump(psp);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoPanelJump(PANEL_SPRITE* psp)
|
|
{
|
|
// adjust jump speed by gravity - if jump speed greater than 0 player
|
|
// have started falling
|
|
if ((psp->jump_speed += psp->jump_grav) > 0)
|
|
{
|
|
// Start falling
|
|
DoBeginPanelFall(psp);
|
|
return 0;
|
|
}
|
|
|
|
// adjust height by jump speed
|
|
psp->backupy();
|
|
psp->pos.Y += psp->jump_speed * synctics * (1. / FRACUNIT);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int DoBeginPanelFall(PANEL_SPRITE* psp)
|
|
{
|
|
psp->flags |= (PANF_FALLING);
|
|
psp->flags &= ~(PANF_JUMPING);
|
|
|
|
psp->jump_grav = PANEL_JUMP_GRAVITY;
|
|
|
|
DoPanelFall(psp);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int DoPanelFall(PANEL_SPRITE* psp)
|
|
{
|
|
// adjust jump speed by gravity
|
|
psp->jump_speed += psp->jump_grav;
|
|
|
|
// adjust player height by jump speed
|
|
psp->backupy();
|
|
psp->pos.Y += psp->jump_speed * synctics * (1. / FRACUNIT);
|
|
|
|
return 0;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GRENADE
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pGrenadePresent(PANEL_SPRITE* psp);
|
|
void pGrenadeRetract(PANEL_SPRITE* psp);
|
|
void pGrenadeAction(PANEL_SPRITE* psp);
|
|
void pGrenadeRest(PANEL_SPRITE* psp);
|
|
void pGrenadeFire(PANEL_SPRITE* psp);
|
|
void pGrenadeHide(PANEL_SPRITE* psp);
|
|
void pGrenadeUnHide(PANEL_SPRITE* psp);
|
|
void pGrenadeRecoilDown(PANEL_SPRITE* psp);
|
|
void pGrenadeRecoilUp(PANEL_SPRITE* psp);
|
|
void pGrenadeBobSetup(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_GrenadeRecoil[];
|
|
|
|
#define Grenade_REST_RATE 24
|
|
#define Grenade_ACTION_RATE 6
|
|
|
|
PANEL_STATE ps_PresentGrenade[] =
|
|
{
|
|
{ID_GrenadePresent0, Grenade_REST_RATE, pGrenadePresent, &ps_PresentGrenade[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_GrenadeRest[] =
|
|
{
|
|
{ID_GrenadePresent0, Grenade_REST_RATE, pGrenadeRest, &ps_GrenadeRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_GrenadeHide[] =
|
|
{
|
|
{ID_GrenadePresent0, Grenade_REST_RATE, pGrenadeHide, &ps_GrenadeHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_GrenadeFire[] =
|
|
{
|
|
{ID_GrenadeFire0, Grenade_ACTION_RATE, pGrenadeAction, &ps_GrenadeFire[1], psf_ShadeHalf, 0,0},
|
|
{ID_GrenadeFire1, Grenade_ACTION_RATE, pGrenadeAction, &ps_GrenadeFire[2], psf_ShadeNone, 0,0},
|
|
{ID_GrenadeFire2, Grenade_ACTION_RATE, pGrenadeAction, &ps_GrenadeFire[3], psf_ShadeNone, 0,0},
|
|
|
|
{ID_GrenadePresent0, 0, pGrenadeFire, &ps_GrenadeFire[4], psf_QuickCall, 0,0},
|
|
{ID_GrenadePresent0, 3, pGrenadeAction, &ps_GrenadeRecoil[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_GrenadeRecoil[] =
|
|
{
|
|
// recoil
|
|
{ID_GrenadePresent0, Grenade_REST_RATE, pGrenadeRecoilDown, &ps_GrenadeRecoil[0], 0,0,0},
|
|
{ID_GrenadePresent0, Grenade_REST_RATE, pGrenadeRecoilUp, &ps_GrenadeRecoil[1], 0,0,0},
|
|
// reload
|
|
{ID_GrenadeReload0, Grenade_REST_RATE/2, pNullAnimator, &ps_GrenadeRecoil[3], 0,0,0},
|
|
{ID_GrenadeReload1, Grenade_REST_RATE/2, pNullAnimator, &ps_GrenadeRecoil[4], 0,0,0},
|
|
// ready to fire again
|
|
{ID_GrenadePresent0, 3, pNullAnimator, &ps_GrenadeRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RetractGrenade[] =
|
|
{
|
|
{ID_GrenadePresent0, Grenade_REST_RATE, pGrenadeRetract, &ps_RetractGrenade[0], 0,0,0}
|
|
};
|
|
|
|
#define GRENADE_YOFF 200
|
|
#define GRENADE_XOFF (160+20)
|
|
|
|
void pGrenadeSetRecoil(PANEL_SPRITE* psp)
|
|
{
|
|
psp->vel = 900;
|
|
psp->ang = NORM_ANGLE(-256);
|
|
}
|
|
|
|
void pGrenadePresentSetup(PANEL_SPRITE* psp)
|
|
{
|
|
psp->rotate_ang = 1800;
|
|
psp->backupcoords();
|
|
psp->pos.Y += 34;
|
|
psp->pos.X -= 45;
|
|
psp->ang = 256 + 128;
|
|
psp->vel = 680;
|
|
}
|
|
|
|
void InitWeaponGrenade(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_GRENADE)) ||
|
|
// pp->WpnAmmo[WPN_GRENADE] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[WPN_GRENADE])
|
|
{
|
|
psp = pp->Wpn[WPN_GRENADE] = pSpawnSprite(pp, ps_PresentGrenade, PRI_MID, GRENADE_XOFF, GRENADE_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_GRENADE])
|
|
{
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_GRENADE);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_GRENADE];
|
|
psp = pp->CurWpn = pp->Wpn[WPN_GRENADE];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_GrenadeFire;
|
|
psp->RetractState = ps_RetractGrenade;
|
|
psp->PresentState = ps_PresentGrenade;
|
|
psp->RestState = ps_GrenadeRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
pGrenadePresentSetup(psp);
|
|
|
|
PlaySound(DIGI_GRENADE_UP, pp, v3df_follow);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
void pGrenadeRecoilDown(PANEL_SPRITE* psp)
|
|
{
|
|
// int picnum = psp->picndx;
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel -= 24 * synctics;
|
|
|
|
// if (psp->y >= GRENADE_YOFF + tileHeight(picnum))
|
|
if (psp->vel < 400)
|
|
{
|
|
// psp->ox = psp->y = GRENADE_YOFF + tileHeight(picnum);
|
|
|
|
psp->vel = 400;
|
|
psp->ang = NORM_ANGLE(psp->ang + 1024);
|
|
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pGrenadeRecoilUp(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel += 15 * synctics;
|
|
|
|
if (psp->pos.Y < GRENADE_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = GRENADE_YOFF;
|
|
psp->opos.X = psp->pos.X = GRENADE_XOFF;
|
|
|
|
pGrenadeSetRecoil(psp);
|
|
|
|
pStatePlusOne(psp);
|
|
psp->flags &= ~(PANF_BOB);
|
|
}
|
|
}
|
|
|
|
void pGrenadePresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += pspCosVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->rotate_ang = NORM_ANGLE(psp->rotate_ang + (6 * synctics));
|
|
|
|
if (psp->rotate_ang < 1024)
|
|
psp->rotate_ang = 0;
|
|
|
|
if (psp->pos.Y < GRENADE_YOFF)
|
|
{
|
|
pGrenadeSetRecoil(psp);
|
|
psp->opos.X = psp->pos.X = GRENADE_XOFF;
|
|
psp->opos.Y = psp->pos.Y = GRENADE_YOFF;
|
|
psp->rotate_ang = 0;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pGrenadeBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 12;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pGrenadeHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= GRENADE_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = GRENADE_YOFF + tileHeight(picnum);
|
|
psp->opos.X = psp->pos.X = GRENADE_XOFF;
|
|
|
|
pGrenadePresentSetup(psp);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pGrenadeRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_GrenadeHide))
|
|
return;
|
|
|
|
pGrenadeBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
pSetState(psp, psp->ActionState);
|
|
}
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pGrenadeAction(PANEL_SPRITE* psp)
|
|
{
|
|
pGrenadeBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pGrenadeFire(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnVis(psp->PlayerP->actor, nullptr, -1, -1, -1, 32);
|
|
InitGrenade(psp->PlayerP);
|
|
}
|
|
|
|
void pGrenadeRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= GRENADE_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_GRENADE] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MINE
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pMinePresent(PANEL_SPRITE* psp);
|
|
void pMineRetract(PANEL_SPRITE* psp);
|
|
void pMineAction(PANEL_SPRITE* psp);
|
|
void pMineRest(PANEL_SPRITE* psp);
|
|
void pMineThrow(PANEL_SPRITE* psp);
|
|
void pMineLower(PANEL_SPRITE* psp);
|
|
void pMineRaise(PANEL_SPRITE* psp);
|
|
void pMineHide(PANEL_SPRITE* psp);
|
|
void pMineUnHide(PANEL_SPRITE* psp);
|
|
void pMineBobSetup(PANEL_SPRITE* psp);
|
|
void pMineUpSound(PANEL_SPRITE* psp);
|
|
|
|
#define Mine_REST_RATE 24
|
|
#define Mine_ACTION_RATE 6
|
|
|
|
PANEL_STATE ps_PresentMine[] =
|
|
{
|
|
{ID_MinePresent0, Mine_REST_RATE, pMinePresent, &ps_PresentMine[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MineRest[] =
|
|
{
|
|
{ID_MinePresent0, 36, pMineRest, &ps_MineRest[1], 0,0,0},
|
|
{ID_MinePresent0, 0, pMineUpSound, &ps_MineRest[2], psf_QuickCall, 0,0},
|
|
{ID_MinePresent1, Mine_REST_RATE, pMineRest, &ps_MineRest[2], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_MineHide[] =
|
|
{
|
|
{ID_MinePresent0, Mine_REST_RATE, pMineHide, &ps_MineHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_MineThrow[] =
|
|
{
|
|
{ID_MineThrow0, 3, pNullAnimator, &ps_MineThrow[1], 0,0,0},
|
|
{ID_MineThrow0, Mine_ACTION_RATE, pMineThrow, &ps_MineThrow[2],psf_QuickCall, 0,0},
|
|
{ID_MineThrow0, Mine_ACTION_RATE, pMineLower, &ps_MineThrow[2], 0,0,0},
|
|
{ID_MineThrow0, Mine_ACTION_RATE*5, pNullAnimator, &ps_MineThrow[4], 0,0,0},
|
|
{ID_MinePresent0, Mine_ACTION_RATE, pMineRaise, &ps_MineThrow[4], 0,0,0},
|
|
{ID_MinePresent0, Mine_ACTION_RATE, pNullAnimator, &ps_MineThrow[6], 0,0,0},
|
|
{ID_MinePresent0, 3, pMineAction, &ps_MineRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_RetractMine[] =
|
|
{
|
|
{ID_MinePresent0, Mine_REST_RATE, pMineRetract, &ps_RetractMine[0], 0,0,0}
|
|
};
|
|
|
|
#define MINE_YOFF 200
|
|
//#define MINE_XOFF (160+20)
|
|
#define MINE_XOFF (160+50)
|
|
|
|
void InitWeaponMine(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (pp->WpnAmmo[WPN_MINE] <= 0)
|
|
PutStringInfo(pp,"Out of Sticky Bombs!");
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_MINE)) ||
|
|
pp->WpnAmmo[WPN_MINE] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
if (!pp->Wpn[WPN_MINE])
|
|
{
|
|
psp = pp->Wpn[WPN_MINE] = pSpawnSprite(pp, ps_PresentMine, PRI_MID, MINE_XOFF, MINE_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_MINE])
|
|
{
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_MINE);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_MINE];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_MineThrow;
|
|
psp->RetractState = ps_RetractMine;
|
|
psp->PresentState = ps_PresentMine;
|
|
psp->RestState = ps_MineRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
PlaySound(DIGI_PULL, pp, v3df_follow|v3df_dontpan);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
void pMineUpSound(PANEL_SPRITE* psp)
|
|
{
|
|
PLAYER* pp = psp->PlayerP;
|
|
|
|
PlaySound(DIGI_MINE_UP, pp, v3df_follow);
|
|
}
|
|
|
|
void pMineLower(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupy();
|
|
psp->pos.Y += 4 * synctics;
|
|
|
|
if (psp->pos.Y > MINE_YOFF + tileHeight(psp->picndx))
|
|
{
|
|
if (!WeaponOK(psp->PlayerP))
|
|
return;
|
|
psp->opos.Y = psp->pos.Y = MINE_YOFF + tileHeight(psp->picndx);
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pMineRaise(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupy();
|
|
psp->pos.Y -= 4 * synctics;
|
|
|
|
if (psp->pos.Y < MINE_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = MINE_YOFF;
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pMinePresent(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < MINE_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = MINE_YOFF;
|
|
psp->rotate_ang = 0;
|
|
psp->backupboby();
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
void pMineBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = 12;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pMineHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= MINE_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = MINE_YOFF + tileHeight(picnum);
|
|
psp->opos.X = psp->pos.X = MINE_XOFF;
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pMineRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_MineHide))
|
|
return;
|
|
|
|
pMineBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
// if (!WeaponOK(psp->PlayerP))
|
|
// return;
|
|
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
pSetState(psp, psp->ActionState);
|
|
}
|
|
}
|
|
else
|
|
WeaponOK(psp->PlayerP);
|
|
}
|
|
|
|
void pMineAction(PANEL_SPRITE* psp)
|
|
{
|
|
pMineBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pMineThrow(PANEL_SPRITE* psp)
|
|
{
|
|
InitMine(psp->PlayerP);
|
|
}
|
|
|
|
void pMineRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= MINE_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_MINE] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CHOP STICKS
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void pChopsUp(PANEL_SPRITE* psp);
|
|
void pChopsDown(PANEL_SPRITE* psp);
|
|
void pChopsDownSlow(PANEL_SPRITE* psp);
|
|
void pChopsWait(PANEL_SPRITE* psp);
|
|
void pChopsShake(PANEL_SPRITE* psp);
|
|
void pChopsClick(PANEL_SPRITE* psp);
|
|
|
|
void pChopsRetract(PANEL_SPRITE* psp);
|
|
|
|
#define Chops_REST_RATE 24
|
|
#define Chops_ACTION_RATE 6
|
|
|
|
#define ID_ChopsRest 2000
|
|
#define ID_ChopsOpen 2001
|
|
#define ID_ChopsClose 2002
|
|
|
|
PANEL_STATE ps_ChopsAttack1[] =
|
|
{
|
|
{ID_ChopsRest, Chops_REST_RATE*3, pNullAnimator, &ps_ChopsAttack1[1], 0,0,0},
|
|
{ID_ChopsRest, Chops_REST_RATE, pChopsUp, &ps_ChopsAttack1[1], 0,0,0},
|
|
{ID_ChopsOpen, Chops_REST_RATE/3, pNullAnimator, &ps_ChopsAttack1[3], 0,0,0},
|
|
{ID_ChopsClose, 0, pChopsClick, &ps_ChopsAttack1[4], psf_QuickCall, 0,0},
|
|
{ID_ChopsClose, Chops_REST_RATE/3, pNullAnimator, &ps_ChopsAttack1[5], 0,0,0},
|
|
{ID_ChopsClose, Chops_REST_RATE, pChopsDown, &ps_ChopsAttack1[5], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_ChopsAttack2[] =
|
|
{
|
|
{ID_ChopsOpen, Chops_REST_RATE*3, pNullAnimator, &ps_ChopsAttack2[1], 0,0,0},
|
|
{ID_ChopsOpen, Chops_REST_RATE, pChopsUp, &ps_ChopsAttack2[1], 0,0,0},
|
|
{ID_ChopsOpen, 0, pChopsClick, &ps_ChopsAttack2[3], psf_QuickCall, 0,0},
|
|
{ID_ChopsOpen, 8, pNullAnimator, &ps_ChopsAttack2[4], 0,0,0},
|
|
{ID_ChopsRest, Chops_REST_RATE, pNullAnimator, &ps_ChopsAttack2[5], 0,0,0},
|
|
{ID_ChopsRest, Chops_REST_RATE, pChopsDown, &ps_ChopsAttack2[5], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_ChopsAttack3[] =
|
|
{
|
|
{ID_ChopsOpen, Chops_REST_RATE*3, pNullAnimator, &ps_ChopsAttack3[1], 0,0,0},
|
|
{ID_ChopsOpen, Chops_REST_RATE, pChopsUp, &ps_ChopsAttack3[1], 0,0,0},
|
|
|
|
{ID_ChopsRest, 0, pNullAnimator, &ps_ChopsAttack3[3], 0,0,0},
|
|
{ID_ChopsRest, 0, pChopsClick, &ps_ChopsAttack3[4], psf_QuickCall, 0,0},
|
|
|
|
{ID_ChopsRest, Chops_REST_RATE, pNullAnimator, &ps_ChopsAttack3[5], 0,0,0},
|
|
{ID_ChopsRest, 24, pNullAnimator, &ps_ChopsAttack3[6], 0,0,0},
|
|
{ID_ChopsOpen, 16, pNullAnimator, &ps_ChopsAttack3[7], 0,0,0},
|
|
|
|
{ID_ChopsRest, 0, pChopsClick, &ps_ChopsAttack3[8], psf_QuickCall, 0,0},
|
|
{ID_ChopsRest, 16, pNullAnimator, &ps_ChopsAttack3[9], 0,0,0},
|
|
{ID_ChopsOpen, 16, pNullAnimator, &ps_ChopsAttack3[10], 0,0,0},
|
|
|
|
{ID_ChopsOpen, 8, pChopsDownSlow, &ps_ChopsAttack3[11], 0,0,0},
|
|
{ID_ChopsRest, 10, pChopsDownSlow, &ps_ChopsAttack3[12], 0,0,0},
|
|
{ID_ChopsRest, 0, pChopsClick, &ps_ChopsAttack3[13], psf_QuickCall, 0,0},
|
|
{ID_ChopsRest, 10, pChopsDownSlow, &ps_ChopsAttack3[14], 0,0,0},
|
|
{ID_ChopsOpen, 10, pChopsDownSlow, &ps_ChopsAttack3[11], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_ChopsAttack4[] =
|
|
{
|
|
{ID_ChopsOpen, Chops_REST_RATE*3, pNullAnimator, &ps_ChopsAttack4[1], 0,0,0},
|
|
{ID_ChopsOpen, Chops_REST_RATE, pChopsUp, &ps_ChopsAttack4[1], 0,0,0},
|
|
|
|
{ID_ChopsOpen, 0, pChopsClick, &ps_ChopsAttack4[3], psf_QuickCall, 0,0},
|
|
{ID_ChopsOpen, 8, pNullAnimator, &ps_ChopsAttack4[4], 0,0,0},
|
|
|
|
{ID_ChopsRest, Chops_REST_RATE, pNullAnimator, &ps_ChopsAttack4[5], 0,0,0},
|
|
{ID_ChopsRest, Chops_REST_RATE*4, pChopsShake, &ps_ChopsAttack4[6], 0,0,0},
|
|
{ID_ChopsRest, Chops_REST_RATE, pChopsDown, &ps_ChopsAttack4[6], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE* psp_ChopsAttack[] = {ps_ChopsAttack1, ps_ChopsAttack2, ps_ChopsAttack3, ps_ChopsAttack4};
|
|
|
|
PANEL_STATE ps_ChopsWait[] =
|
|
{
|
|
{ID_ChopsRest, Chops_REST_RATE, pChopsWait, &ps_ChopsWait[0], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_ChopsRetract[] =
|
|
{
|
|
{ID_ChopsRest, Chops_REST_RATE, pChopsRetract, &ps_ChopsRetract[0], 0,0,0}
|
|
};
|
|
|
|
#define CHOPS_YOFF 200
|
|
#define CHOPS_XOFF (160+20)
|
|
|
|
void InitChops(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
|
|
if (!pp->Chops)
|
|
{
|
|
psp = pp->Chops = pSpawnSprite(pp, ps_ChopsAttack1, PRI_MID, CHOPS_XOFF, CHOPS_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->Chops;
|
|
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_ChopsAttack1;
|
|
psp->PresentState = ps_ChopsAttack1;
|
|
psp->RetractState = ps_ChopsRetract;
|
|
psp->RestState = ps_ChopsAttack1;
|
|
|
|
PlaySound(DIGI_BUZZZ, psp->PlayerP,v3df_none);
|
|
|
|
if (RandomRange(1000) > 750)
|
|
PlayerSound(DIGI_MRFLY,v3df_follow|v3df_dontpan,psp->PlayerP);
|
|
|
|
}
|
|
|
|
void pChopsClick(PANEL_SPRITE* psp)
|
|
{
|
|
int16_t rnd_rng;
|
|
PlaySound(DIGI_CHOP_CLICK,psp->PlayerP,v3df_none);
|
|
|
|
rnd_rng = RandomRange(1000);
|
|
if (rnd_rng > 950)
|
|
PlayerSound(DIGI_SEARCHWALL,v3df_follow|v3df_dontpan,psp->PlayerP);
|
|
else if (rnd_rng > 900)
|
|
PlayerSound(DIGI_EVADEFOREVER,v3df_follow|v3df_dontpan,psp->PlayerP);
|
|
else if (rnd_rng > 800)
|
|
PlayerSound(DIGI_SHISEISI,v3df_follow|v3df_dontpan,psp->PlayerP);
|
|
}
|
|
|
|
void pChopsUp(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < CHOPS_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = CHOPS_YOFF;
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
void pChopsDown(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y > CHOPS_YOFF+110)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = CHOPS_YOFF+110;
|
|
pSetState(psp, ps_ChopsWait);
|
|
}
|
|
}
|
|
|
|
void pChopsDownSlow(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupy();
|
|
psp->pos.Y += 1 * synctics;
|
|
|
|
if (psp->pos.Y > CHOPS_YOFF+110)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = CHOPS_YOFF+110;
|
|
pSetState(psp, ps_ChopsWait);
|
|
}
|
|
}
|
|
|
|
void pChopsShake(PANEL_SPRITE* psp)
|
|
{
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X += (RANDOM_P2(4<<8) * (1. / 256.)) - 2;
|
|
psp->pos.Y += (RANDOM_P2(4<<8) * (1. / 256.)) - 2;
|
|
|
|
if (psp->pos.Y < CHOPS_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = CHOPS_YOFF;
|
|
}
|
|
}
|
|
|
|
void pChopsWait(PANEL_SPRITE* psp)
|
|
{
|
|
//if (!paused && RANDOM_P2(1024) < 10)
|
|
if (RANDOM_P2(1024) < 10)
|
|
{
|
|
// random x position
|
|
// do a random attack here
|
|
psp->opos.X = psp->pos.X = CHOPS_XOFF + (RANDOM_P2(128) - 64);
|
|
|
|
PlaySound(DIGI_BUZZZ,psp->PlayerP,v3df_none);
|
|
pSetState(psp, psp_ChopsAttack[RandomRange(SIZ(psp_ChopsAttack))]);
|
|
}
|
|
}
|
|
|
|
void ChopsSetRetract(PLAYER* pp)
|
|
{
|
|
if (pp == nullptr || pp->Chops == nullptr)
|
|
return;
|
|
|
|
pSetState(pp->Chops, pp->Chops->RetractState);
|
|
}
|
|
|
|
void pChopsRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 6 * synctics;
|
|
|
|
if (psp->pos.Y >= CHOPS_YOFF + tileHeight(picnum))
|
|
{
|
|
if (RandomRange(1000) > 800)
|
|
PlayerSound(DIGI_GETTINGSTIFF,v3df_follow|v3df_dontpan,psp->PlayerP);
|
|
psp->PlayerP->Chops = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// FIST
|
|
//
|
|
// KungFu Move Numbers: (Used to calculate damages, etc....
|
|
// 0: Normal Punch
|
|
// 1: Palm Heel Strike
|
|
// 2: Kick
|
|
// 3: Block
|
|
// 4: FISTS OF FURY - (Flury of fast punches) DUCK, then BLOCK and ATTACK
|
|
// 5: CHI STRIKE - (Devastating Palm Heel) DUCK 2 secs, then ATTACK
|
|
// 6: DEATH TOUCH - (Make enemy bleed and does alot of damage) JUMP, DUCK, BLOCK, ATTACK
|
|
// 7: FIREBALL - BLOCK 4 secs, then ATTACK
|
|
// 8: HAMMER KICK - Run forward while holding BLOCK, JUMP, ATTACK
|
|
// 9: CYCLONE KICK - (Spin 5 circles with leg out to hit multiples)
|
|
// 180 KEY, BLOCK, JUMP, ATTACK
|
|
// 10: DESPERATION - Finish Move (Cling to enemy like sticky bomb and beat the crap out of him)
|
|
// Player is invincible until he unattaches
|
|
// LOW HEALTH < 20, TOUCH ENEMY, BLOCK then ATTACK
|
|
// 11: SUPER NOVA - Become a human nuclear bomb!
|
|
// SUPER SECRET MOVE
|
|
// >150 Health + >50 Armor + Shadow Spell ACTIVE, DUCK + BLOCK 3 secs, JUMP, ATTACK
|
|
// Player will die, so it's only useful if you get more than 1 frag in the deal.
|
|
// 12: TOTAL INVISIBILITY - Worse than shadow spell, you go 100% invisible for 10 secs!
|
|
// SUPER SECRET MOVE
|
|
// >180 Health + Shadow Spell ACTIVE, BLOCK 4 secs, JUMP, ATTACK
|
|
// 13: FREEZE - Turn all enemies in your view to ice for 10 secs
|
|
// SUPER SECRET MOVE
|
|
// 100 Armor + Shadow Spell ACTIVE, BLOCK 5 secs, DUCK, ATTACK
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
static short FistAngTable[] =
|
|
{
|
|
82,
|
|
168,
|
|
256+64
|
|
};
|
|
|
|
short FistAng = 0;
|
|
|
|
void FistBlur(PANEL_SPRITE* psp)
|
|
{
|
|
psp->kill_tics -= synctics;
|
|
if (psp->kill_tics <= 0)
|
|
{
|
|
pKillSprite(psp);
|
|
return;
|
|
}
|
|
else if (psp->kill_tics <= 6)
|
|
{
|
|
psp->flags |= (PANF_TRANS_FLIP);
|
|
}
|
|
|
|
psp->shade += 10;
|
|
// change sprites priority
|
|
REMOVE(psp);
|
|
psp->priority--;
|
|
InsertPanelSprite(psp->PlayerP, psp);
|
|
}
|
|
|
|
void SpawnFistBlur(PANEL_SPRITE* psp)
|
|
{
|
|
if (cl_nomeleeblur) return;
|
|
|
|
PANEL_SPRITE* nsp;
|
|
//PICITEMp pip;
|
|
|
|
if (psp->PlayerP->FistAng > 200)
|
|
return;
|
|
|
|
nsp = pSpawnSprite(psp->PlayerP, nullptr, PRI_BACK, psp->pos.X, psp->pos.Y);
|
|
|
|
nsp->flags |= (PANF_WEAPON_SPRITE);
|
|
nsp->ang = psp->ang;
|
|
nsp->vel = psp->vel;
|
|
nsp->PanelSpriteFunc = FistBlur;
|
|
nsp->kill_tics = 9;
|
|
nsp->shade = psp->shade + 10;
|
|
|
|
nsp->picndx = -1;
|
|
nsp->picnum = psp->picndx;
|
|
|
|
if ((psp->State->flags & psf_Xflip))
|
|
nsp->flags |= (PANF_XFLIP);
|
|
|
|
nsp->rotate_ang = psp->rotate_ang;
|
|
nsp->scale = psp->scale;
|
|
|
|
nsp->flags |= (PANF_TRANSLUCENT);
|
|
}
|
|
|
|
void pFistPresent(PANEL_SPRITE* psp);
|
|
void pFistRetract(PANEL_SPRITE* psp);
|
|
void pFistAction(PANEL_SPRITE* psp);
|
|
void pFistRest(PANEL_SPRITE* psp);
|
|
void pFistAttack(PANEL_SPRITE* psp);
|
|
void pFistSlide(PANEL_SPRITE* psp);
|
|
void pFistSlideDown(PANEL_SPRITE* psp);
|
|
void pFistHide(PANEL_SPRITE* psp);
|
|
void pFistUnHide(PANEL_SPRITE* psp);
|
|
|
|
void pFistSlideR(PANEL_SPRITE* psp);
|
|
void pFistSlideDownR(PANEL_SPRITE* psp);
|
|
void pFistBlock(PANEL_SPRITE* psp);
|
|
|
|
extern PANEL_STATE ps_FistSwing[];
|
|
extern PANEL_STATE ps_ReloadFist[];
|
|
|
|
#define Fist_BEAT_RATE 16
|
|
#define Fist_ACTION_RATE 5
|
|
|
|
PANEL_STATE ps_PresentFist[] =
|
|
{
|
|
{ID_FistPresent0, Fist_BEAT_RATE, pFistPresent, &ps_PresentFist[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_FistRest[] =
|
|
{
|
|
{ID_FistPresent0, Fist_BEAT_RATE, pFistRest, &ps_FistRest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_FistHide[] =
|
|
{
|
|
{ID_FistPresent0, Fist_BEAT_RATE, pFistHide, &ps_FistHide[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_PresentFist2[] =
|
|
{
|
|
{ID_Fist2Present0, Fist_BEAT_RATE, pFistPresent, &ps_PresentFist2[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_Fist2Rest[] =
|
|
{
|
|
{ID_Fist2Present0, Fist_BEAT_RATE, pFistRest, &ps_Fist2Rest[0], 0,0,0}
|
|
};
|
|
|
|
PANEL_STATE ps_Fist2Hide[] =
|
|
{
|
|
{ID_Fist2Present0, Fist_BEAT_RATE, pFistHide, &ps_Fist2Hide[0], 0,0,0}
|
|
};
|
|
|
|
|
|
#define FIST_PAUSE_TICS 6
|
|
#define FIST_SLIDE_TICS 6
|
|
#define FIST_MID_SLIDE_TICS 16
|
|
|
|
PANEL_STATE ps_FistSwing[] =
|
|
{
|
|
{ID_FistSwing0, FIST_PAUSE_TICS, pNullAnimator, &ps_FistSwing[1], 0,0,0},
|
|
{ID_FistSwing1, FIST_SLIDE_TICS, /* start slide */ pNullAnimator, &ps_FistSwing[2], 0,0,0},
|
|
{ID_FistSwing2, 0, /* damage */ pFistAttack, &ps_FistSwing[3], psf_QuickCall, 0,0},
|
|
{ID_FistSwing2, FIST_MID_SLIDE_TICS, /* mid slide */ pFistSlideDown, &ps_FistSwing[4], 0,0,0},
|
|
|
|
{ID_FistSwing2, 2, /* end slide */ pFistSlideDown, &ps_FistSwing[4], 0,0,0},
|
|
|
|
{ID_FistSwing1, FIST_SLIDE_TICS, /* start slide */ pFistSlideR, &ps_FistSwing[6], psf_Xflip, 0,0},
|
|
{ID_FistSwing2, 0, /* damage */ pFistAttack, &ps_FistSwing[7], psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_FistSwing2, FIST_MID_SLIDE_TICS, /* mid slide */ pFistSlideDownR, &ps_FistSwing[8], psf_Xflip, 0,0},
|
|
|
|
{ID_FistSwing2, 2, /* end slide */ pFistSlideDownR, &ps_FistSwing[8], psf_Xflip, 0,0},
|
|
{ID_FistSwing2, 2, /* end slide */ pNullAnimator, &ps_FistSwing[1], psf_Xflip, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_Fist2Swing[] =
|
|
{
|
|
{4058, FIST_PAUSE_TICS, pNullAnimator, &ps_Fist2Swing[1], 0,0,0},
|
|
{4058, FIST_SLIDE_TICS, /* start slide */ pNullAnimator, &ps_Fist2Swing[2], 0,0,0},
|
|
{4058, 0, /* damage */ pFistBlock, &ps_Fist2Swing[0], psf_QuickCall, 0,0},
|
|
{4058, FIST_MID_SLIDE_TICS+5, /* mid slide */ pFistSlideDown, &ps_Fist2Swing[4], 0,0,0},
|
|
|
|
{4058, 2, /* end slide */ pFistSlideDown, &ps_Fist2Swing[4], 0,0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_Fist3Swing[] =
|
|
{
|
|
{ID_Fist3Swing0, FIST_PAUSE_TICS+25, pNullAnimator, &ps_Fist3Swing[1], 0,0,0},
|
|
{ID_Fist3Swing1, 0, /* damage */ pFistAttack, &ps_Fist3Swing[2], psf_QuickCall, 0,0},
|
|
{ID_Fist3Swing2, FIST_PAUSE_TICS+10, pNullAnimator, &ps_Fist3Swing[3], 0,0,0},
|
|
{ID_Fist3Swing2, FIST_MID_SLIDE_TICS+3, /* mid slide */ pFistSlideDown, &ps_Fist3Swing[4], 0,0,0},
|
|
|
|
{ID_Fist3Swing2, 8, /* end slide */ pFistSlideDown, &ps_Fist3Swing[4], 0,0,0},
|
|
|
|
{ID_Fist3Swing1, FIST_SLIDE_TICS+20, /* start slide */ pFistSlideR, &ps_Fist3Swing[6], psf_Xflip, 0,0},
|
|
{ID_Fist3Swing2, 0, /* damage */ pFistAttack, &ps_Fist3Swing[7], psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_Fist3Swing2, FIST_MID_SLIDE_TICS+3, /* mid slide */ pFistSlideDownR, &ps_Fist3Swing[8], psf_Xflip, 0,0},
|
|
|
|
{ID_Fist3Swing2, 8, /* end slide */ pFistSlideDownR, &ps_Fist3Swing[8], psf_Xflip, 0,0},
|
|
{ID_Fist3Swing2, 8, /* end slide */ pNullAnimator, &ps_Fist3Swing[1], psf_Xflip, 0,0},
|
|
};
|
|
|
|
#define KICK_PAUSE_TICS 40
|
|
#define KICK_SLIDE_TICS 30
|
|
#define KICK_MID_SLIDE_TICS 20
|
|
|
|
PANEL_STATE ps_Kick[] =
|
|
{
|
|
{ID_Kick0, KICK_PAUSE_TICS, pNullAnimator, &ps_Kick[1], 0,0,0},
|
|
{ID_Kick1, 0, /* damage */ pFistAttack, &ps_Kick[2], psf_QuickCall, 0,0},
|
|
{ID_Kick1, KICK_SLIDE_TICS, /* start slide */ pNullAnimator, &ps_Kick[3], 0,0,0},
|
|
{ID_Kick1, KICK_MID_SLIDE_TICS, /* mid slide */ pFistSlideDown, &ps_Kick[4], 0,0,0},
|
|
|
|
{ID_Kick1, 30, /* end slide */ pFistSlideDown, &ps_Kick[4], 0,0,0},
|
|
|
|
{ID_Kick0, KICK_SLIDE_TICS, /* start slide */ pNullAnimator, &ps_Kick[6], psf_Xflip, 0,0},
|
|
{ID_Kick1, 0, /* damage */ pFistAttack, &ps_Kick[7], psf_QuickCall|psf_Xflip, 0,0},
|
|
{ID_Kick1, KICK_MID_SLIDE_TICS,/* mid slide */ pFistSlideDownR, &ps_Kick[8], psf_Xflip, 0, 0},
|
|
|
|
{ID_Kick1, 30, /* end slide */ pFistSlideDownR, &ps_Kick[8], psf_Xflip, 0,0},
|
|
{ID_Kick1, 30, /* end slide */ pNullAnimator, &ps_Kick[1], psf_Xflip, 0,0},
|
|
};
|
|
|
|
PANEL_STATE ps_RetractFist[] =
|
|
{
|
|
{ID_FistPresent0, Fist_BEAT_RATE, pFistRetract, &ps_RetractFist[0], 0,0,0}
|
|
};
|
|
|
|
#define FIST_SWAY_AMT 12
|
|
|
|
// left swing
|
|
#define FIST_XOFF (290 + FIST_SWAY_AMT)
|
|
#define FIST_YOFF 200
|
|
|
|
// right swing
|
|
#define FISTR_XOFF (0 - 80)
|
|
|
|
#define FIST_VEL 3000
|
|
#define FIST_POWER_VEL 3000
|
|
|
|
void InitWeaponFist(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp;
|
|
short rnd_num;
|
|
|
|
|
|
if (Prediction)
|
|
return;
|
|
|
|
if (!(pp->WpnFlags &BIT(WPN_FIST)) ||
|
|
//pp->WpnAmmo[WPN_FIST] <= 0 ||
|
|
(pp->Flags & PF_WEAPON_RETRACT))
|
|
{
|
|
pp->WpnFirstType = WPN_SWORD;
|
|
InitWeaponSword(pp);
|
|
return;
|
|
}
|
|
|
|
if (!pp->Wpn[WPN_FIST])
|
|
{
|
|
psp = pp->Wpn[WPN_FIST] = pSpawnSprite(pp, ps_PresentFist, PRI_MID, FIST_XOFF, FIST_YOFF);
|
|
psp->pos.Y += tileHeight(psp->picndx);
|
|
psp->backupy();
|
|
}
|
|
|
|
if (pp->CurWpn == pp->Wpn[WPN_FIST])
|
|
{
|
|
return;
|
|
}
|
|
|
|
PlayerUpdateWeapon(pp, WPN_FIST);
|
|
|
|
pp->WpnUziType = 2; // Make uzi's go away!
|
|
RetractCurWpn(pp);
|
|
|
|
// Set up the new weapon variables
|
|
psp = pp->CurWpn = pp->Wpn[WPN_FIST];
|
|
psp->flags |= (PANF_WEAPON_SPRITE);
|
|
psp->ActionState = ps_FistSwing;
|
|
psp->RetractState = ps_RetractFist;
|
|
psp->PresentState = ps_PresentFist;
|
|
psp->RestState = ps_FistRest;
|
|
pSetState(psp, psp->PresentState);
|
|
|
|
pp->WpnKungFuMove = 0; // Set to default strike
|
|
|
|
rnd_num = RANDOM_P2(1024);
|
|
if (rnd_num > 900)
|
|
PlaySound(DIGI_TAUNTAI2, pp, v3df_follow|v3df_dontpan);
|
|
else if (rnd_num > 800)
|
|
PlaySound(DIGI_PLAYERYELL1, pp, v3df_follow|v3df_dontpan);
|
|
else if (rnd_num > 700)
|
|
PlaySound(DIGI_PLAYERYELL2, pp, v3df_follow|v3df_dontpan);
|
|
|
|
psp->PlayerP->KeyPressBits |= SB_FIRE;
|
|
}
|
|
|
|
|
|
void pFistPresent(PANEL_SPRITE* psp)
|
|
{
|
|
int rnd;
|
|
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_RETRACT))
|
|
return;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y -= 3 * synctics;
|
|
|
|
if (psp->pos.Y < FIST_YOFF)
|
|
{
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF;
|
|
psp->backupboby();
|
|
|
|
rnd = RandomRange(1000);
|
|
if (rnd > 500)
|
|
{
|
|
psp->PresentState = ps_PresentFist;
|
|
psp->RestState = ps_FistRest;
|
|
}
|
|
else
|
|
{
|
|
psp->PresentState = ps_PresentFist2;
|
|
psp->RestState = ps_Fist2Rest;
|
|
}
|
|
pSetState(psp, psp->RestState);
|
|
}
|
|
}
|
|
|
|
//
|
|
// LEFT SWING
|
|
//
|
|
|
|
void pFistSlide(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnFistBlur(psp);
|
|
|
|
//psp->backupx();
|
|
psp->backupy();
|
|
|
|
//psp->x += pspSinVel(psp);
|
|
psp->pos.Y -= pspSinVel(psp);
|
|
|
|
psp->vel += 68 * synctics;
|
|
}
|
|
|
|
void pFistSlideDown(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnFistBlur(psp);
|
|
|
|
short vel = 3500;
|
|
auto ang = FistAng + psp->ang + psp->PlayerP->FistAng;
|
|
|
|
psp->backupcoords();
|
|
|
|
if (psp->ActionState == ps_Kick || psp->PlayerP->WpnKungFuMove == 3)
|
|
{
|
|
psp->pos.Y -= pspSinVel(psp, ang);
|
|
}
|
|
else
|
|
{
|
|
psp->pos.X -= pspSinVel(psp, ang);
|
|
psp->pos.Y -= pspSinVel(psp, ang) * synctics;
|
|
}
|
|
|
|
psp->vel += 48 * synctics;
|
|
|
|
if (psp->pos.Y > 440)
|
|
{
|
|
// if still holding down the fire key - continue swinging
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
|
|
if (RandomRange(1000) > 500)
|
|
{
|
|
//if(RandomRange(1000) > 300)
|
|
// {
|
|
psp->ActionState = ps_FistSwing;
|
|
psp->PlayerP->WpnKungFuMove = 0;
|
|
// } else
|
|
// {
|
|
// psp->ActionState = ps_Fist3Swing;
|
|
// psp->PlayerP->WpnKungFuMove = 1;
|
|
// }
|
|
pSetState(psp, psp->ActionState);
|
|
|
|
psp->opos.X = psp->pos.X = FIST_XOFF;
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF;
|
|
psp->backupboby();
|
|
psp->PlayerP->FistAng = FistAngTable[RandomRange(SIZ(FistAngTable))];
|
|
psp->ang = 1024;
|
|
psp->vel = vel;
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
pSetState(psp, ps_FistSwing+(psp->State - psp->ActionState)+1);
|
|
psp->ActionState = ps_FistSwing;
|
|
psp->PlayerP->WpnKungFuMove = 0;
|
|
}
|
|
|
|
psp->opos.X = psp->pos.X = FISTR_XOFF+100;
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF;
|
|
psp->backupboby();
|
|
psp->ang = 1024;
|
|
psp->PlayerP->FistAng = FistAngTable[RandomRange(SIZ(FistAngTable))];
|
|
psp->vel = vel;
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// NOT still holding down the fire key - stop swinging
|
|
pSetState(psp, psp->PresentState);
|
|
psp->opos.X = psp->pos.X = FIST_XOFF;
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF + tileHeight(psp->picndx);
|
|
psp->backupboby();
|
|
}
|
|
}
|
|
|
|
//
|
|
// RIGHT SWING
|
|
//
|
|
|
|
void pFistSlideR(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnFistBlur(psp);
|
|
|
|
//psp->backupx();
|
|
psp->backupy();
|
|
|
|
//psp->x += pspSinVel(psp);
|
|
psp->pos.Y += pspSinVel(psp);
|
|
|
|
psp->vel += 68 * synctics;
|
|
}
|
|
|
|
void pFistSlideDownR(PANEL_SPRITE* psp)
|
|
{
|
|
SpawnFistBlur(psp);
|
|
|
|
short vel = 3500;
|
|
auto ang = FistAng + psp->ang + psp->PlayerP->FistAng;
|
|
|
|
psp->backupcoords();
|
|
|
|
if (psp->ActionState == ps_Kick || psp->PlayerP->WpnKungFuMove == 3)
|
|
{
|
|
psp->pos.Y -= pspSinVel(psp, ang);
|
|
}
|
|
else
|
|
{
|
|
psp->pos.X -= pspSinVel(psp, ang);
|
|
psp->pos.Y -= pspSinVel(psp, ang) * synctics;
|
|
}
|
|
|
|
psp->vel += 48 * synctics;
|
|
|
|
if (psp->pos.Y > 440)
|
|
{
|
|
// if still holding down the fire key - continue swinging
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
DoPlayerChooseYell(psp->PlayerP);
|
|
|
|
if (RandomRange(1000) > 500)
|
|
{
|
|
psp->ActionState = ps_FistSwing+5;
|
|
psp->PlayerP->WpnKungFuMove = 0;
|
|
pSetState(psp, psp->ActionState);
|
|
|
|
psp->opos.X = psp->pos.X = FISTR_XOFF+100;
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF;
|
|
psp->backupboby();
|
|
psp->ang = 1024;
|
|
psp->PlayerP->FistAng = FistAngTable[RandomRange(SIZ(FistAngTable))];
|
|
psp->vel = vel;
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
pSetState(psp, ps_FistSwing+(psp->State - psp->ActionState)+1);
|
|
psp->ActionState = ps_FistSwing;
|
|
psp->PlayerP->WpnKungFuMove = 0;
|
|
}
|
|
|
|
psp->opos.X = psp->pos.X = FIST_XOFF;
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF;
|
|
psp->backupboby();
|
|
psp->PlayerP->FistAng = FistAngTable[RandomRange(SIZ(FistAngTable))];
|
|
psp->ang = 1024;
|
|
psp->vel = vel;
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// NOT still holding down the fire key - stop swinging
|
|
pSetState(psp, psp->PresentState);
|
|
psp->opos.X = psp->pos.X = FIST_XOFF;
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF + tileHeight(psp->picndx);
|
|
psp->backupboby();
|
|
}
|
|
}
|
|
|
|
void pFistBobSetup(PANEL_SPRITE* psp)
|
|
{
|
|
if (psp->flags & (PANF_BOB))
|
|
return;
|
|
|
|
psp->backupbobcoords();
|
|
|
|
psp->sin_amt = FIST_SWAY_AMT;
|
|
psp->sin_ndx = 0;
|
|
psp->bob_height_divider = 8;
|
|
}
|
|
|
|
void pFistHide(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= FIST_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->opos.Y = psp->pos.Y = FIST_YOFF + tileHeight(picnum);
|
|
|
|
pWeaponUnHideKeys(psp, psp->PresentState);
|
|
}
|
|
}
|
|
|
|
void pFistRest(PANEL_SPRITE* psp)
|
|
{
|
|
bool force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (pWeaponHideKeys(psp, ps_FistHide))
|
|
return;
|
|
|
|
psp->bobpos.Y += synctics;
|
|
|
|
if (psp->bobpos.Y > FIST_YOFF)
|
|
{
|
|
psp->bobpos.Y = FIST_YOFF;
|
|
}
|
|
|
|
pFistBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
force = !!(psp->flags & PANF_UNHIDE_SHOOT);
|
|
|
|
if (psp->ActionState == ps_Kick)
|
|
psp->ActionState = ps_FistSwing;
|
|
|
|
// Reset move to default
|
|
psp->PlayerP->WpnKungFuMove = 0;
|
|
|
|
if ((psp->PlayerP->input.actions & SB_FIRE) || force)
|
|
{
|
|
if ((psp->PlayerP->KeyPressBits & SB_FIRE) || force)
|
|
{
|
|
psp->flags &= ~(PANF_UNHIDE_SHOOT);
|
|
|
|
psp->ActionState = ps_FistSwing;
|
|
psp->PlayerP->WpnKungFuMove = 0;
|
|
|
|
pSetState(psp, psp->ActionState);
|
|
|
|
psp->vel = 5500;
|
|
|
|
psp->ang = 1024;
|
|
psp->PlayerP->FistAng = FistAngTable[RandomRange(SIZ(FistAngTable))];
|
|
DoPlayerSpriteThrow(psp->PlayerP);
|
|
}
|
|
}
|
|
}
|
|
|
|
void pFistAction(PANEL_SPRITE* psp)
|
|
{
|
|
pFistBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
}
|
|
|
|
|
|
void pFistAttack(PANEL_SPRITE* psp)
|
|
{
|
|
InitFistAttack(psp->PlayerP);
|
|
}
|
|
|
|
void pFistRetract(PANEL_SPRITE* psp)
|
|
{
|
|
int picnum = psp->picndx;
|
|
|
|
psp->backupy();
|
|
psp->pos.Y += 3 * synctics;
|
|
|
|
if (psp->pos.Y >= FIST_YOFF + tileHeight(picnum))
|
|
{
|
|
psp->PlayerP->Flags &= ~(PF_WEAPON_RETRACT);
|
|
psp->PlayerP->Wpn[WPN_FIST] = nullptr;
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
void pFistBlock(PANEL_SPRITE* psp)
|
|
{
|
|
psp->bobpos.Y += synctics;
|
|
|
|
if (psp->bobpos.Y > FIST_YOFF)
|
|
{
|
|
psp->bobpos.Y = FIST_YOFF;
|
|
}
|
|
|
|
pFistBobSetup(psp);
|
|
pWeaponBob(psp, PLAYER_MOVING(psp->PlayerP));
|
|
|
|
if (!(psp->PlayerP->input.actions & SB_OPEN))
|
|
{
|
|
pStatePlusOne(psp);
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// PANEL SPRITE GENERAL ROUTINES
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void pWeaponForceRest(PLAYER* pp)
|
|
{
|
|
pSetState(pp->CurWpn, pp->CurWpn->RestState);
|
|
}
|
|
|
|
bool pWeaponUnHideKeys(PANEL_SPRITE* psp, PANEL_STATE* state)
|
|
{
|
|
// initing the other weapon will take care of this
|
|
if (psp->flags & (PANF_DEATH_HIDE))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (psp->flags & (PANF_WEAPON_HIDE))
|
|
{
|
|
if (!(psp->PlayerP->Flags & PF_WEAPON_DOWN))
|
|
{
|
|
psp->flags &= ~(PANF_WEAPON_HIDE);
|
|
pSetState(psp, state);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
if (psp->PlayerP->input.actions & SB_HOLSTER)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_HOLSTER)
|
|
{
|
|
psp->PlayerP->KeyPressBits &= ~SB_HOLSTER;
|
|
pSetState(psp, state);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
psp->PlayerP->KeyPressBits |= SB_HOLSTER;
|
|
}
|
|
|
|
if (psp->PlayerP->input.actions & SB_FIRE)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_FIRE)
|
|
{
|
|
psp->flags |= (PANF_UNHIDE_SHOOT);
|
|
pSetState(psp, state);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool pWeaponHideKeys(PANEL_SPRITE* psp, PANEL_STATE* state)
|
|
{
|
|
if (psp->PlayerP->Flags & (PF_DEAD))
|
|
{
|
|
psp->flags |= (PANF_DEATH_HIDE);
|
|
pSetState(psp, state);
|
|
return true;
|
|
}
|
|
|
|
if (psp->PlayerP->Flags & (PF_WEAPON_DOWN))
|
|
{
|
|
psp->flags |= (PANF_WEAPON_HIDE);
|
|
pSetState(psp, state);
|
|
return true;
|
|
}
|
|
|
|
if (psp->PlayerP->input.actions & SB_HOLSTER)
|
|
{
|
|
if (psp->PlayerP->KeyPressBits & SB_HOLSTER)
|
|
{
|
|
psp->PlayerP->KeyPressBits &= ~SB_HOLSTER;
|
|
PutStringInfo(psp->PlayerP,"Weapon Holstered");
|
|
pSetState(psp, state);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
psp->PlayerP->KeyPressBits |= SB_HOLSTER;
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
void InsertPanelSprite(PLAYER* pp, PANEL_SPRITE* psp)
|
|
{
|
|
PANEL_SPRITE* cur,* nxt;
|
|
|
|
ASSERT(psp);
|
|
|
|
// if list is empty, insert at front
|
|
if (EMPTY(&pp->PanelSpriteList))
|
|
{
|
|
INSERT(&pp->PanelSpriteList, psp);
|
|
return;
|
|
}
|
|
|
|
// if new pri is <= first pri in list, insert at front
|
|
if (psp->priority <= pp->PanelSpriteList.Next->priority)
|
|
{
|
|
INSERT(&pp->PanelSpriteList, psp);
|
|
return;
|
|
}
|
|
|
|
// search for first pri in list thats less than the new pri
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (cur = list->Next; nxt = cur->Next, cur != list; cur = nxt)
|
|
{
|
|
// if the next pointer is the end of the list, insert it
|
|
if ((List*) cur->Next == (List*) &pp->PanelSpriteList)
|
|
{
|
|
INSERT(cur, psp);
|
|
return;
|
|
}
|
|
|
|
|
|
// if between cur and next, insert here
|
|
if (psp->priority >= cur->priority && psp->priority <= cur->Next->priority)
|
|
{
|
|
INSERT(cur, psp);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
PANEL_SPRITE* pSpawnSprite(PLAYER* pp, PANEL_STATE* state, uint8_t priority, double x, double y)
|
|
{
|
|
unsigned i;
|
|
PANEL_SPRITE* psp;
|
|
|
|
|
|
ASSERT(pp);
|
|
|
|
psp = (PANEL_SPRITE*)CallocMem(sizeof(PANEL_SPRITE), 1);
|
|
|
|
PRODUCTION_ASSERT(psp);
|
|
|
|
psp->priority = priority;
|
|
InsertPanelSprite(pp, psp);
|
|
// INSERT(&pp->PanelSpriteList, psp);
|
|
|
|
psp->PlayerP = pp;
|
|
|
|
psp->opos.X = psp->pos.X = x;
|
|
psp->opos.Y = psp->pos.Y = y;
|
|
pSetState(psp, state);
|
|
if (state == nullptr)
|
|
psp->picndx = -1;
|
|
else
|
|
psp->picndx = state->picndx;
|
|
psp->ang = 0;
|
|
psp->vel = 0;
|
|
psp->rotate_ang = 0;
|
|
psp->scale = FRACUNIT;
|
|
psp->ID = 0;
|
|
|
|
for (i = 0; i < SIZ(psp->over); i++)
|
|
{
|
|
psp->over[i].State = nullptr;
|
|
psp->over[i].pic = -1;
|
|
psp->over[i].xoff = -1;
|
|
psp->over[i].yoff = -1;
|
|
}
|
|
|
|
return psp;
|
|
}
|
|
|
|
void pSuicide(PANEL_SPRITE* psp)
|
|
{
|
|
pKillSprite(psp);
|
|
}
|
|
|
|
void pKillSprite(PANEL_SPRITE* psp)
|
|
{
|
|
PRODUCTION_ASSERT(psp);
|
|
|
|
REMOVE(psp);
|
|
|
|
FreeMem(psp);
|
|
}
|
|
|
|
void pClearSpriteList(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* psp = nullptr, * next_psp = nullptr;
|
|
auto l = &pp->PanelSpriteList;
|
|
|
|
for (psp = l->Next; next_psp = psp->Next, (List*)psp != (List*)l; psp = next_psp)
|
|
{
|
|
if (psp->Next == nullptr || psp->Prev == nullptr) return; // this can happen when cleaning up a fatal error.
|
|
pKillSprite(psp);
|
|
}
|
|
}
|
|
|
|
void pWeaponBob(PANEL_SPRITE* psp, short condition)
|
|
{
|
|
double xdiff = 0, ydiff = 0;
|
|
|
|
double bobvel = fFindDistance2D(psp->PlayerP->vect.X, psp->PlayerP->vect.Y) * (1. / 32768.);
|
|
bobvel = bobvel + (bobvel * (1. / 4.));
|
|
bobvel = min(bobvel, 128.);
|
|
|
|
if (condition)
|
|
{
|
|
psp->flags |= (PANF_BOB);
|
|
}
|
|
else
|
|
{
|
|
if (labs((psp->sin_ndx & 1023) - 0) < 70)
|
|
{
|
|
psp->flags &= ~(PANF_BOB);
|
|
psp->sin_ndx = (RANDOM_P2(1024) < 512) ? 1024 : 0;
|
|
}
|
|
}
|
|
|
|
if (cl_weaponsway && (psp->flags & PANF_BOB))
|
|
{
|
|
// //
|
|
// sin_xxx moves the weapon left-right
|
|
// //
|
|
|
|
// increment the ndx into the sin table
|
|
psp->sin_ndx = (psp->sin_ndx + (synctics << 3) + (cl_swsmoothsway ? (synctics << 2) : RANDOM_P2(8) * synctics)) & 2047;
|
|
|
|
// get height
|
|
xdiff = psp->sin_amt * bsinf(psp->sin_ndx, -14);
|
|
|
|
// //
|
|
// bob_xxx moves the weapon up-down
|
|
// //
|
|
|
|
// as the weapon moves left-right move the weapon up-down in the same
|
|
// proportion
|
|
double bob_ndx = (psp->sin_ndx + 512) & 1023;
|
|
|
|
// base bob_amt on the players velocity - Max of 128
|
|
double bob_amt = bobvel / psp->bob_height_divider;
|
|
ydiff = bob_amt * bsinf(bob_ndx, -14);
|
|
}
|
|
|
|
// Back up current coordinates for interpolating.
|
|
psp->backupcoords();
|
|
|
|
psp->pos.X = psp->bobpos.X + xdiff;
|
|
psp->pos.Y = psp->bobpos.Y + ydiff + UziRecoilYadj;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// PANEL SPRITE CONTROL ROUTINES
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool DrawBeforeView = false;
|
|
void pDisplaySprites(PLAYER* pp, double smoothratio)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
PANEL_SPRITE* next=nullptr;
|
|
short shade, picnum, overlay_shade = 0;
|
|
double x, y;
|
|
unsigned i;
|
|
|
|
uint8_t pal = 0;
|
|
short ang;
|
|
int flags;
|
|
|
|
double const look_anghalf = pp->angle.look_anghalf(smoothratio);
|
|
double const looking_arc = pp->angle.looking_arc(smoothratio);
|
|
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (auto psp = list->Next; next = psp->Next, psp != list; psp = next)
|
|
{
|
|
ang = psp->rotate_ang;
|
|
shade = 0;
|
|
flags = 0;
|
|
if (cl_hudinterpolation)
|
|
{
|
|
x = interpolatedvaluef(psp->opos.X, psp->pos.X, smoothratio);
|
|
y = interpolatedvaluef(psp->opos.Y, psp->pos.Y, smoothratio);
|
|
|
|
}
|
|
else
|
|
{
|
|
x = psp->pos.X;
|
|
y = psp->pos.Y;
|
|
}
|
|
|
|
x -= look_anghalf;
|
|
y += looking_arc;
|
|
|
|
// initilize pal here - jack with it below
|
|
pal = uint8_t(psp->pal);
|
|
|
|
if (DrawBeforeView)
|
|
if (!(psp->flags & PANF_DRAW_BEFORE_VIEW))
|
|
continue;
|
|
|
|
if (psp->flags & (PANF_SUICIDE))
|
|
{
|
|
//pKillSprite(psp);
|
|
continue;
|
|
}
|
|
|
|
// if the state is null get the picnum for other than picndx
|
|
if (psp->picndx == -1 || !psp->State)
|
|
picnum = psp->picnum;
|
|
else
|
|
picnum = psp->picndx;
|
|
|
|
// UK panzies have to have darts instead of shurikens.
|
|
if (sw_darts)
|
|
switch (picnum)
|
|
{
|
|
case STAR_REST:
|
|
picnum = 2510;
|
|
break;
|
|
case STAR_REST + 1:
|
|
picnum = 2511;
|
|
break;
|
|
case STAR_REST + 2:
|
|
picnum = 2512;
|
|
break;
|
|
case STAR_REST + 3:
|
|
picnum = 2513;
|
|
break;
|
|
case STAR_REST + 4:
|
|
picnum = 2514;
|
|
break;
|
|
case STAR_REST + 5:
|
|
picnum = 2515;
|
|
break;
|
|
case STAR_REST + 6:
|
|
picnum = 2516;
|
|
break;
|
|
case STAR_REST + 7:
|
|
picnum = 2517;
|
|
break;
|
|
}
|
|
|
|
if (pp->Bloody)
|
|
{
|
|
switch (picnum)
|
|
{
|
|
case SWORD_REST:
|
|
picnum = BLOODYSWORD_REST;
|
|
break;
|
|
case SWORD_SWING0:
|
|
picnum = BLOODYSWORD_SWING0;
|
|
break;
|
|
case SWORD_SWING1:
|
|
picnum = BLOODYSWORD_SWING1;
|
|
break;
|
|
case SWORD_SWING2:
|
|
picnum = BLOODYSWORD_SWING2;
|
|
break;
|
|
|
|
case FIST_REST:
|
|
picnum = 4077;
|
|
break;
|
|
case FIST2_REST:
|
|
picnum = 4051;
|
|
break;
|
|
case FIST_SWING0:
|
|
picnum = 4074;
|
|
break;
|
|
case FIST_SWING1:
|
|
picnum = 4075;
|
|
break;
|
|
case FIST_SWING2:
|
|
picnum = 4076;
|
|
break;
|
|
|
|
case STAR_REST:
|
|
case 2510:
|
|
if (!sw_darts)
|
|
picnum = 2138;
|
|
else
|
|
picnum = 2518; // Bloody Dart Hand
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pp->WpnShotgunType == 1)
|
|
{
|
|
switch (picnum)
|
|
{
|
|
case SHOTGUN_REST:
|
|
case SHOTGUN_RELOAD0:
|
|
picnum = 2227;
|
|
break;
|
|
case SHOTGUN_RELOAD1:
|
|
picnum = 2226;
|
|
break;
|
|
case SHOTGUN_RELOAD2:
|
|
picnum = 2225;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// don't draw
|
|
if (psp->flags & (PANF_INVISIBLE))
|
|
continue;
|
|
|
|
if (psp->State && (psp->State->flags & psf_Invisible))
|
|
continue;
|
|
|
|
// if its a weapon sprite and the view is set to the outside don't draw the sprite
|
|
if (psp->flags & (PANF_WEAPON_SPRITE))
|
|
{
|
|
sectortype* sectp = nullptr;
|
|
int16_t floorshade = 0;
|
|
if (pp->insector())
|
|
{
|
|
sectp = pp->cursector;
|
|
pal = sectp->floorpal;
|
|
floorshade = sectp->floorshade;
|
|
|
|
if (pal != PALETTE_DEFAULT)
|
|
{
|
|
if (sectp->hasU() && (sectp->flags & SECTFU_DONT_COPY_PALETTE))
|
|
pal = PALETTE_DEFAULT;
|
|
}
|
|
|
|
if (pal == PALETTE_FOG || pal == PALETTE_DIVE || pal == PALETTE_DIVE_LAVA)
|
|
pal = uint8_t(psp->pal); // Set it back
|
|
}
|
|
|
|
///////////
|
|
|
|
if (pp->InventoryActive[INVENTORY_CLOAK])
|
|
{
|
|
flags |= (RS_TRANS1);
|
|
}
|
|
|
|
shade = overlay_shade = floorshade - 10;
|
|
|
|
if (psp->PlayerP->Flags & (PF_VIEW_FROM_OUTSIDE))
|
|
{
|
|
if (!(psp->PlayerP->Flags & PF_VIEW_OUTSIDE_WEAPON))
|
|
continue;
|
|
}
|
|
|
|
if (psp->PlayerP->Flags & (PF_VIEW_FROM_CAMERA))
|
|
continue;
|
|
|
|
// !FRANK - this was moved from BELOW this IF statement
|
|
// if it doesn't have a picflag or its in the view
|
|
if (sectp && sectp->hasU() && (sectp->flags & SECTFU_DONT_COPY_PALETTE))
|
|
pal = 0;
|
|
}
|
|
|
|
if (psp->flags & (PANF_TRANSLUCENT))
|
|
flags |= (RS_TRANS1);
|
|
|
|
flags |= (psp->flags & (PANF_TRANS_FLIP));
|
|
|
|
if (psp->flags & (PANF_CORNER))
|
|
flags |= (RS_TOPLEFT);
|
|
|
|
if ((psp->State && (psp->State->flags & psf_Xflip)) || (psp->flags & PANF_XFLIP))
|
|
{
|
|
flags |= (RS_XFLIPHUD);
|
|
}
|
|
|
|
// shading
|
|
if (psp->State && (psp->State->flags & (psf_ShadeHalf|psf_ShadeNone)))
|
|
{
|
|
if ((psp->State->flags & psf_ShadeNone))
|
|
shade = 0;
|
|
else if ((psp->State->flags & psf_ShadeHalf))
|
|
shade /= 2;
|
|
}
|
|
|
|
if (pal == PALETTE_DEFAULT)
|
|
{
|
|
switch (picnum)
|
|
{
|
|
case 4080:
|
|
case 4081:
|
|
case 2220:
|
|
case 2221:
|
|
pal = plActor->user.spal;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// temporary hack to fix fist artifacts until a solution is found in the panel system itself
|
|
switch (picnum)
|
|
{
|
|
case FIST_SWING0:
|
|
case FIST_SWING1:
|
|
case FIST_SWING2:
|
|
case FIST2_SWING0:
|
|
case FIST2_SWING1:
|
|
case FIST2_SWING2:
|
|
case FIST3_SWING0:
|
|
case FIST3_SWING1:
|
|
case FIST3_SWING2:
|
|
case BLOODYFIST_SWING0:
|
|
case BLOODYFIST_SWING1:
|
|
case BLOODYFIST_SWING2:
|
|
case BLOODYFIST2_SWING0:
|
|
case BLOODYFIST2_SWING1:
|
|
case BLOODYFIST2_SWING2:
|
|
case BLOODYFIST3_SWING0:
|
|
case BLOODYFIST3_SWING1:
|
|
case BLOODYFIST3_SWING2:
|
|
if ((flags & RS_XFLIPHUD) && x > 160)
|
|
x = 65;
|
|
else if (!(flags & RS_XFLIPHUD) && x < 160)
|
|
x = 345;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
hud_drawsprite(x, y, psp->scale, ang, picnum, shade, pal, flags);
|
|
|
|
// do overlays (if any)
|
|
for (i = 0; i < SIZ(psp->over); i++)
|
|
{
|
|
// get pic from state
|
|
if (psp->over[i].State)
|
|
picnum = psp->over[i].State->picndx;
|
|
else
|
|
// get pic from over variable
|
|
if (psp->over[i].pic >= 0)
|
|
picnum = psp->over[i].pic;
|
|
else
|
|
continue;
|
|
|
|
if ((psp->over[i].flags & psf_ShadeNone))
|
|
overlay_shade = 0;
|
|
|
|
if (picnum)
|
|
{
|
|
hud_drawsprite((x + psp->over[i].xoff), (y + psp->over[i].yoff), psp->scale, ang, picnum, overlay_shade, pal, flags);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void pSpriteControl(PLAYER* pp)
|
|
{
|
|
PANEL_SPRITE* next=nullptr;
|
|
|
|
auto list = pp->GetPanelSpriteList();
|
|
for (auto psp = list->Next; next = psp->Next, psp != list; psp = next)
|
|
{
|
|
// reminder - if these give an assertion look for pKillSprites
|
|
// somewhere else other than by themselves
|
|
// RULE: Sprites can only kill themselves
|
|
PRODUCTION_ASSERT(psp);
|
|
|
|
if (psp->State)
|
|
pStateControl(psp);
|
|
else
|
|
// for sprits that are not state controled but still need to call
|
|
// something
|
|
if (psp->PanelSpriteFunc)
|
|
{
|
|
(*psp->PanelSpriteFunc)(psp);
|
|
}
|
|
}
|
|
}
|
|
|
|
void pSetState(PANEL_SPRITE* psp, PANEL_STATE* panel_state)
|
|
{
|
|
PRODUCTION_ASSERT(psp);
|
|
psp->tics = 0;
|
|
psp->State = panel_state;
|
|
psp->picndx = panel_state ? panel_state->picndx : 0;
|
|
}
|
|
|
|
|
|
void pNextState(PANEL_SPRITE* psp)
|
|
{
|
|
// Transition to the next state
|
|
psp->State = psp->State->NextState;
|
|
|
|
if ((psp->State->flags & psf_QuickCall))
|
|
{
|
|
(*psp->State->Animator)(psp);
|
|
psp->State = psp->State->NextState;
|
|
}
|
|
}
|
|
|
|
void pStatePlusOne(PANEL_SPRITE* psp)
|
|
{
|
|
psp->tics = 0;
|
|
psp->State++;
|
|
|
|
if ((psp->State->flags & psf_QuickCall))
|
|
{
|
|
(*psp->State->Animator)(psp);
|
|
psp->State = psp->State->NextState;
|
|
}
|
|
}
|
|
|
|
|
|
void pStateControl(PANEL_SPRITE* psp)
|
|
{
|
|
unsigned i;
|
|
short tics = synctics;
|
|
|
|
psp->tics += tics;
|
|
|
|
// Skip states if too much time has passed
|
|
while (psp->tics >= psp->State->tics)
|
|
{
|
|
// Set Tics
|
|
psp->tics -= psp->State->tics;
|
|
|
|
pNextState(psp);
|
|
}
|
|
|
|
// Set picnum to the correct pic
|
|
psp->picndx = psp->State->picndx;
|
|
|
|
// do overlay states
|
|
for (i = 0; i < SIZ(psp->over); i++)
|
|
{
|
|
if (!psp->over[i].State)
|
|
continue;
|
|
|
|
psp->over[i].tics += tics;
|
|
|
|
// Skip states if too much time has passed
|
|
while (psp->over[i].tics >= psp->over[i].State->tics)
|
|
{
|
|
// Set Tics
|
|
psp->over[i].tics -= psp->over[i].State->tics;
|
|
psp->over[i].State = psp->over[i].State->NextState;
|
|
|
|
if (!psp->over[i].State)
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Call the correct animator
|
|
if (psp->State->Animator)
|
|
(*psp->State->Animator)(psp);
|
|
|
|
}
|
|
|
|
|
|
void UpdatePanel(double smoothratio)
|
|
{
|
|
short pnum;
|
|
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
if (pnum == screenpeek)
|
|
pDisplaySprites(Player + pnum, smoothratio);
|
|
}
|
|
}
|
|
|
|
void PreUpdatePanel(double smoothratio)
|
|
{
|
|
short pnum;
|
|
DrawBeforeView = true;
|
|
|
|
//if (DrawBeforeView)
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
if (pnum == screenpeek)
|
|
pDisplaySprites(Player + pnum, smoothratio);
|
|
}
|
|
|
|
DrawBeforeView = false;
|
|
}
|
|
|
|
#define EnvironSuit_RATE 10
|
|
#define Fly_RATE 10
|
|
#define Cloak_RATE 10
|
|
#define Night_RATE 10
|
|
#define Box_RATE 10
|
|
#define Medkit_RATE 10
|
|
#define RepairKit_RATE 10
|
|
#define ChemBomb_RATE 10
|
|
#define FlashBomb_RATE 10
|
|
#define SmokeBomb_RATE 10
|
|
#define Caltrops_RATE 10
|
|
|
|
#define ID_PanelEnvironSuit 2397
|
|
PANEL_STATE ps_PanelEnvironSuit[] =
|
|
{
|
|
{ID_PanelEnvironSuit, EnvironSuit_RATE, PanelInvTestSuicide, &ps_PanelEnvironSuit[0], 0,0,0}
|
|
};
|
|
|
|
|
|
#include "saveable.h"
|
|
|
|
static saveable_code saveable_panel_code[] =
|
|
{
|
|
SAVE_CODE(pSuicide),
|
|
SAVE_CODE(SwordBlur),
|
|
SAVE_CODE(SpecialUziRetractFunc),
|
|
SAVE_CODE(FistBlur),
|
|
};
|
|
|
|
static saveable_data saveable_panel_data[] =
|
|
{
|
|
SAVE_DATA(ps_PresentSword),
|
|
SAVE_DATA(ps_SwordRest),
|
|
SAVE_DATA(ps_SwordHide),
|
|
SAVE_DATA(ps_SwordSwing),
|
|
SAVE_DATA(ps_RetractSword),
|
|
|
|
SAVE_DATA(ps_PresentStar),
|
|
SAVE_DATA(ps_StarHide),
|
|
SAVE_DATA(ps_StarRest),
|
|
SAVE_DATA(ps_ThrowStar),
|
|
SAVE_DATA(ps_RetractStar),
|
|
|
|
SAVE_DATA(ps_FireUzi),
|
|
SAVE_DATA(ps_UziNull),
|
|
SAVE_DATA(ps_UziHide),
|
|
SAVE_DATA(ps_PresentUzi),
|
|
SAVE_DATA(ps_PresentUziReload),
|
|
SAVE_DATA(ps_RetractUzi),
|
|
SAVE_DATA(ps_FireUzi2),
|
|
SAVE_DATA(ps_PresentUzi2),
|
|
SAVE_DATA(ps_Uzi2Hide),
|
|
SAVE_DATA(ps_RetractUzi2),
|
|
SAVE_DATA(ps_Uzi2Suicide),
|
|
SAVE_DATA(ps_Uzi2Null),
|
|
SAVE_DATA(ps_UziEject),
|
|
SAVE_DATA(ps_UziClip),
|
|
SAVE_DATA(ps_UziReload),
|
|
SAVE_DATA(ps_UziDoneReload),
|
|
SAVE_DATA(ps_UziShell),
|
|
SAVE_DATA(ps_Uzi2Shell),
|
|
|
|
SAVE_DATA(ps_ShotgunShell),
|
|
SAVE_DATA(ps_PresentShotgun),
|
|
SAVE_DATA(ps_ShotgunRest),
|
|
SAVE_DATA(ps_ShotgunHide),
|
|
SAVE_DATA(ps_ShotgunRecoil),
|
|
SAVE_DATA(ps_ShotgunRecoilAuto),
|
|
SAVE_DATA(ps_ShotgunFire),
|
|
SAVE_DATA(ps_ShotgunAutoFire),
|
|
SAVE_DATA(ps_ShotgunReload),
|
|
SAVE_DATA(ps_RetractShotgun),
|
|
SAVE_DATA(ps_ShotgunFlash),
|
|
|
|
SAVE_DATA(ps_PresentRail),
|
|
SAVE_DATA(ps_RailRest),
|
|
SAVE_DATA(ps_RailHide),
|
|
SAVE_DATA(ps_RailRecoil),
|
|
SAVE_DATA(ps_RailFire),
|
|
SAVE_DATA(ps_RailFireEMP),
|
|
SAVE_DATA(ps_RetractRail),
|
|
|
|
SAVE_DATA(ps_PresentHothead),
|
|
SAVE_DATA(ps_HotheadHide),
|
|
SAVE_DATA(ps_RetractHothead),
|
|
SAVE_DATA(ps_HotheadRest),
|
|
SAVE_DATA(ps_HotheadRestRing),
|
|
SAVE_DATA(ps_HotheadRestNapalm),
|
|
SAVE_DATA(ps_HotheadAttack),
|
|
SAVE_DATA(ps_HotheadRing),
|
|
SAVE_DATA(ps_HotheadNapalm),
|
|
SAVE_DATA(ps_HotheadTurn),
|
|
SAVE_DATA(ps_HotheadTurnRing),
|
|
SAVE_DATA(ps_HotheadTurnNapalm),
|
|
SAVE_DATA(ps_OnFire),
|
|
|
|
SAVE_DATA(ps_PresentMicro),
|
|
SAVE_DATA(ps_MicroRest),
|
|
SAVE_DATA(ps_MicroHide),
|
|
SAVE_DATA(ps_InitNuke),
|
|
SAVE_DATA(ps_MicroRecoil),
|
|
SAVE_DATA(ps_MicroFire),
|
|
SAVE_DATA(ps_MicroSingleFire),
|
|
SAVE_DATA(ps_RetractMicro),
|
|
SAVE_DATA(ps_MicroHeatFlash),
|
|
SAVE_DATA(ps_MicroNukeFlash),
|
|
|
|
SAVE_DATA(ps_PresentHeart),
|
|
SAVE_DATA(ps_HeartRest),
|
|
SAVE_DATA(ps_HeartHide),
|
|
SAVE_DATA(ps_HeartAttack),
|
|
SAVE_DATA(ps_RetractHeart),
|
|
SAVE_DATA(ps_HeartBlood),
|
|
SAVE_DATA(ps_HeartBloodSmall),
|
|
SAVE_DATA(ps_HeartBlood),
|
|
|
|
SAVE_DATA(ps_PresentGrenade),
|
|
SAVE_DATA(ps_GrenadeRest),
|
|
SAVE_DATA(ps_GrenadeHide),
|
|
SAVE_DATA(ps_GrenadeFire),
|
|
SAVE_DATA(ps_GrenadeRecoil),
|
|
SAVE_DATA(ps_RetractGrenade),
|
|
|
|
SAVE_DATA(ps_PresentMine),
|
|
SAVE_DATA(ps_MineRest),
|
|
SAVE_DATA(ps_MineHide),
|
|
SAVE_DATA(ps_MineThrow),
|
|
SAVE_DATA(ps_RetractMine),
|
|
|
|
SAVE_DATA(ps_ChopsAttack1),
|
|
SAVE_DATA(ps_ChopsAttack2),
|
|
SAVE_DATA(ps_ChopsAttack3),
|
|
SAVE_DATA(ps_ChopsAttack4),
|
|
SAVE_DATA(ps_ChopsWait),
|
|
SAVE_DATA(ps_ChopsRetract),
|
|
|
|
SAVE_DATA(ps_PresentFist),
|
|
SAVE_DATA(ps_FistRest),
|
|
SAVE_DATA(ps_FistHide),
|
|
SAVE_DATA(ps_PresentFist2),
|
|
SAVE_DATA(ps_Fist2Rest),
|
|
SAVE_DATA(ps_Fist2Hide),
|
|
/*
|
|
SAVE_DATA(ps_PresentFist3),
|
|
SAVE_DATA(ps_Fist3Rest),
|
|
SAVE_DATA(ps_Fist3Hide),
|
|
*/
|
|
SAVE_DATA(ps_FistSwing),
|
|
SAVE_DATA(ps_Fist2Swing),
|
|
SAVE_DATA(ps_Fist3Swing),
|
|
SAVE_DATA(ps_Kick),
|
|
SAVE_DATA(ps_RetractFist),
|
|
|
|
SAVE_DATA(ps_PanelEnvironSuit),
|
|
};
|
|
|
|
saveable_module saveable_panel =
|
|
{
|
|
// code
|
|
saveable_panel_code,
|
|
SIZ(saveable_panel_code),
|
|
|
|
// data
|
|
saveable_panel_data,
|
|
SIZ(saveable_panel_data)
|
|
};
|
|
|
|
END_SW_NS
|