NS/releases/3.2.0/source/mod/AvHPlayer.cpp

10313 lines
347 KiB
C++
Raw Normal View History

//======== (C) Copyright 2001 Charles G. Cleveland All rights reserved. =========
//
// The copyright to the contents herein is the property of Charles G. Cleveland.
// The contents may be used and/or copied only with the written permission of
// Charles G. Cleveland, or in accordance with the terms and conditions stipulated in
// the agreement/contract under which the contents have been supplied.
//
// Purpose:
//
// $Workfile: AvHPlayer.cpp $
// $Date: 2002/11/22 21:18:24 $
//
//-------------------------------------------------------------------------------
// $Log: AvHPlayer.cpp,v $
// Revision 1.86 2002/11/22 21:18:24 Flayra
// - Potentially fixed strange Onos collision crash
// - Don't allow player to join team after he's seen another team
// - "lastinv" support
// - Fixed perma-cloak bug after losing last sensory chamber whilst cloaked
// - Started fixing commander PAS problem
// - Fixed readyroom "ghost player" exploit when F4 during REIN
// - Draw damage in debug, never otherwise
//
// Revision 1.85 2002/11/15 04:42:50 Flayra
// - Regenerate now returns true if healing was successful
// - Logging changes and fixes
//
// Revision 1.84 2002/11/13 01:49:08 Flayra
// - Proper death message logging for Psychostats
//
// Revision 1.83 2002/11/12 22:39:25 Flayra
// - Logging changes for Psychostats compatibility
//
// Revision 1.82 2002/11/12 18:44:54 Flayra
// - Added mp_logdetail support for damage messages
// - Changed the alien ability anti-exploit code to try to co-exist with scripters
//
// Revision 1.81 2002/11/12 02:28:39 Flayra
// - Fixed problems with armor not being updated when armor upgrades completed
// - Aliens now keep same percentage of health and armor when morphing
// - Much better logging, up to standard
// - Don't add enemy buildings to hive sight unless parasited (solves blip spam)
// - Removed draw damage from public build
// - Changes to minimap to less overflows at end of big games
//
// Revision 1.80 2002/11/06 01:38:54 Flayra
// - Damage refactoring (TakeDamage assumes caller has already adjusted for friendly fire, etc.)
// - Regeneration update
//
// Revision 1.79 2002/11/05 06:17:26 Flayra
// - Balance changes
//
// Revision 1.78 2002/10/28 20:36:18 Flayra
// - Updated auth mask
//
// Revision 1.77 2002/10/25 21:48:01 Flayra
// - Added more auth masks
// - Moved mSkin to pev->skin so playerclass could hold what used to be mPlayMode could be used to quickly perform culling, to try to prevent overflowage at the end of large games
//
// Revision 1.76 2002/10/24 21:40:02 Flayra
// - Reworked jetpack effect
// - Authicons update
// - Set builder for alien buildings, so turret kills can credit builder
// - Alien easter eggs
// - Network optimizations after game reset on huge (20-32) player games
// - Allow server ops to disable auth icons
// - Alien energy tweaks
// - Show unbuilt hives in hive sight
// - Move alien energy updating fully into shared code
// - Tried to fix full health ring showing for dead selected players
// - Moved help text client-side
// - Cache info_locations and gamma until map change
// - Skin fixes
//
// Revision 1.75 2002/10/20 21:10:57 Flayra
// - Optimizations
//
// Revision 1.74 2002/10/20 16:36:37 Flayra
// - Code optimization! Took forever to find.
//
// Revision 1.73 2002/10/20 02:36:14 Flayra
// - Regular update
//
// Revision 1.72 2002/10/19 22:33:44 Flayra
// - Various server optimizations
//
// Revision 1.71 2002/10/18 22:21:54 Flayra
// - Limit alien buildings in sphere
// - Fix various spawning problems (morphing as level 5, redemption for level 5)
// - Max motd length fix
//
// Revision 1.70 2002/10/17 17:34:09 Flayra
// - Authmask update
// - Part 1 of persistent weapons fix (found with Grendel)
//
// Revision 1.69 2002/10/16 20:55:40 Flayra
// - Debug code for tracking down death animation problems
// - Updated auth masks
//
// Revision 1.68 2002/10/16 01:05:38 Flayra
// - Sent health as short for big aliens
// - Fixed sayings not triggering commander alerts
// - Now name changes are queued until the next match
// - Added authmask support
// - Egg idle sounds play more frequently, refactored too
// - Fixed preserving model in ready room after game end
// - Profiling of AddToFullPack
// - Fix for falling through lifts when morphing on them (untested)
//
// Revision 1.67 2002/10/04 18:04:07 Flayra
// - Fixed floating gestation sacs
// - Aliens now fall all the way to ground during countdown (instead of floating and shaking)
//
// Revision 1.66 2002/10/03 20:24:39 Flayra
// - Changes for "more resources required"
//
// Revision 1.65 2002/10/03 19:32:06 Flayra
// - Reworked orders completely
// - Send max resources to players
// - Heavy armor sped up slightly
// - Kill players who illegally try to use alien abilities
// - Moved energy to a new variable, send health for aliens
// - Only freeze players during countdown
// - Removed slowdown when taking damage
// - Send blips in two messages, one friendly and one enemy (old bad hack)
//
// Revision 1.64 2002/09/25 20:50:03 Flayra
// - Added 3 new sayings
// - Frame-rate independent updating
// - Don't allow player to kill self while commanding
//
// Revision 1.63 2002/09/23 22:27:52 Flayra
// - Added skin support
// - Added client connected/disconnected hooks for particle system propagation optimizations
// - Removed power armor, added heavy armor
// - Fixed death animations
// - Added hook to see if commander has given an order and to see if he's idle
// - Bound resources for aliens
// - Soldiers asking for ammo and health trigger commander alert
// - Added gestation anims
// - Slowed down Onos movement
// - When cheats are enabled, purchases are free
//
// Revision 1.62 2002/09/09 20:04:53 Flayra
// - Added commander voting
// - Commander score is now the average of the rest of his players (reverted back when he leaves CC)
// - Fixed bug where upgrades were getting removed and then add repeatedly
// - Added multiple skins for marines
// - Play sound when aliens lose an upgrade
// - Changed fov to 90 for all aliens for software compatibility
// - Added hiveinfo drawing
// - Tweaked marine and alien speeds (to compensate for loss of drastic alien fov)
//
// Revision 1.61 2002/08/31 18:01:02 Flayra
// - Work at VALVe
//
// Revision 1.60 2002/08/16 02:42:57 Flayra
// - New damage types
// - Much more efficient blip calculation (at least it should be, I haven't measured it so who really knows)
// - New way of representing ensnare state for shared code (disabling jumping, jetpacking)
// - Removed old overwatch code
// - Store health in fuser2 for drawing health for commander
// - Swap bile bomb and umbra
//
// Revision 1.59 2002/08/09 01:10:30 Flayra
// - Keep previous model when a game is over and going back to ready room
// - Refactoring for scoreboard
// - Support for "jump" animation
// - Freeze player before game starts
// - Reset score when leaving a team
//
// Revision 1.58 2002/08/02 21:52:18 Flayra
// - Cleaned up jetpacks, made webbing useful again, help messages, added "orderself" cheat, changed gestation messages for new tooltip system, added GetHasAvailableUpgrades() for help system, removed particle template messages, slowed down particle template update rate to reduce overflows (woo!)
//
// Revision 1.57 2002/07/28 19:21:28 Flayra
// - Balance changes after/during RC4a
//
// Revision 1.56 2002/07/26 23:07:53 Flayra
// - Numerical feedback
// - New artwork for marine, with jetpack as body group (this code doesn't work)
// - Misc. fixes (alien vision mode not being preserved when it should, and staying when it shouldn't)
//
// Revision 1.55 2002/07/24 18:45:42 Flayra
// - Linux and scripting changes
//
// Revision 1.54 2002/07/23 17:17:57 Flayra
// - Added power armor, added "talking" state for hive sight, changes for new resource model, removed max buildings, create buildings with random orientation, added stimpack, tweaked redemption, added new hive sight info
//
// Revision 1.53 2002/07/10 14:44:39 Flayra
// - Draw chat text while dead (bug #280)
//
// Revision 1.52 2002/07/08 17:15:39 Flayra
// - Added validation of all impulses (assumes invalid by default, instead of valid), reset players like all other entities, ensnaring fixes, players are temporarily invulnerable when they spawn, preserve health and armor when using command station, fixed up redemption, add hooks for commander voting and going back to the ready room, models can't be changed via the console anymore
//
// Revision 1.51 2002/07/01 22:41:40 Flayra
// - Removed outdated overwatch target and tension events
//
// Revision 1.50 2002/07/01 21:43:16 Flayra
// - Removed outdated adrenaline concept, added new alien sight mode, primal scream, removed flashlight battery message, fixed morphing problems for level 5 (and others), tweaked marine and alien speeds, fixed triple speed upgrade problem, disabled overwatch, moved player assets out of precache() (wasn't being called reliably), get full energy after a lifeform change, hive sight only draws out of sight now, added scent of fear
//
// Revision 1.49 2002/06/25 18:13:57 Flayra
// - Added death animations, added more general animation support, leaps do damage, general construct effects, new alien inventory system, added charging, info_locations, galloping, new alien building code, better morphing system (prevents getting stuck), more builder error messages, clear deaths on game end, hide health while gestating
//
// Revision 1.48 2002/06/10 20:03:13 Flayra
// - Updated for new minimap (remember killed position). Updated blood so marines aren't bloody, and aliens emit yellow blood. Removed slowing when hit (now players fly back when hit though), UpdateBlips() hack (fix when time)
//
// Revision 1.47 2002/06/03 16:54:59 Flayra
// - Added more effective player classes for scoreboard, send player class every time role changes (more network usage, always updated), changed hive sight to always be visible when under attack, all entities added to hive sight, not just players
//
// Revision 1.46 2002/05/28 18:03:40 Flayra
// - Refactored size for role code for movement chambers, deal with inventory properly (entity leak), increased web ensnaring effects (put weapon away!), reinforcement refactoring, tweaked speeds so marines are a little slower, and speed upgrade works properly again (now level 1 can generally outrun marines), added recycling support, play destroy egg sound when killed when morphing, new hive sight, EffectivePlayerClassChanged() refactoring
//
// Revision 1.45 2002/05/23 02:33:20 Flayra
// - Post-crash checkin. Restored @Backup from around 4/16. Contains changes for last four weeks of development.
//
//===============================================================================
#include "util/nowarnings.h"
#include "mod/AvHPlayer.h"
#include "mod/AvHMessage.h"
#include "mod/AvHParticleTemplateServer.h"
#include "mod/AvHEntities.h"
#include "mod/AvHGamerules.h"
#include "mod/AvHServerVariables.h"
#include "mod/AvHConstants.h"
#include "mod/AvHMarineWeapons.h"
#include "dlls/client.h"
#include "dlls/util.h"
#include "mod/AvHSoundListManager.h"
#include "mod/AvHServerUtil.h"
#include "mod/AvHMarineEquipment.h"
#include "mod/AvHTitles.h"
#include "mod/AvHMarineEquipmentConstants.h"
#include "mod/AvHParticleTemplate.h"
#include "common/vector_util.h"
#include "dlls/roach.h"
#include "mod/AvHSelectionHelper.h"
#include "mod/AvHPlayerUpgrade.h"
#include "mod/AvHSharedUtil.h"
#include "mod/AvHServerUtil.h"
#include "mod/AvHDramaticPriority.h"
#include "mod/AvHHulls.h"
#include "mod/AvHMovementUtil.h"
#include "mod/AvHAlienWeaponConstants.h"
#include "mod/AvHParticleSystemEntity.h"
#include "mod/AvHAlienAbilities.h"
#include "mod/AvHAlienAbilityConstants.h"
#include "mod/AvHAlienEquipmentConstants.h"
#include "mod/AvHMarineTurret.h"
#include "mod/AvHSiegeTurret.h"
#include "mod/AvHBlipConstants.h"
#include "mod/AvHParticleConstants.h"
#include "util/MathUtil.h"
#include "types.h"
#include "mod/AvHNetworkMessages.h"
#include "mod/AvHNexusServer.h"
std::string GetLogStringForPlayer( edict_t *pEntity );
extern int gJetpackEventID;
extern int gAlienSightOnEventID;
extern int gAlienSightOffEventID;
extern int gStartOverwatchEventID;
extern int gEndOverwatchEventID;
extern int gRegenerationEventID;
extern int gStartCloakEventID;
extern int gEndCloakEventID;
extern int gNumFullPackCalls;
extern int gWeaponAnimationEventID;
extern int gMetabolizeSuccessEventID;
extern int gPhaseInEventID;
// Yucky globals
extern AvHParticleTemplateListServer gParticleTemplateList;
extern AvHSoundListManager gSoundListManager;
extern cvar_t allow_spectators;
extern cvar_t avh_marinereinforcementcost;
#ifdef USE_OLDAUTH
extern cvar_t avh_uplink;
#endif
#ifdef DEBUG
extern cvar_t avh_spawninvulnerabletime;
#endif
AvHSelectionHelper gSelectionHelper;
extern vec3_t gPMDebugPoint;
extern void ResetPlayerPVS( edict_t *client, int clientnum );
LINK_ENTITY_TO_CLASS( player, AvHPlayer );
const float kTransitionFadeTime = .6f;
const float kSpawnInFadeTime = .9f;
AvHPlayer::AvHPlayer()
{
this->Init();
//TODO: find out lifecycle of entity vs. lifecycle of client and reset only when we have a new client.
this->InitBalanceVariables();
}
void AvHPlayer::AddDebugEnemyBlip(float inX, float inY, float inZ)
{
this->mEnemyBlips.AddBlip(inX, inY, inZ);
}
void AvHPlayer::AddPoints( int score, BOOL bAllowNegativeScore )
{
// Positive score always adds
if ( score < 0 )
{
if ( !bAllowNegativeScore )
{
if ( this->mScore < 0 ) // Can't go more negative
return;
if ( -score > this->mScore ) // Will this go negative?
{
score = -this->mScore; // Sum will be 0
}
}
}
this->mScore += score;
this->EffectivePlayerClassChanged();
}
void AvHPlayer::AwardKill( entvars_t* inTargetPEV)
{
// Don't award points in new resource model
//CBaseEntity* inTarget = CBaseEntity::Instance(inTargetPEV);
//GetGameRules()->AwardPointsToPlayer(this, inTarget);
}
int AvHPlayer::BloodColor(void)
{
int theBloodColor = DONT_BLEED;
if(this->GetIsMarine() && !this->GetHasHeavyArmor())
{
theBloodColor = BLOOD_COLOR_RED;
}
else if(this->GetIsAlien())
{
theBloodColor = BLOOD_COLOR_YELLOW;
}
return theBloodColor;
}
void AvHPlayer::AcquireOverwatchTarget()
{
ASSERT(this->mInOverwatch);
// Find closest enemy in FOV
// Find cockroaches for now
CBaseEntity* theCurrentTarget = NULL;
float theCurrentRange = 1000000;
//FOR_ALL_ENTITIES(kRoachClassName, CRoach*)
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*)
if(this->pev->team != theEntity->pev->team)
{
float theDistance = (theEntity->pev->origin - this->pev->origin).Length();
if(this->GetIsEntityInSight(theEntity) && (theDistance <= theCurrentRange) && theEntity->GetIsRelevant())
{
theCurrentTarget = theEntity;
}
}
END_FOR_ALL_ENTITIES(kAvHPlayerClassName)
//END_FOR_ALL_ENTITIES(kRoachClassName)
if(theCurrentTarget)
{
this->mOverwatchTarget = ENTINDEX(theCurrentTarget->edict());
this->pev->fuser1 = this->mOverwatchTarget;
// Playback target event
//PLAYBACK_EVENT_FULL(0, this->edict(), gTargetOverwatchEventID, 0, this->pev->origin, (float *)&g_vecZero, 0.0, 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
}
}
bool AvHPlayer::AttemptToBuildAlienStructure(AvHMessageID inMessageID)
{
bool theSuccess = false;
string theErrorMessage;
int theBuildingPointCost = 0;
bool thePurchaseAllowed = this->GetPurchaseAllowed(inMessageID, theBuildingPointCost, &theErrorMessage);
if(thePurchaseAllowed)
{
if(inMessageID == ALIEN_BUILD_HIVE)
{
// See if there is an inactive hive within range
UTIL_MakeVectors(this->pev->v_angle);
Vector theStart = this->GetGunPosition();
Vector theEnd = theStart + gpGlobals->v_forward*50;
// Collide with world to find potential build site
TraceResult theTR;
UTIL_TraceLine(theStart, theEnd, dont_ignore_monsters, this->edict(), &theTR);
Vector theLocation = theTR.vecEndPos;
// Do we have enough points?
if(AvHSHUGetIsSiteValidForBuild(ALIEN_BUILD_HIVE, &theLocation, this->entindex()))
{
// Get the hive at this location
CBaseEntity* theBaseEntity = NULL;
AvHHive* theNearestHive = NULL;
// Find the nearest hive
while((theBaseEntity = UTIL_FindEntityByClassname(theBaseEntity, kesTeamHive)) != NULL)
{
if(theBaseEntity)
{
AvHHive* theCurrentHive = dynamic_cast<AvHHive*>(theBaseEntity);
if(theCurrentHive)
{
float theCurrentDistance = VectorDistance(theLocation, theCurrentHive->pev->origin);
if(!theNearestHive || (theCurrentDistance < VectorDistance(theLocation, theNearestHive->pev->origin)))
{
theNearestHive = theCurrentHive;
}
}
}
}
if(theNearestHive)
{
// Make sure another hive isn't already building for this team
bool theAnotherHiveBuilding = false;
FOR_ALL_ENTITIES(kesTeamHive, AvHHive*)
if((theEntity->pev->team == this->pev->team) && theEntity->GetIsSpawning())
{
theAnotherHiveBuilding = true;
break;
}
END_FOR_ALL_ENTITIES(kesTeamHive);
if(!theAnotherHiveBuilding)
{
// If so, set it as growing
if(theNearestHive->StartSpawningForTeam(this->GetTeam()))
{
AvHSUBuildingJustCreated(inMessageID, theNearestHive, this);
theSuccess = true;
}
else
{
this->SendMessage(kHelpTextHiveBlocked);
}
}
else
{
this->SendMessage(kHelpTextOtherHiveBuilding);
}
}
}
else
{
this->SendMessage(kHelpTextEmptyHiveNotNearby);
}
}
else
{
char* theClassName = NULL;
if(AvHSHUGetBuildTechClassName(inMessageID, theClassName))
{
// Make sure we haven't exceeded the structure limit
int theNumBuildings = AvHSUGetStructureCount(inMessageID);
// Now check to make sure the space is big enough to hold the building
UTIL_MakeVectors(this->pev->v_angle);
const int kAimRange = 48;
Vector theStart = this->GetGunPosition();
Vector theEnd = theStart + gpGlobals->v_forward*kAimRange;
// Collide with world to find potential build site
TraceResult theTR;
UTIL_TraceLine(theStart, theEnd, dont_ignore_monsters, this->edict(), &theTR);
Vector theLocation = theTR.vecEndPos;
// Check if collision point is valid for building
if(AvHSHUGetIsSiteValidForBuild(inMessageID, &theLocation))
{
if (theNumBuildings < GetGameRules()->GetStructureLimit())
{
// Make sure there aren't too many buildings in this area already
int theNumBuildingsNearby = UTIL_CountEntitiesInSphere(theLocation, BALANCE_VAR(kBuildingVisibilityRadius), theClassName);
if(theNumBuildingsNearby < BALANCE_VAR(kNumSameAlienStructuresAllowedInRadius) || FStrEq(theClassName, kwsAlienResourceTower))//voogru: allow the building of rt's regardless of how many may be close by (for maps that have a lot of nodes close to each other)
{
// Create the new building
CBaseEntity* theEntity = CBaseEntity::Create(theClassName, theLocation, AvHSUGetRandomBuildingAngles());
// Set building's team
theEntity->pev->team = this->pev->team;
AvHSUBuildingJustCreated(inMessageID, theEntity, this);
// Set owner (this prevents collisions between the entity and it's owner though)
//theEntity->pev->owner = ENT(this->pev);
//voogru: I've moved this here because whats the point of playing the sound if the building didnt get placed? (it was after " Vector theLocation = theTR.vecEndPos;")
// Play sound
char* theSoundEffect = kAlienBuildingSound1;
if(RANDOM_LONG(0, 1) == 1)
theSoundEffect = kAlienBuildingSound2;
EMIT_SOUND(this->edict(), CHAN_AUTO, theSoundEffect, this->GetAlienAdjustedEventVolume(), ATTN_NORM);
theSuccess = true;
}
else
{
this->SendMessage(kTooManyStructuresOfThisTypeNearby);
}
}
else
{
this->SendMessage(kTooManyStructuresOnServer);
}
}
else
{
// Play hive easter egg sometimes when trying to build a hive where one already exists
if(inMessageID == ALIEN_BUILD_HIVE)
{
//// If we are close to a hive we already own
//AvHHive* theHive = AvHSUGetRandomActiveHive((AvHTeamNumber)this->pev->team);
//if(theHive)
//{
// float theDistance = VectorDistance(theHive->pev->origin, this->pev->origin);
// if(theDistance <= 300)
// {
// // Randomly play easter egg
// if(RANDOM_LONG(0, 10) == 0)
// {
// EMIT_SOUND(this->edict(), CHAN_AUTO, kMyHiveEasterEgg, 1.0f, ATTN_NORM);
// }
// }
//}
}
}
}
else
{
this->PlayHUDSound(HUD_SOUND_ALIEN_MORE);
}
}
}
else
{
this->SendMessage(theErrorMessage.c_str());
}
return theSuccess;
}
bool AvHPlayer::BuildTech(AvHMessageID inBuildID, const Vector& inPickRay)
{
bool theSuccess = false;
//AvHSUSetIsDebugging(true);
// If valid
if(AvHSHUGetIsBuildTech(inBuildID))
{
// Make sure this is a valid place to build
Vector theLocation;
if(AvHSHUTraceAndGetIsSiteValidForBuild(inBuildID, this->GetVisualOrigin(), inPickRay, &theLocation))
{
// puzl: 1097 Commander created entities now created 4 units above the scan location and they drop to the floor.
if ( this->GetIsInTopDownMode() ) {
theLocation[2]+=4;
}
// Decrement resources
string theErrorMessage;
int theCost = 0;
bool thePurchaseAllowed = this->GetPurchaseAllowed(inBuildID, theCost, &theErrorMessage);
if(thePurchaseAllowed)
{
// Count how many entities on our team we have in area
int theNumFriendlyEntitiesInArea = 0;
CBaseEntity* theEntity = NULL;
while((theEntity = UTIL_FindEntityInSphere(theEntity, theLocation, BALANCE_VAR(kBuildingVisibilityRadius))) != NULL)
{
// Don't count players
if(!theEntity->IsPlayer() && (theEntity->pev->team == this->pev->team))
{
theNumFriendlyEntitiesInArea++;
}
}
// Make sure we haven't exceeded the structure limit
int theNumBuildings = AvHSUGetStructureCount(inBuildID);
if(theNumBuildings < GetGameRules()->GetStructureLimit())
{
if(theNumFriendlyEntitiesInArea < BALANCE_VAR(kMaxMarineEntitiesAllowedInRadius))
{
// Build it!
theSuccess = (AvHSUBuildTechForPlayer(inBuildID, theLocation, this) != NULL);
// Inform structure about build if possible
if(theSuccess)
{
if(this->mSelected.size() > 0)
{
// Get selected structure and inform
int theFirstEntitySelected = *this->mSelected.begin();
AvHBaseBuildable* theBaseBuildable = dynamic_cast<AvHBaseBuildable*>(CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(theFirstEntitySelected)));
if(theBaseBuildable)
{
theBaseBuildable->TechnologyBuilt(inBuildID);
}
}
this->PayPurchaseCost(theCost);
}
}
else
{
this->SendMessage(kTooManyStructuresInArea);
}
}
else
{
this->SendMessage(kTooManyStructuresOnServer);
}
}
else
{
this->PlayHUDSound(HUD_SOUND_MARINE_MORE);
}
}
}
//AvHSUSetIsDebugging(false);
// If successful
// Send confirmation if needed
// else
// Send failure so client is updated
return theSuccess;
}
bool AvHPlayer::GroupMessage(AvHMessageID inGroupMessage)
{
bool theSuccess = false;
if((inGroupMessage >= GROUP_CREATE_1) && (inGroupMessage < (GROUP_CREATE_1 + kNumHotkeyGroups)))
{
int theOffset = (int)(inGroupMessage - GROUP_CREATE_1);
ASSERT(theOffset >= 0);
ASSERT(theOffset < kNumHotkeyGroups);
if(this->mSelected.size() > 0)
{
this->GetTeamPointer()->SetGroup(theOffset, this->mSelected);
AvHUser3 theUser3 = this->GetTeamPointer()->GetGroupType(theOffset);
AvHHUDSound theHudSound = HUD_SOUND_SELECT;
if(theUser3 == AVH_USER3_MARINE_PLAYER)
{
theHudSound = AvHHUDSound(HUD_SOUND_SQUAD1 + theOffset);
}
this->PlayHUDSound(theHudSound);
// If this is a squad, tell all the players in the squad also. This also tells them they are in the squad.
if(theUser3 == AVH_USER3_MARINE_PLAYER)
{
// Loop through them
for(EntityListType::iterator theIter = this->mSelected.begin(); theIter != this->mSelected.end(); theIter++)
{
AvHPlayer* thePlayer = NULL;
AvHSUGetEntityFromIndex(*theIter, thePlayer);
if(thePlayer)
{
thePlayer->PlayHUDSound(theHudSound);
}
}
}
// Now run through all the hotgroups and remove these entities from them (entities can only be part of one group)
for(int theCurrentIndex = 0; theCurrentIndex < kNumHotkeyGroups; theCurrentIndex++)
{
if(theCurrentIndex != theOffset)
{
//this->mGroups[theCurrentIndex];
EntityListType theCurrentGroup = this->GetTeamPointer()->GetGroup(theCurrentIndex);
// Remove all members of mSelected from this group
for(EntityListType::iterator theIterator = theCurrentGroup.begin(); theIterator != theCurrentGroup.end(); /* nothing */)
{
// If the current entity is in selection
EntityListType::iterator theFindIter = std::find(this->mSelected.begin(), this->mSelected.end(), *theIterator);
bool theEntityInSelection = (theFindIter != this->mSelected.end());
if(theEntityInSelection)
{
// Remove it from this list
theIterator = theCurrentGroup.erase(theIterator);
}
else
{
// Else, increment
theIterator++;
}
}
// Set the group again
this->GetTeamPointer()->SetGroup(theCurrentIndex, theCurrentGroup);
}
}
}
theSuccess = true;
}
else if((inGroupMessage >= GROUP_SELECT_1) && (inGroupMessage < (GROUP_SELECT_1 + kNumHotkeyGroups)))
{
int theOffset = (int)(inGroupMessage - GROUP_SELECT_1);
ASSERT(theOffset >= 0);
ASSERT(theOffset < kNumHotkeyGroups);
//const EntityListType& theGroup = this->mGroups[theOffset];
EntityListType theGroup = this->GetTeamPointer()->GetGroup(theOffset);
if(theGroup.size() > 0)
{
if(this->mSelected != theGroup)
{
this->mSelected = theGroup;
this->mTrackingEntity = 0;
}
else
{
// If we received the same select message twice in a row, go instead to the last place we went to
if(inGroupMessage == this->mLastSelectEvent)
{
// Go to last saved position
VectorCopy(this->mPositionBeforeLastGotoGroup, this->pev->origin);
// Clear last select so hitting it again will go to group
this->mLastSelectEvent = MESSAGE_NULL;
}
else
{
// Find nearest entity in group and track it
int theNearestGroupEntity = 0;
float theClosestDistance = sqrt(2*kMaxMapDimension*kMaxMapDimension);
for(EntityListType::const_iterator theIter = theGroup.begin(); theIter != theGroup.end(); theIter++)
{
CBaseEntity* theBaseEntity = CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(*theIter));
ASSERT(theBaseEntity);
float theCurrentDistance = VectorDistance(theBaseEntity->pev->origin, this->pev->origin);
if(theCurrentDistance < theClosestDistance)
{
theNearestGroupEntity = *theIter;
theClosestDistance = theCurrentDistance;
}
}
this->mTrackingEntity = theNearestGroupEntity;
// Move player in vicinity of player so entity is in PVS
if(this->mTrackingEntity > 0)
{
CBaseEntity* theBaseEntity = CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(this->mTrackingEntity));
if(theBaseEntity)
{
// Save last position so we can jump back to it easily
VectorCopy(this->pev->origin, this->mPositionBeforeLastGotoGroup);
this->mLastSelectEvent = inGroupMessage;
// Goto group
VectorCopy(theBaseEntity->pev->origin, this->pev->origin);
}
}
}
}
}
theSuccess = true;
}
return theSuccess;
}
bool AvHPlayer::GetCenterPositionForGroup(int inGroupNumber, float& outX, float& outY)
{
bool theSuccess = false;
if((inGroupNumber >= 1) && (inGroupNumber <= kNumHotkeyGroups))
{
int theNumFound = 0;
Vector theLocation;
theLocation.x = theLocation.y = theLocation.z = 0.0f;
//EntityListType& theGroup = this->mGroups[inGroupNumber-1];
EntityListType theGroup = this->GetTeamPointer()->GetGroup(inGroupNumber-1);
for(EntityListType::iterator theIter = theGroup.begin(); theIter != theGroup.end(); theIter++)
{
Vector theEntityLocation;
if(AvHSHUGetEntityLocation(*theIter, theEntityLocation))
{
theLocation.x += theEntityLocation.x;
theLocation.y += theEntityLocation.y;
theNumFound++;
}
}
if(theNumFound > 0)
{
outX = theLocation.x/theNumFound;
outY = theLocation.y/theNumFound;
theSuccess = true;
}
}
return theSuccess;
}
string AvHPlayer::GetNetworkID() const
{
return this->mNetworkID;
}
void AvHPlayer::SetNetworkID(string& inNetworkID)
{
this->mNetworkID = inNetworkID;
}
string AvHPlayer::GetNetworkAddress() const
{
return this->mNetworkAddress;
}
void AvHPlayer::SetNetworkAddress(string& inNetworkAddress)
{
this->mNetworkAddress = inNetworkAddress;
}
void AvHPlayer::ClearRoleAbilities()
{
// Clear all abilities that don't stick around between changing role, like all alien purchaseable upgrades
//SetUpgradeMask(&this->pev->iuser4, ALIEN_ABILITY_2, false);
this->pev->iuser3 = AVH_USER3_NONE;
this->pev->iuser4 &= ~MASK_ALIEN_EMBRYO;
this->pev->iuser4 &= ~MASK_ALIEN_MOVEMENT;
this->mIsScreaming = false;
this->mAlienSightActive = false;
this->mDesiredRoomType = 0;
}
void AvHPlayer::ClearUserVariables()
{
this->pev->iuser1 = 0;
this->pev->iuser2 = 0;
this->pev->iuser3 = 0;
this->pev->iuser4 = 0;
this->pev->fuser1 = 0;
this->pev->fuser2 = 0;
this->pev->fuser3 = 0;
this->pev->fuser4 = 0;
}
void AvHPlayer::SetScore(int inScore)
{
this->mScore = inScore;
}
int AvHPlayer::GetScore() const
{
return this->mScore;
}
CBaseEntity* AvHPlayer::CreateAndDrop(const char* inItemName)
{
UTIL_MakeVectors(pev->v_angle);
CBaseEntity* theEntity = CBaseEntity::Create(inItemName, pev->origin + gpGlobals->v_forward * 10, pev->angles, edict() );
theEntity->pev->angles.x = 0;
theEntity->pev->angles.z = 0;
//theEntity->PackWeapon( pWeapon );
theEntity->pev->velocity = gpGlobals->v_forward * 300 + gpGlobals->v_forward * 100;
return theEntity;
}
void AvHPlayer::DeployCurrent()
{
if(this->m_pActiveItem /*&& !this->GetIsBeingDigested()*/)
{
CBasePlayerWeapon* theCurrentWeapon = (CBasePlayerWeapon *)this->m_pActiveItem->GetWeaponPtr();
if(theCurrentWeapon && theCurrentWeapon->CanDeploy())
{
theCurrentWeapon->Deploy();
}
}
else
{
// No current weapon, so get the best weapon we have.
g_pGameRules->GetNextBestWeapon( this, NULL );
}
}
// Drop the current item, not weapon, if any.
bool AvHPlayer::DropItem(const char* inItemName)
{
bool theSuccess = false;
if(!GetGameRules()->GetIsCombatMode())
{
CBasePlayerItem* theItem = this->m_pActiveItem;
if(inItemName)
{
bool theIsDone = false;
theItem = NULL; //we're looking for an item by name, don't bias with active item
// Find item with this name
for(int i = 0; (i < MAX_ITEM_TYPES) && !theIsDone; i++)
{
CBasePlayerItem* theCurrentItem = this->m_rgpPlayerItems[i];
while(theCurrentItem && !theIsDone)
{
if(FStrEq(STRING(theCurrentItem->pev->classname), inItemName))
{
theIsDone = true;
theItem = theCurrentItem;
}
theCurrentItem = theCurrentItem->m_pNext;
}
}
}
if(theItem)
{
AvHBasePlayerWeapon* theOriginalDroppedWeapon = dynamic_cast<AvHBasePlayerWeapon*>(theItem);
if(theOriginalDroppedWeapon && theOriginalDroppedWeapon->GetIsDroppable())
{
// Dropping current weapon turns off overwatch
this->TurnOffOverwatch();
//char theItemName[256];
//strcpy(theItemName, STRING(theItem->pev->classname));
//this->DropPlayerItem(theItemName);
// Get ammo left, if it's a weapon
int theAmmoLeft = -1;
if(theOriginalDroppedWeapon)
{
theAmmoLeft = theOriginalDroppedWeapon->m_iClip;
GetGameRules()->GetNextBestWeapon(this, theItem);
}
CBaseEntity* theDroppedEntity = this->CreateAndDrop(STRING(theItem->pev->classname));
if(theAmmoLeft != -1)
{
CBasePlayerWeapon* theNewlyDroppedWeapon = dynamic_cast<CBasePlayerWeapon*>(theDroppedEntity);
ASSERT(theNewlyDroppedWeapon);
// Set ammo, make sure client ammo isn't the same so update is forced
theNewlyDroppedWeapon->m_iClip = theAmmoLeft;
theNewlyDroppedWeapon->m_iClientClip = theAmmoLeft - 1;
// Means "only ammo is that in the clip"
theNewlyDroppedWeapon->m_iDefaultAmmo = 0;
ItemInfo theItemInfo;
if(theOriginalDroppedWeapon->GetItemInfo(&theItemInfo) != 0)
{
int iAmmoIndex = GetAmmoIndex ( (char *) theItemInfo.pszAmmo1 );
if ( theItemInfo.iSlot == 0 )
this->EffectivePlayerClassChanged();
if ( iAmmoIndex != -1 && m_rgAmmo[ iAmmoIndex ] > 0)
this->DropAmmo((char *)theItemInfo.pszAmmo1, m_rgAmmo[ iAmmoIndex ], theItemInfo.iMaxAmmo1, theItemInfo.iId, theNewlyDroppedWeapon->pev->angles);
}
}
// Finally remove the original
theOriginalDroppedWeapon->DestroyItem();
// Weak potential fix for occasional crash. Saw this when someone hit "lastinv" while holding a knife,
// and m_pLastItem was garbage, not NULL
this->m_pLastItem = NULL;
SetAnimation(PLAYER_PRIME); //Elven - hack to kill TPRA if we throw a weapon.
theSuccess = true;
}
}
}
return theSuccess;
}
void AvHPlayer :: DropAmmo(char *pszAmmoType, int iAmmoAmt, int iMax, int iWeaponID, Vector vecAngles)
{
if(!pszAmmoType || !iAmmoAmt || !iMax || !iWeaponID)
return;
int iAmmoIndex = GetAmmoIndex ( pszAmmoType );
//Not a valid ammo type.
if(iAmmoIndex == -1)
return;
if(m_rgAmmo[ iAmmoIndex ] > 0)
m_rgAmmo[ iAmmoIndex ] -= iAmmoAmt;
else
return;
int theSubModel = -1;
switch(iWeaponID)
{
case AVH_WEAPON_PISTOL: theSubModel = 3; break;
case AVH_WEAPON_MG: theSubModel = 1; break;
case AVH_WEAPON_SONIC: theSubModel = 0; break;
case AVH_WEAPON_HMG: theSubModel = 2; break;
case AVH_WEAPON_GRENADE_GUN: theSubModel = 4; break;
}
if(theSubModel == -1)
return;
CBaseEntity *theDroppedEntity = this->CreateAndDrop( kwsAmmoPack );
AvHAmmoPack *theRealAmmoPack = dynamic_cast<AvHAmmoPack*>(theDroppedEntity);
strncpy(theRealAmmoPack->m_szAmmoType, pszAmmoType, 31);
theRealAmmoPack->m_iMaxAmmo = iMax;
theRealAmmoPack->m_iAmmoAmt = iAmmoAmt;
theRealAmmoPack->m_iWeaponID = iWeaponID;
theRealAmmoPack->pev->body = theSubModel;
theRealAmmoPack->pev->angles = vecAngles;
theRealAmmoPack->SetThink( &CBaseEntity::SUB_Remove );
theRealAmmoPack->pev->nextthink = gpGlobals->time + AvHSUGetWeaponStayTime();
}
void AvHPlayer::EffectivePlayerClassChanged()
{
this->mEffectivePlayerClassChanged = true;
}
void AvHPlayer::NeedsTeamUpdate()
{
this->mNeedsTeamUpdate = true;
}
void AvHPlayer::SendTeamUpdate()
{
this->mSendTeamUpdate = true;
}
bool AvHPlayer::ExecuteAlienMorphMessage(AvHMessageID inMessageID, bool inInstantaneous)
{
bool theMessageExecuted = false;
string theErrorMessage;
bool theHadEnoughPoints = false;
switch(inMessageID)
{
case ALIEN_LIFEFORM_ONE:
case ALIEN_LIFEFORM_TWO:
case ALIEN_LIFEFORM_THREE:
case ALIEN_LIFEFORM_FOUR:
case ALIEN_LIFEFORM_FIVE:
case ALIEN_EVOLUTION_ONE:
case ALIEN_EVOLUTION_TWO:
case ALIEN_EVOLUTION_THREE:
//case ALIEN_EVOLUTION_FOUR:
//case ALIEN_EVOLUTION_FIVE:
//case ALIEN_EVOLUTION_SIX:
case ALIEN_EVOLUTION_SEVEN:
case ALIEN_EVOLUTION_EIGHT:
case ALIEN_EVOLUTION_NINE:
case ALIEN_EVOLUTION_TEN:
case ALIEN_EVOLUTION_ELEVEN:
case ALIEN_EVOLUTION_TWELVE:
case ALIEN_HIVE_TWO_UNLOCK:
case ALIEN_HIVE_THREE_UNLOCK:
// Now only allow upgrading from level1
if(this->GetCanGestate(inMessageID, theErrorMessage))
{
// Stay as current lifeform by default
bool theCheckDucking = true;
int theTargetIuser3 = this->pev->iuser3;
switch(inMessageID)
{
case ALIEN_LIFEFORM_ONE:
theTargetIuser3 = AVH_USER3_ALIEN_PLAYER1;
break;
case ALIEN_LIFEFORM_TWO:
theTargetIuser3 = AVH_USER3_ALIEN_PLAYER2;
break;
case ALIEN_LIFEFORM_THREE:
theTargetIuser3 = AVH_USER3_ALIEN_PLAYER3;
break;
case ALIEN_LIFEFORM_FOUR:
theTargetIuser3 = AVH_USER3_ALIEN_PLAYER4;
break;
case ALIEN_LIFEFORM_FIVE:
theTargetIuser3 = AVH_USER3_ALIEN_PLAYER5;
theCheckDucking = false;
break;
}
int theTargetHull = AvHMUGetHull(theCheckDucking, theTargetIuser3);
vec3_t theOrigin;
GetNewOrigin((AvHUser3)theTargetIuser3, theCheckDucking, theOrigin);
// removed by puzl to fix gestating in vents.
//theOrigin.z += 5;
bool theIsEnoughRoom = AvHSUGetIsEnoughRoomForHull(theOrigin, theTargetHull, this->edict());
//voogru: try again but higher
if(!theIsEnoughRoom)
{
theOrigin.z += AvHMUGetOriginOffsetForMessageID(inMessageID);
theIsEnoughRoom = AvHSUGetIsEnoughRoomForHull(theOrigin, theTargetHull, this->edict());
}
if(theIsEnoughRoom || inInstantaneous)
{
if(!theIsEnoughRoom)
{
int a = 0;
}
this->Evolve(inMessageID, inInstantaneous);
theMessageExecuted = true;
}
else
{
this->SendMessage(kNeedMoreRoomToGestate);
}
}
else
{
this->SendMessage(theErrorMessage.c_str());
}
break;
}
return theMessageExecuted;
}
bool AvHPlayer::ExecuteMessage(AvHMessageID inMessageID, bool inInstantaneous, bool inForce)
{
bool theMessageExecuted = false;
AvHTeam* theTeam = this->GetTeamPointer();
string theErrorMessage;
// If valid
if((inMessageID != MESSAGE_NULL) && theTeam)
{
bool theIsMarine = theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE;
bool theIsAlien = theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN;
bool theGameStarted = GetGameRules()->GetGameStarted();
bool theIsInTopDownMode = this->GetIsInTopDownMode();
bool theIsCombat = GetGameRules()->GetIsCombatMode();
bool theIsBeingDigested = this->GetIsBeingDigested();
// If we can purchase it
string theErrorMessage;
int theCost = 0;
if(this->GetPurchaseAllowed(inMessageID, theCost, &theErrorMessage) || inForce)
{
// If we're in Combat mode
bool theIsAvailable = true;
if(theIsCombat)
{
// Try to execute message
theMessageExecuted = this->ExecuteCombatMessage(inMessageID, theIsAvailable, inForce);
if(theMessageExecuted)
{
this->PurchaseCombatUpgrade(inMessageID);
}
}
// If it's a fall-through alien upgrade, try it next
if(!theMessageExecuted && theIsAlien && !theIsBeingDigested)
{
// Try to morph or build
theMessageExecuted = this->ExecuteAlienMorphMessage(inMessageID, inInstantaneous);
if(!theMessageExecuted && !theIsCombat)
{
switch(inMessageID)
{
case ALIEN_BUILD_RESOURCES:
case ALIEN_BUILD_OFFENSE_CHAMBER:
case ALIEN_BUILD_DEFENSE_CHAMBER:
case ALIEN_BUILD_SENSORY_CHAMBER:
case ALIEN_BUILD_MOVEMENT_CHAMBER:
case ALIEN_BUILD_HIVE:
theMessageExecuted = this->AttemptToBuildAlienStructure(inMessageID);
break;
}
}
}
// Add upgrade to our list to be preserved on respawn
if(theMessageExecuted && theIsCombat)
this->mCombatNodes.SetResearchDone(inMessageID);
if(theMessageExecuted)
this->PayPurchaseCost(theCost);
}
// tankefugl: 0000971
int theIssuedOrderIcon = -1;
// :tankefugl
if(theIsMarine
&& !theIsInTopDownMode
&& !theIsBeingDigested)
{
switch (inMessageID)
{
case WEAPON_RELOAD:
if(theGameStarted)
this->ReloadWeapon();
break;
case WEAPON_DROP:
if(!this->DropItem())
this->SendMessageOnce(kWeaponCantBeDropped, TOOLTIP);
break;
case ADMIN_VOTEDOWNCOMMANDER:
if(theTeam->PlayerVote(this->entindex(), theErrorMessage))
theMessageExecuted = true;
else
this->SendMessage(theErrorMessage.c_str());
break;
// Talk to your teammates
case SAYING_1:
case SAYING_2:
case SAYING_3:
case SAYING_4:
case SAYING_5:
case SAYING_6:
case SAYING_7:
case SAYING_8:
case SAYING_9:
case ORDER_REQUEST:
case ORDER_ACK:
if(this->PlaySaying(inMessageID))
{
AvHAlertType theAlertType = ALERT_NONE;
AvHMessageID theMessageID = MESSAGE_NULL;
if(this->GetIsMarine())
{
switch(inMessageID)
{
// tankefugl: 0000971
// decides whether icon updates should be sent
case SAYING_1:
theIssuedOrderIcon = TEAMMATE_MARINE_ORDER_FOLLOW;
break;
case SAYING_2:
theIssuedOrderIcon = TEAMMATE_MARINE_ORDER_COVER;
break;
case SAYING_8:
theIssuedOrderIcon = TEAMMATE_MARINE_ORDER_WELD;
break;
// :tankefugl
case SAYING_5:
theAlertType = ALERT_SOLDIER_NEEDS_AMMO;
theMessageID = COMMANDER_NEXTAMMO;
break;
case SAYING_4:
theAlertType = ALERT_SOLDIER_NEEDS_HEALTH;
theMessageID = COMMANDER_NEXTHEALTH;
break;
case ORDER_REQUEST:
theAlertType = ALERT_ORDER_NEEDED;
theMessageID = COMMANDER_NEXTIDLE;
break;
}
// Send alert for commander
if(theAlertType != ALERT_NONE)
{
GetGameRules()->TriggerAlert((AvHTeamNumber)this->pev->team, theAlertType, this->entindex(), theMessageID);
}
}
theMessageExecuted = true;
}
break;
}
}
else if(theIsAlien && !theIsBeingDigested)
{
switch (inMessageID)
{
// Talk to your teammates
case SAYING_1:
case SAYING_2:
case SAYING_3:
case SAYING_4:
case SAYING_5:
case SAYING_6:
case SAYING_7:
case SAYING_8:
case SAYING_9:
if(this->PlaySaying(inMessageID))
theMessageExecuted = true;
break;
case ALIEN_ABILITY_LEAP:
this->StartLeap();
break;
case 100:
// Eat flashlight event. Add special mode for alien view here?
if(!this->mAlienSightActive)
PLAYBACK_EVENT_FULL(FEV_HOSTONLY, this->edict(), gAlienSightOnEventID, 0, this->pev->origin, (float *)&g_vecZero, this->GetAlienAdjustedEventVolume(), 0.0, 0, 0, 0, 0 );
else
PLAYBACK_EVENT_FULL(FEV_HOSTONLY, this->edict(), gAlienSightOffEventID, 0, this->pev->origin, (float *)&g_vecZero, this->GetAlienAdjustedEventVolume(), 0.0, 0, 0, 0, 0 );
this->mAlienSightActive = !this->mAlienSightActive;
theMessageExecuted = true;
break;
}
// tankefugl: 0000971
// decides whether icon updates should be sent
switch (inMessageID)
{
case SAYING_1:
theIssuedOrderIcon = TEAMMATE_ALIEN_ORDER_FOLLOW;
break;
case SAYING_2:
theIssuedOrderIcon = TEAMMATE_ALIEN_ORDER_COVER;
break;
case SAYING_4:
case SAYING_8:
theIssuedOrderIcon = TEAMMATE_ALIEN_ORDER_HEAL;
break;
}
// :tankefugl
}
// tankefugl: 0000971 and 0000992
if (theIssuedOrderIcon > -1)
{
int theOrderTarget = 0;
vec3_t vecDir;
VectorCopy(this->GetAutoaimVector(0.0f), vecDir);
VectorNormalize(vecDir);
float currentResult = 0.0f;
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*);
float dotResult = 0.0f;
float theDistance = 0.0f;
vec3_t vecDistance;
int theTraced = 0;
vec3_t vecFrom, vecTo;
if ((theEntity->entindex() != this->entindex()) && (theEntity->GetTeam() == this->GetTeam()))
{
VectorSubtract(theEntity->pev->origin, this->pev->origin, vecDistance);
// theDistance = Length(vecDistance);
VectorNormalize(vecDistance);
dotResult = DotProduct(vecDistance, vecDir);
if ((dotResult > 0.9f) && (dotResult > currentResult))
{
TraceResult theTrace;
vecFrom = AvHSHUGetRealLocation(this->pev->origin, this->pev->mins, this->pev->maxs);
vecTo = AvHSHUGetRealLocation(theEntity->pev->origin, theEntity->pev->mins, theEntity->pev->maxs);
UTIL_TraceLine(vecFrom, vecTo, ignore_monsters, this->edict(), &theTrace);
if (theTrace.flFraction == 1.0f)
{
theTraced = 1;
currentResult = dotResult;
theOrderTarget = theEntity->entindex();
}
}
}
//ALERT(at_console, "-------------------\n");
//ALERT(at_console, UTIL_VarArgs("vecDir %f %f %f\n", vecDir[0], vecDir[1], vecDir[2]));
//ALERT(at_console, UTIL_VarArgs("vecDistance %f %f %f\n", vecDistance[0], vecDistance[1], vecDistance[2]));
//ALERT(at_console, UTIL_VarArgs("vecFrom %f %f %f\n", vecFrom[0], vecFrom[1], vecFrom[2]));
//ALERT(at_console, UTIL_VarArgs("vecTo %f %f %f\n", vecTo[0], vecTo[1], vecTo[2]));
//ALERT(at_console, UTIL_VarArgs("dotResult %f\n", dotResult));
//ALERT(at_console, UTIL_VarArgs("currentResult %f\n", currentResult));
//ALERT(at_console, UTIL_VarArgs("theTraced %d\n", theTraced));
//ALERT(at_console, UTIL_VarArgs("theOrderTarget %d\n", theOrderTarget));
END_FOR_ALL_ENTITIES(kAvHPlayerClassName);
// ALERT(at_console, UTIL_VarArgs("theIssuedOrderIcon %d source %d target %d\n", theIssuedOrderIcon, this->entindex(), theOrderTarget));
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*);
if(theEntity->GetTeam() == this->GetTeam())
{
NetMsg_IssueOrder(theEntity->pev, theIssuedOrderIcon, this->entindex(), theOrderTarget);
}
END_FOR_ALL_ENTITIES(kAvHPlayerClassName);
}
// Common messages here
if(!theMessageExecuted)
{
switch(inMessageID)
{
case WEAPON_NEXT:
this->NextWeapon();
theMessageExecuted = true;
break;
case ADMIN_READYROOM:
if(this->GetPlayMode() != PLAYMODE_READYROOM)
{
this->SetPlayMode(PLAYMODE_READYROOM);
theMessageExecuted = true;
}
break;
}
}
}
return theMessageExecuted;
}
bool AvHPlayer::GetIsAlienSightActive() const
{
return this->mAlienSightActive;
}
void AvHPlayer::SetDesiredNetName(string inDesiredNetName)
{
this->mDesiredNetName = inDesiredNetName;
}
void AvHPlayer::SetDesiredRoomType(int inRoomType, bool inForceUpdate)
{
this->mDesiredRoomType = inRoomType;
if(inForceUpdate)
{
this->mClientDesiredRoomType = -1;
}
}
void AvHPlayer::ResetPlayerPVS()
{
// Force recompute of PVS
::ResetPlayerPVS(this->edict(), this->entindex());
}
void AvHPlayer::SetPosition(const Vector& inNewPosition)
{
// Set new position
VectorCopy(inNewPosition, this->pev->origin);
this->ResetPlayerPVS();
}
// Play sounds quieter when the alien has the silence upgrade
float AvHPlayer::GetAlienAdjustedEventVolume() const
{
return AvHPlayerUpgrade::GetSilenceVolumeLevel((AvHUser3)this->pev->iuser3, this->pev->iuser4);
}
void AvHPlayer::GetAnimationForActivity(int inActivity, char outAnimation[64], bool inGaitSequence)
{
strcpy(outAnimation, "");
bool theCanCrouch = !((this->GetUser3() == AVH_USER3_ALIEN_PLAYER1) || (this->GetUser3() == AVH_USER3_ALIEN_PLAYER2) || (this->GetUser3() == AVH_USER3_ALIEN_PLAYER3));
bool theIsCrouched = FBitSet(this->pev->flags, FL_DUCKING);
bool theIsAlien = this->GetIsAlien();
bool theIsGestating = (this->GetUser3() == AVH_USER3_ALIEN_EMBRYO);
bool theIsDeathAnim = false;
bool theIsReloading = false;
int theDebugAnimations = BALANCE_VAR(kDebugAnimations);
//bool theIsBlinking = this->GetIsBlinking();
switch(inActivity)
{
case ACT_RELOAD:
if(!theIsAlien)
{
strcat(outAnimation, this->m_szAnimExtention);
strcat(outAnimation, "_reload");
theIsReloading = true;
};
break;
// updated by Elven for TPRAs
case ACT_RELOAD_START:
if(!theIsAlien)
{
strcat(outAnimation, this->m_szAnimExtention);
strcat(outAnimation, "_reload_start");
theIsReloading = true;
};
break;
case ACT_RELOAD_INSERT:
if(!theIsAlien)
{
strcat(outAnimation, this->m_szAnimExtention);
strcat(outAnimation, "_reload_insert");
theIsReloading = true;
};
break;
case ACT_RELOAD_END:
if(!theIsAlien)
{
strcat(outAnimation, this->m_szAnimExtention);
strcat(outAnimation, "_reload_end");
theIsReloading = true;
};
break;
case ACT_RANGE_PRIME:
if(theCanCrouch && theIsCrouched)
{
strcpy(outAnimation, "crouch_" );
}
strcat(outAnimation, this->m_szAnimExtention);
strcat(outAnimation, "_prime");
break;
case ACT_RANGE_ATTACK1:
if(theCanCrouch && theIsCrouched)
{
strcpy(outAnimation, "crouch_" );
}
strcat(outAnimation, this->m_szAnimExtention);
if(theIsAlien)
{
strcat(outAnimation, "_alien");
}
else
{
strcat(outAnimation, "_fire");
}
break;
case ACT_HOP:
strcat(outAnimation, "jump");
break;
case ACT_WALK:
if(inGaitSequence || !strcmp(this->m_szAnimExtention, ""))
{
strcat(outAnimation, "walk");
}
else
{
if(theCanCrouch && theIsCrouched)
{
strcpy(outAnimation, "crouch_" );
}
strcat(outAnimation, this->m_szAnimExtention);
if(theCanCrouch)
{
if(theIsReloading)
{
strcat(outAnimation, "_reload");
}
else
{
strcat(outAnimation, "_look");
}
}
}
break;
case ACT_RUN:
if(inGaitSequence || !strcmp(this->m_szAnimExtention, ""))
{
strcat(outAnimation, "run");
}
else
{
if(theCanCrouch && theIsCrouched)
{
strcpy(outAnimation, "crouch_" );
}
strcat(outAnimation, this->m_szAnimExtention);
if(theCanCrouch)
{
//if(theIsReloading)
//{
// strcat(outAnimation, "_reload");
//}
//else
//{
if(theIsReloading)
{
strcpy(outAnimation, this->m_szAnimExtention);
strcat(outAnimation, "_reload");
}
else
{
strcat(outAnimation, "_look");
}
//}
}
}
break;
case ACT_CROUCHIDLE:
if(theCanCrouch)
{
if(inGaitSequence)
{
strcat(outAnimation, "crouch_idle");
}
}
break;
case ACT_CROUCH:
if(theCanCrouch)
{
if(inGaitSequence)
{
strcat(outAnimation, "crawl");
}
else
{
}
}
break;
case ACT_IDLE:
// Weird hack/fix for gyrating and ready room spazzing
//if(this->GetPlayMode() != PLAYMODE_READYROOM)
//{
if(inGaitSequence)
{
if(theIsReloading)
{
strcat(outAnimation, "_reload");
}
else
{
strcat(outAnimation, "idle1");
}
}
else
{
strcat(outAnimation, "look_idle");
}
//}
break;
case ACT_DIESIMPLE:
case ACT_DIEVIOLENT:
if(this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER3)
{
strcpy(outAnimation, "death1_die");
}
else
{
if(theIsCrouched)
{
strcpy(outAnimation, "crouch_die");
}
else
{
switch(RANDOM_LONG(0, 2))
{
case 0:
strcpy(outAnimation, "death1_die");
break;
case 1:
strcpy(outAnimation, "death2_die");
break;
case 2:
strcpy(outAnimation, "death3_die");
break;
}
}
}
theIsDeathAnim = true;
break;
case ACT_DIE_HEADSHOT:
strcpy(outAnimation, "head_die");
theIsDeathAnim = true;
break;
case ACT_DIE_GUTSHOT:
strcpy(outAnimation, "gut_die");
theIsDeathAnim = true;
break;
case ACT_DIEBACKWARD:
// Hack for skulk until artwork can be updated (it has no back_die)
if(this->pev->iuser3 != AVH_USER3_ALIEN_PLAYER1)
{
strcpy(outAnimation, "back_die");
}
else
{
strcpy(outAnimation, "gut_die");
}
theIsDeathAnim = true;
break;
case ACT_DIEFORWARD:
strcpy(outAnimation, "forward_die");
theIsDeathAnim = true;
break;
case ACT_SWIM:
// die
strcpy(outAnimation, "treadwater");
break;
}
if(theIsGestating)
{
float theFullTimeToGestate = GetGameRules()->GetBuildTimeForMessageID(this->mEvolution);
bool theAlmostDoneGestation = (gpGlobals->time > (this->mTimeGestationStarted + theFullTimeToGestate*.75f));
switch(inActivity)
{
case ACT_SMALL_FLINCH:
case ACT_BIG_FLINCH:
case ACT_FLINCH_HEAD:
case ACT_FLINCH_CHEST:
case ACT_FLINCH_STOMACH:
case ACT_FLINCH_LEFTARM:
case ACT_FLINCH_RIGHTARM:
case ACT_FLINCH_LEFTLEG:
case ACT_FLINCH_RIGHTLEG:
if(RANDOM_LONG(0, 1))
{
// flinch1
strcpy(outAnimation, "flinch1");
}
else
{
// flinch2
strcpy(outAnimation, "flinch2");
}
break;
case ACT_DIESIMPLE:
case ACT_DIEBACKWARD:
case ACT_DIEFORWARD:
case ACT_DIEVIOLENT:
// die
strcpy(outAnimation, "die");
theIsDeathAnim = true;
break;
default:
if(theAlmostDoneGestation)
{
// gestation_fast
strcpy(outAnimation, "gestation_fast");
}
else
{
// gestation
strcpy(outAnimation, "gestation");
}
break;
}
}
#ifdef DEBUG
if(theDebugAnimations && theIsDeathAnim)
{
char theMessage[256];
sprintf(theMessage, "AvHPlayer::GetAnimationForActivity(death) returned %s\n", outAnimation);
ALERT(at_console, theMessage);
}
#endif
}
bool AvHPlayer::GetCanGestate(AvHMessageID inMessageID, string& outErrorMessage)
{
bool theCanGestate = false;
int theNumHives = this->GetTeamPointer()->GetNumActiveHives();
if(this->pev->iuser3 != AVH_USER3_ALIEN_EMBRYO)
{
if(this->mDigestee <= 0)
{
bool theIsRangeRequirementMet = true;
if(theIsRangeRequirementMet)
{
theCanGestate = true;
}
else
{
// TODO: Add error message
}
}
else
{
outErrorMessage = kNotWhileDigesting;
}
}
return theCanGestate;
}
bool AvHPlayer::GetCanCommand(string& outErrorMessage)
{
bool theCanCommand = false;
// Jack up command station use time to avoid huge kick-everyone-off-server bug (I think this is an overflow issue)
// I think the overflow issue is fixed, but design-wise, this prevents annoying pop-out/pop-back in attacking (Reaver popping)
const int theCommandStationReuseTime = BALANCE_VAR(kCommandStationReuseTime);
// Have we been banned from the command station?
AvHServerPlayerData* theServerPlayerData = this->GetServerPlayerData();
if(!theServerPlayerData || (theServerPlayerData->GetTimeVotedDown() == -1))
{
//if(!this->GetCurrentWeaponCannotHolster())
if(this->m_pActiveItem)
{
CBasePlayerWeapon* theCurrentWeapon = (CBasePlayerWeapon *)this->m_pActiveItem->GetWeaponPtr();
if(theCurrentWeapon && theCurrentWeapon->CanHolster())
{
float theLastTime = this->GetLastTimeInCommandStation();
if((theLastTime == -1) || (gpGlobals->time > (theLastTime + theCommandStationReuseTime)))
{
if(!this->GetIsEnsnared())
{
theCanCommand = true;
}
}
else
{
outErrorMessage = kCommandStationWaitTime;
}
}
else
{
outErrorMessage = kWeaponPreventingCommandStation;
}
}
}
else
{
outErrorMessage = kBannedFromCommand;
}
return theCanCommand;
}
bool AvHPlayer::GetCanReceiveResources() const
{
bool theCanReceiveResources = true;
if(this->GetIsAlien())
{
theCanReceiveResources = false;
if(this->GetIsRelevant() && this->IsAlive())
{
theCanReceiveResources = true;
}
}
return theCanReceiveResources;
}
int AvHPlayer::GetEffectivePlayerClass()
{
AvHPlayerClass theEffectivePlayerClass = PLAYERCLASS_NONE;
if(this->GetUser3() == AVH_USER3_COMMANDER_PLAYER)
{
theEffectivePlayerClass = PLAYERCLASS_COMMANDER;
}
// Players can be both reinforcing and observer, but reinforcing takes precedence
else if(this->GetPlayMode() == PLAYMODE_REINFORCING)
{
theEffectivePlayerClass = PLAYERCLASS_REINFORCING;
}
// Digesting overrides other states
else if(this->GetIsBeingDigested())
{
theEffectivePlayerClass = PLAYERCLASS_ALIVE_DIGESTING;
}
else if((this->pev->team == 0) && this->IsObserver())
{
theEffectivePlayerClass = PLAYERCLASS_SPECTATOR;
}
else
{
// Check team type
if(this->GetIsMarine())
{
if(this->IsAlive())
{
if(this->GetHasHeavyArmor())
{
theEffectivePlayerClass = PLAYERCLASS_ALIVE_HEAVY_MARINE;
}
else if (this->GetHasJetpack()) {
theEffectivePlayerClass = PLAYERCLASS_ALIVE_JETPACK_MARINE;
}
else
{
theEffectivePlayerClass = PLAYERCLASS_ALIVE_MARINE;
}
}
else
{
theEffectivePlayerClass = PLAYERCLASS_DEAD_MARINE;
}
}
else if(this->GetIsAlien())
{
if(this->IsAlive())
{
switch(this->pev->iuser3)
{
case AVH_USER3_ALIEN_PLAYER1:
theEffectivePlayerClass = PLAYERCLASS_ALIVE_LEVEL1;
break;
case AVH_USER3_ALIEN_PLAYER2:
theEffectivePlayerClass = PLAYERCLASS_ALIVE_LEVEL2;
break;
case AVH_USER3_ALIEN_PLAYER3:
theEffectivePlayerClass = PLAYERCLASS_ALIVE_LEVEL3;
break;
case AVH_USER3_ALIEN_PLAYER4:
theEffectivePlayerClass = PLAYERCLASS_ALIVE_LEVEL4;
break;
case AVH_USER3_ALIEN_PLAYER5:
theEffectivePlayerClass = PLAYERCLASS_ALIVE_LEVEL5;
break;
case AVH_USER3_ALIEN_EMBRYO:
theEffectivePlayerClass = PLAYERCLASS_ALIVE_GESTATING;
break;
}
}
else
{
theEffectivePlayerClass = PLAYERCLASS_DEAD_ALIEN;
}
}
}
return theEffectivePlayerClass;
}
AvHServerPlayerData* AvHPlayer::GetServerPlayerData()
{
AvHServerPlayerData* theServerPlayerData = NULL;
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
theServerPlayerData = theTeam->GetServerPlayerData(this->edict());
}
return theServerPlayerData;
}
#ifdef USE_OLDAUTH
int AvHPlayer::GetAuthenticationMask()
{
int theMask = 0;
// Get WON id
if(this->GetAllowAuth())
{
// Use cached value if valid
theMask = this->mCachedAuthenticationMask;
// Look it up if uninitialized or Steam isn't ready yet
if((theMask == -1) || (theMask == PLAYERAUTH_PENDING))
{
string theAuthID = AvHSUGetPlayerAuthIDString(this->edict());
theMask = GetGameRules()->GetAuthenticationMask(theAuthID);
// Save cached value
this->mCachedAuthenticationMask = theMask;
}
}
bool theAllowAuthCheatMask = GetGameRules()->GetCheatsEnabled();
#ifdef AVH_LAN_PLAYTEST_BUILD
theAllowAuthCheatMask = true;
#endif
if(theAllowAuthCheatMask)
{
theMask |= this->mAuthCheatMask;
}
return theMask;
}
bool AvHPlayer::GetAllowAuth() const
{
return (this->mAllowAuth && (avh_uplink.value > 0));
}
void AvHPlayer::SetAllowAuth(bool inAllowAuth)
{
this->mAllowAuth = inAllowAuth;
}
void AvHPlayer::SetAuthCheatMask(int inAuthCheatMask)
{
this->mAuthCheatMask = inAuthCheatMask;
}
#endif
bool AvHPlayer::GetCurrentWeaponCannotHolster() const
{
bool theCannotHolster = false;
if(m_pActiveItem)
{
theCannotHolster = !this->m_pActiveItem->CanHolster();
}
return theCannotHolster;
}
bool AvHPlayer::GetInReadyRoom() const
{
return (this->GetPlayMode() == PLAYMODE_READYROOM);
}
bool AvHPlayer::GetIsAlien(bool inIncludeSpectating) const
{
bool theIsAlien = false;
AvHTeam* theTeam = this->GetTeamPointer(inIncludeSpectating);
if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN))
{
theIsAlien = true;
}
return theIsAlien;
}
bool AvHPlayer::GetIsMarine(bool inIncludeSpectating) const
{
bool theIsMarine = false;
AvHTeam* theTeam = this->GetTeamPointer(inIncludeSpectating);
if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE))
{
theIsMarine = true;
}
return theIsMarine;
}
bool AvHPlayer::GetIsInTopDownMode(bool inIncludeSpectating) const
{
return this->mInTopDownMode;
}
AvHPlayer* AvHPlayer::GetCommander(void)
{
AvHPlayer* theCommander = NULL;
AvHTeam* theMarinesTeam = this->GetTeamPointer();
if(theMarinesTeam)
{
int theCommanderIndex = theMarinesTeam->GetCommander();
AvHSUGetEntityFromIndex(theCommanderIndex, theCommander);
}
return theCommander;
}
bool AvHPlayer::GetHasBeenSpectator(void) const
{
return this->mHasBeenSpectator;
}
AvHPlayMode AvHPlayer::GetPlayMode(bool inIncludeSpectating) const
{
int theVarToUse = this->pev->playerclass;
if(inIncludeSpectating && (this->pev->iuser1 == OBS_IN_EYE))
{
AvHPlayer* theEntity = NULL;
if(AvHSUGetEntityFromIndex(this->pev->iuser2, theEntity))
{
theVarToUse = theEntity->pev->playerclass;
}
}
return (AvHPlayMode)theVarToUse;
}
bool AvHPlayer::GetIsValidReinforcementFor(AvHTeamNumber inTeam) const
{
bool theSuccess = false;
if(this->GetPlayMode() == PLAYMODE_AWAITINGREINFORCEMENT)
{
if(this->pev->team == inTeam)
{
theSuccess = true;
}
}
return theSuccess;
}
int AvHPlayer::GetPointValue(void) const
{
int thePointValue = BALANCE_VAR(kScoringKillPlayer);
if(this->GetIsAlien())
{
switch(this->pev->iuser3)
{
case AVH_USER3_ALIEN_PLAYER2:
thePointValue = BALANCE_VAR(kScoringKillPlayerGorge);
break;
case AVH_USER3_ALIEN_PLAYER3:
thePointValue = BALANCE_VAR(kScoringKillPlayerLerk);
break;
case AVH_USER3_ALIEN_PLAYER4:
thePointValue = BALANCE_VAR(kScoringKillPlayerFade);
break;
case AVH_USER3_ALIEN_PLAYER5:
thePointValue = BALANCE_VAR(kScoringKillPlayerOnos);
break;
}
}
else if(this->GetIsMarine())
{
if(this->GetHasJetpack())
{
thePointValue = BALANCE_VAR(kScoringKillPlayerJetpack);
}
else if(this->GetHasHeavyArmor())
{
thePointValue = BALANCE_VAR(kScoringKillPlayerHA);
}
}
return thePointValue;
}
vec3_t AvHPlayer::GetVisualOrigin() const
{
vec3_t theOrigin = this->pev->origin;
//theOrigin[2] += this->pev->view_ofs[2];
if(this->mInTopDownMode)
{
theOrigin[2] = GetGameRules()->GetMapExtents().GetMaxViewHeight();
}
return theOrigin;
}
int AvHPlayer::GetRespawnCost() const
{
int theRespawnCost = 0;
const AvHGameplay& theGameplay = GetGameRules()->GetGameplay();
if(this->GetClassType() == AVH_CLASS_TYPE_MARINE)
{
// This function shouldn't be used now that there are reinforcements...rework a bit?
theRespawnCost = 0;
}
else if(this->GetClassType() == AVH_CLASS_TYPE_ALIEN)
{
theRespawnCost = theGameplay.GetAlienRespawnCost();
}
return theRespawnCost;
}
bool AvHPlayer::GetHasItem(const char *szName)
{
bool theHasItem = false;
for(int i = 0; (i < MAX_ITEM_TYPES) && !theHasItem; i++)
{
CBasePlayerItem* theCurrentItem = this->m_rgpPlayerItems[i];
while(theCurrentItem && !theHasItem)
{
if(FStrEq(STRING(theCurrentItem->pev->classname), szName))
{
theHasItem = true;
}
theCurrentItem = theCurrentItem->m_pNext;
}
}
return theHasItem;
}
void AvHPlayer::GiveNamedItem(const char *szName, bool inSendMessage)
{
// Check to make sure we don't have this weapon before giving it. Fixes potential problems where
// weapon settings would be reset if a duplicate weapon was given
if(!this->GetHasItem(szName))
{
// Send visible pickup message after game has started
bool theSendMessage = GetGameRules()->GetGameStarted();
CBasePlayer::GiveNamedItem(szName, theSendMessage);
}
}
int AvHPlayer::GetNumberOfItems()
{
int theNumItems = 0;
for(int i = 0; i < MAX_ITEM_TYPES; i++)
{
CBasePlayerItem* theCurrentItem = this->m_rgpPlayerItems[i];
while(theCurrentItem)
{
theNumItems++;
theCurrentItem = theCurrentItem->m_pNext;
}
}
return theNumItems;
}
void AvHPlayer::GiveResources(float inResources)
{
this->SetResources(this->GetResources() + inResources);
}
void AvHPlayer::StartLeap()
{
// Make sure player has leap
if(this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER1)
{
this->mTimeLeapEnd = gpGlobals->time + kLeapDuration;
}
}
void AvHPlayer::PlayerTouch(CBaseEntity* inOther)
{
if(this && this->pev && inOther && this->m_pActiveItem && this->m_pActiveItem->pev)
{
// Uncloak when we touch enemy players to prevent blocking
if(inOther->IsPlayer() && inOther->IsAlive() && (inOther->pev->team != this->pev->team))
{
this->Uncloak();
}
// Don't do "touch" damage too quickly
float theTouchDamageInterval = BALANCE_VAR(kTouchDamageInterval);
if((this->mTimeOfLastTouchDamage == -1) || (gpGlobals->time > (this->mTimeOfLastTouchDamage + theTouchDamageInterval)))
{
entvars_t* theAttacker = this->pev;
entvars_t* theInflictor = this->m_pActiveItem->pev;
float theScalar = 1.0f;
if((this->mTimeLeapEnd != -1) && (gpGlobals->time < this->mTimeLeapEnd))
{
// Do damage to entity
if(GetGameRules()->CanEntityDoDamageTo(this, inOther, &theScalar) && (inOther->pev->team != this->pev->team))
{
float theDamage = BALANCE_VAR(kLeapDamage)*theScalar*theTouchDamageInterval;
inOther->TakeDamage(theInflictor, theAttacker, theDamage, NS_DMG_NORMAL);
this->mTimeOfLastTouchDamage = gpGlobals->time;
}
}
// Are we charging?
// if(GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_MOVEMENT))
// {
// // Don't do friendly fire
// if(GetGameRules()->CanEntityDoDamageTo(this, inOther, &theScalar))
// {
// float theDamage = BALANCE_VAR(kChargeDamage)*theScalar*theTouchDamageInterval;
//
// inOther->TakeDamage(theInflictor, theAttacker, theDamage, NS_DMG_NORMAL);
//
// if(inOther->IsPlayer() && !inOther->IsAlive())
// {
// EMIT_SOUND(ENT(this->pev), CHAN_WEAPON, kChargeKillSound, 1.0, ATTN_NORM);
// }
//
// this->mTimeOfLastTouchDamage = gpGlobals->time;
// }
// }
}
}
}
AvHTeamNumber AvHPlayer::GetTeam(bool inIncludeSpectating) const
{
AvHTeamNumber theTeamNumber = (AvHTeamNumber)this->pev->team;
if(inIncludeSpectating)
{
CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
theTeamNumber = (AvHTeamNumber)theSpectatingEntity->pev->team;
}
}
return theTeamNumber;
}
AvHTeam* AvHPlayer::GetTeamPointer(bool inIncludeSpectating) const
{
AvHTeamNumber theTeamNumber = this->GetTeam(inIncludeSpectating);
AvHTeam* theTeamPointer = GetGameRules()->GetTeam(theTeamNumber);
return theTeamPointer;
}
float AvHPlayer::GetKilledX() const
{
return this->mKilledX;
}
float AvHPlayer::GetKilledY() const
{
return this->mKilledY;
}
AvHClassType AvHPlayer::GetClassType(void) const
{
AvHClassType theClassType = AVH_CLASS_TYPE_UNDEFINED;
AvHTeamNumber theTeamNumber = (AvHTeamNumber)(this->pev->team);
const AvHTeam* theTeam = GetGameRules()->GetTeam(theTeamNumber);
if(theTeam)
{
theClassType = theTeam->GetTeamType();
}
ASSERT(theClassType >= AVH_CLASS_TYPE_UNDEFINED);
ASSERT(theClassType <= AVH_CLASS_TYPE_AUTOASSIGN);
return theClassType;
}
bool AvHPlayer::GetPurchaseAllowed(AvHMessageID inUpgrade, int& outCost, string* outErrorMessage) const
{
bool thePurchaseAllowed = false;
bool theGameStarted = GetGameRules()->GetGameStarted();
bool theIsMarine = this->GetTeamPointer()->GetTeamType() == AVH_CLASS_TYPE_MARINE;
bool theIsAlien = this->GetIsAlien();
int theNumHives = this->GetTeamPointer()->GetNumActiveHives();
bool theIsBuilder = (this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER2);
bool theIsResearchTech = AvHSHUGetIsResearchTech(inUpgrade);
bool theIsCombatMode = GetGameRules()->GetIsCombatMode();
string theErrorMessage;
// Check tech tree first
const AvHTechTree* theTechNodes = &this->GetTeamPointer()->GetTechNodes();
// If combat, use Combat nodes
if(GetGameRules()->GetIsCombatMode())
{
theTechNodes = &this->mCombatNodes;
}
if(theTechNodes->GetIsMessageAvailable(inUpgrade))
{
if(theIsResearchTech && (this->pev->iuser3 == AVH_USER3_COMMANDER_PLAYER) && theIsMarine)
{
thePurchaseAllowed = true;
}
else
{
if(theIsCombatMode && (this->GetExperienceLevel() <= (this->mExperienceLevelsSpent + GetGameRules()->GetCostForMessageID(inUpgrade))))
{
return false;
}
switch(inUpgrade)
{
case ALIEN_EVOLUTION_ONE:
case ALIEN_EVOLUTION_TWO:
case ALIEN_EVOLUTION_THREE:
//case ALIEN_EVOLUTION_FOUR:
//case ALIEN_EVOLUTION_FIVE:
//case ALIEN_EVOLUTION_SIX:
case ALIEN_EVOLUTION_SEVEN:
case ALIEN_EVOLUTION_EIGHT:
case ALIEN_EVOLUTION_NINE:
case ALIEN_EVOLUTION_TEN:
case ALIEN_EVOLUTION_ELEVEN:
case ALIEN_EVOLUTION_TWELVE:
// If we are an alien and we don't have the upgrade
if(theIsAlien)
{
AvHUpgradeMask theUpgradeMask = MASK_NONE;
if(AvHGetAlienUpgradeMask(inUpgrade, theUpgradeMask))
{
if(!GetHasUpgrade(this->pev->iuser4, theUpgradeMask))
{
AvHAlienUpgradeCategory theCategory = ALIEN_UPGRADE_CATEGORY_INVALID;
if(AvHGetAlienUpgradeCategory(inUpgrade, theCategory))
{
// Now make sure we have an unspent upgrade available
AvHTeam* theTeamPointer = this->GetTeamPointer();
if(theTeamPointer && (AvHGetHasFreeUpgradeCategory(theCategory, theTeamPointer->GetAlienUpgrades(), this->pev->iuser4) || GetGameRules()->GetIsCombatMode()))
{
thePurchaseAllowed = true;
}
else
{
theErrorMessage = kUpgradeNotAvailable;
}
}
}
}
}
break;
case ALIEN_LIFEFORM_ONE:
if(theIsAlien && theGameStarted && (this->GetUser3() != AVH_USER3_ALIEN_PLAYER1))
{
thePurchaseAllowed = true;
}
break;
case ALIEN_LIFEFORM_TWO:
if(theIsAlien && theGameStarted && (this->GetUser3() != AVH_USER3_ALIEN_PLAYER2))
{
thePurchaseAllowed = true;
}
break;
case ALIEN_LIFEFORM_THREE:
if(theIsAlien && theGameStarted && (this->GetUser3() != AVH_USER3_ALIEN_PLAYER3))
{
// if(theNumHives >= 1)
// {
thePurchaseAllowed = true;
// }
// else
// {
// outErrorMessage = kNeedOneHiveToGestate;
// }
}
break;
case ALIEN_LIFEFORM_FOUR:
if(theIsAlien && theGameStarted && (this->GetUser3() != AVH_USER3_ALIEN_PLAYER4))
{
// if(theNumHives >= 2)
// {
thePurchaseAllowed = true;
// }
// else
// {
// outErrorMessage = kNeedTwoHivesToGestate;
// }
}
break;
case ALIEN_LIFEFORM_FIVE:
if(theIsAlien && theGameStarted && (this->GetUser3() != AVH_USER3_ALIEN_PLAYER5))
{
// if(theNumHives >= 3)
// {
thePurchaseAllowed = true;
// }
// else
// {
// outErrorMessage = kNeedThreeHivesToGestate;
// }
}
break;
case ALIEN_BUILD_RESOURCES:
case ALIEN_BUILD_HIVE:
if(theIsBuilder)
{
thePurchaseAllowed = true;
}
else
{
theErrorMessage = kMustBeBuilder;
}
break;
case ALIEN_BUILD_OFFENSE_CHAMBER:
if(theIsBuilder)
{
if(theNumHives >= 1)
{
thePurchaseAllowed = true;
}
else
{
theErrorMessage = kNeedOneHiveForStructure;
}
}
else
{
theErrorMessage = kMustBeBuilder;
}
break;
// Make sure we have a hive that can provide this tech
case ALIEN_BUILD_DEFENSE_CHAMBER:
case ALIEN_BUILD_MOVEMENT_CHAMBER:
case ALIEN_BUILD_SENSORY_CHAMBER:
if(theIsBuilder)
{
FOR_ALL_ENTITIES(kesTeamHive, AvHHive*)
if(theEntity && theEntity->GetIsActive() && (theEntity->pev->team == this->pev->team))
{
AvHMessageID theTechnology = theEntity->GetTechnology();
if((theTechnology == inUpgrade) || (theTechnology == MESSAGE_NULL))
{
thePurchaseAllowed = true;
break;
}
}
END_FOR_ALL_ENTITIES(kesTeamHive);
if(!thePurchaseAllowed)
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
// int theNumHives = theTeam->GetNumActiveHives();
// switch(theNumHives)
// {
// case 0:
// outErrorMessage = kNeedOneHiveForStructure;
// break;
// case 1:
// outErrorMessage = kNeedTwoHivesForStructure;
// break;
// case 2:
// outErrorMessage = kNeedThreeHivesForStructure;
// break;
// }
theErrorMessage = kNeedsAnotherHiveForStructure;
}
}
}
else
{
theErrorMessage = kMustBeBuilder;
}
break;
default:
thePurchaseAllowed = true;
break;
}
}
}
if(thePurchaseAllowed)
{
// This is either resources, energy, or levels
int theCost = GetGameRules()->GetCostForMessageID(inUpgrade);
if(GetGameRules()->GetIsCombatMode())
{
int theLevelsFree = max(this->GetExperienceLevel() - this->GetExperienceLevelsSpent() - 1, 0);
if(theCost > theLevelsFree)
{
thePurchaseAllowed = false;
}
}
else
{
if(AvHSHUGetDoesTechCostEnergy(inUpgrade) || GetGameRules()->GetIsCheatEnabled(kcLowCost))
{
theCost = 0;
}
if(this->GetResources() < theCost)
{
thePurchaseAllowed = false;
}
}
outCost = theCost;
}
if(outErrorMessage)
{
*outErrorMessage = theErrorMessage;
}
return thePurchaseAllowed;
}
int AvHPlayer::GiveAmmo( int iAmount, char *szName, int iMax )
{
int theReturnValue = -1;
int theRealAmount = iAmount;
string theRealName = szName;
char theRealNameArray[256];
int theRealMax = iMax;
// TODO: Change this to a more generic call on AvHBasePlayerWeapon for next client update
AvHBasePlayerWeapon* theBaseWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
if(theBaseWeapon != NULL && theBaseWeapon->GetCanBeResupplied())
{
if(FStrEq(szName, kwsGenericAmmo))
{
AvHBasePlayerWeapon* theWeaponToGiveTo = theBaseWeapon;
if(theBaseWeapon->UsesAmmo())
{
// Translate iAmount, szName and iMax into the ammo for our current weapon
ItemInfo theItemInfo;
if(this->m_pActiveItem->GetItemInfo(&theItemInfo) != 0)
{
//theRealAmount = theBaseWeapon->m_iDefaultAmmo;
theRealAmount = theItemInfo.iMaxClip;
theRealName = theItemInfo.pszAmmo1;
theRealMax = theItemInfo.iMaxAmmo1;
}
}
}
}
strcpy(theRealNameArray, theRealName.c_str());
theReturnValue = CBasePlayer::GiveAmmo(theRealAmount, theRealNameArray, theRealMax);
return theReturnValue;
}
bool AvHPlayer::GetShouldResupplyAmmo()
{
bool theResupply = false;
if(this->GetIsMarine() && this->GetIsRelevant())
{
AvHBasePlayerWeapon* theBaseWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
if(theBaseWeapon != NULL && (theBaseWeapon->m_iId != AVH_WEAPON_MINE))
{
AvHBasePlayerWeapon* theWeaponToGiveTo = theBaseWeapon;
if(theBaseWeapon->UsesAmmo())
{
// Translate iAmount, szName and iMax into the ammo for our current weapon
ItemInfo theItemInfo;
if(this->m_pActiveItem->GetItemInfo(&theItemInfo) != 0)
{
int theMaxAmmo = theItemInfo.iMaxAmmo1;
if(theMaxAmmo > 0)
{
int i = GetAmmoIndex(theItemInfo.pszAmmo1);
if ((i > 0) && (i < MAX_AMMO_SLOTS))
{
int theCurrentAmmo = this->m_rgAmmo[i];
if (theCurrentAmmo < theWeaponToGiveTo->GetClipSize()) {
theResupply = true;
}
}
}
}
}
}
}
return theResupply;
}
float AvHPlayer::GetCurrentWeaponAmmoPercentage()
{
float thePercentage = 1.0f;
if(this->GetIsMarine() && this->GetIsRelevant())
{
AvHBasePlayerWeapon* theBaseWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
if(theBaseWeapon != NULL && (theBaseWeapon->m_iId != AVH_WEAPON_MINE))
{
AvHBasePlayerWeapon* theWeaponToGiveTo = theBaseWeapon;
if(theBaseWeapon->UsesAmmo())
{
// Translate iAmount, szName and iMax into the ammo for our current weapon
ItemInfo theItemInfo;
if(this->m_pActiveItem->GetItemInfo(&theItemInfo) != 0)
{
int theMaxAmmo = theItemInfo.iMaxAmmo1;
if(theMaxAmmo > 0)
{
int i = GetAmmoIndex(theItemInfo.pszAmmo1);
if ((i > 0) && (i < MAX_AMMO_SLOTS))
{
int theCurrentAmmo = this->m_rgAmmo[i];// + theWeaponToGiveTo->GetShotsInClip();
thePercentage = (float)theCurrentAmmo/theMaxAmmo;
}
}
}
}
}
}
return thePercentage;
}
bool AvHPlayer::GetEnemySighted(void) const
{
return this->mEnemySighted;
}
bool AvHPlayer::GetIsFiring(void) const
{
bool theIsFiring = false;
if(this->pev->button & IN_ATTACK)
{
AvHBasePlayerWeapon* theWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
if(theWeapon && theWeapon->m_iClip > 0)
{
theIsFiring = true;
}
}
return theIsFiring;
}
bool AvHPlayer::GetIsInOverwatch(void) const
{
return this->mInOverwatch;
}
bool AvHPlayer::GetIsSpectatingTeam(AvHTeamNumber inTeamNumber) const
{
bool theIsSpectatingTeam = false;
CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
if((AvHTeamNumber)theSpectatingEntity->pev->team == inTeamNumber)
{
theIsSpectatingTeam = true;
}
}
return theIsSpectatingTeam;
}
bool AvHPlayer::GetIsSpectatingPlayer(int inPlayerNumber) const
{
bool theIsSpectatingPlayer = false;
CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
if(theSpectatingEntity->entindex() == inPlayerNumber)
{
theIsSpectatingPlayer = true;
}
}
return theIsSpectatingPlayer;
}
bool AvHPlayer::GetIsSpeaking(void) const
{
return this->mIsSpeaking;
}
AvHMessageID AvHPlayer::GetLastSaying() const
{
return this->mLastSaying;
}
bool AvHPlayer::GetOrdersRequested(void) const
{
return this->mOrdersRequested;
}
bool AvHPlayer::GetOrderAcknowledged(void) const
{
return this->mOrderAcknowledged;
}
string AvHPlayer::GetPlayerName() const
{
string thePlayerName = (this->pev->netname && STRING(this->pev->netname)[0] != 0 ) ? STRING(this->pev->netname) : "unconnected" ;
return thePlayerName;
}
int AvHPlayer::GetRelevantWeight(void) const
{
float theRelevantWeight = 0;
// Add current weapon if any
//CBasePlayerWeapon* theActiveWeapon = dynamic_cast<CBasePlayerWeapon*>(this->m_pActiveItem);
//if(theActiveWeapon)
//{
// theRelevantWeight += this->GetRelevantWeightForWeapon(theActiveWeapon);
//}
// Run through items and tally up the ones that we count
for(int i = 0; i< MAX_ITEM_TYPES; i++)
{
AvHBasePlayerWeapon* theCurrentWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_rgpPlayerItems[i]);
while(theCurrentWeapon)
{
int theWeight = this->GetRelevantWeightForWeapon(theCurrentWeapon);
// Active items count full, count less when stowed
float theMultiplier = (theCurrentWeapon == this->m_pActiveItem) ? 1.0f : .7f;
theRelevantWeight += theWeight*theMultiplier;
theCurrentWeapon = dynamic_cast<AvHBasePlayerWeapon*>(theCurrentWeapon->m_pNext);
}
}
return (int)(theRelevantWeight);
}
int AvHPlayer::GetRelevantWeightForWeapon(AvHBasePlayerWeapon* inWeapon) const
{
ASSERT(inWeapon != NULL);
AvHWeaponID theWeaponID = (AvHWeaponID)inWeapon->m_iId;
int theNumRounds = 0;
if(inWeapon->UsesAmmo())
{
int theAmmoIndex = inWeapon->PrimaryAmmoIndex();
ASSERT(theAmmoIndex >= 0);
ASSERT(theAmmoIndex < MAX_AMMO_SLOTS);
theNumRounds = max(inWeapon->m_iClip + this->m_rgAmmo[theAmmoIndex], 0);
}
return GetGameRules()->GetWeightForItemAndAmmo(theWeaponID, theNumRounds);
}
AvHUser3 AvHPlayer::GetPreviousUser3(bool inIncludeSpectating) const
{
AvHUser3 theUser3 = this->mPreviousUser3;
if(inIncludeSpectating)
{
const CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
const AvHPlayer* theSpectatingPlayer = dynamic_cast<const AvHPlayer*>(theSpectatingEntity);
if(theSpectatingPlayer)
{
theUser3 = theSpectatingPlayer->GetPreviousUser3();
}
}
}
return theUser3;
}
AvHUser3 AvHPlayer::GetUser3(bool inIncludeSpectating) const
{
AvHUser3 theUser3 = (AvHUser3)this->pev->iuser3;
if(inIncludeSpectating)
{
const CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
const AvHPlayer* theSpectatingPlayer = dynamic_cast<const AvHPlayer*>(theSpectatingEntity);
if(theSpectatingPlayer)
{
theUser3 = theSpectatingPlayer->GetUser3();
}
}
}
return theUser3;
}
AvHMessageID AvHPlayer::GetEvolution(bool inIncludeSpectating) const
{
AvHMessageID theEvolution = mEvolution;
if(inIncludeSpectating)
{
const CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
const AvHPlayer* theSpectatingPlayer = dynamic_cast<const AvHPlayer*>(theSpectatingEntity);
if(theSpectatingPlayer)
{
theEvolution = theSpectatingPlayer->GetEvolution();
}
}
}
return theEvolution;
}
bool AvHPlayer::GetSpecialPASOrigin(Vector& outOrigin)
{
bool theUseSpecial = (this->GetUser3() == AVH_USER3_COMMANDER_PLAYER);
if(theUseSpecial)
{
VectorCopy(this->mSpecialPASOrigin, outOrigin);
}
return theUseSpecial;
}
// Don't check validity. ASSERT if an error is encountered.
void AvHPlayer::GiveUpgrade(AvHMessageID inUpgrade)
{
}
void AvHPlayer::GiveTeamUpgrade(AvHMessageID inUpgrade, bool inGive)
{
AvHUser3 theUser3 = (AvHUser3)this->pev->iuser3;
int thePlayerLevel = this->GetExperienceLevel();
float theHealthPercentage = this->pev->health/AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, theUser3, thePlayerLevel);
float theArmorPercentage = this->pev->armorvalue/AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, theUser3);
// If we're in combat mode, handle upgrades differently
ProcessGenericUpgrade(this->pev->iuser4, inUpgrade, inGive);
//AvHTeam* theTeam = this->GetTeamPointer();
//if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE))
//{
AvHUpgradeMask theMask = ProcessGenericUpgrade(this->pev->iuser4, inUpgrade, inGive);
if(inUpgrade == BUILD_HEAVY)
{
this->EffectivePlayerClassChanged();
}
//}
this->pev->health = theHealthPercentage*AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, theUser3, thePlayerLevel);
this->pev->armorvalue = theArmorPercentage*AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, theUser3);
}
float AvHPlayer::GetResources(bool inIncludeSpectating) const
{
float theResources = this->mResources;
if(inIncludeSpectating && (this->pev->iuser1 == OBS_IN_EYE))
{
AvHPlayer* theEntity = NULL;
if(AvHSUGetEntityFromIndex(this->pev->iuser2, theEntity))
{
theResources = theEntity->GetResources(false);
}
}
const AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
if(theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE)
{
theResources = theTeam->GetTeamResources();
}
}
return theResources;
}
void AvHPlayer::GetSpeeds(int& outBaseSpeed, int& outUnemcumberedSpeed) const
{
// Use marine speed when in ready room (ie, AVH_CLASS_TYPE_UNDEFINED)
//int theBaseSpeed = 160;
// Counter-strike speeds are around 260 for knife running, and 215 with the para (according to Adrian's memory)
int theBaseSpeed = BALANCE_VAR(kBasePlayerSpeed);
int theUnencumberedSpeed = BALANCE_VAR(kUnencumberedPlayerSpeed);
if(this->IsObserver())
{
theBaseSpeed = theUnencumberedSpeed = 1000;
}
else if(this->GetClassType() == AVH_CLASS_TYPE_MARINE)
{
if(this->pev->iuser3 == AVH_USER3_COMMANDER_PLAYER)
{
if(this->mInTopDownMode)
{
// For scrolling
theBaseSpeed = theUnencumberedSpeed = BALANCE_VAR(kCommanderPlayerSpeed);
}
}
else if(this->GetHasHeavyArmor())
{
float theHeavyMultiplier = BALANCE_VAR(kHeavySpeedMultiplier);
theBaseSpeed *= theHeavyMultiplier;
theUnencumberedSpeed *= theHeavyMultiplier;
}
else if(this->GetHasJetpack())
{
float theJetpackMultiplier = BALANCE_VAR(kJetpackSpeedMultiplier);
theBaseSpeed *= theJetpackMultiplier;
theUnencumberedSpeed *= theJetpackMultiplier;
}
if(GetHasUpgrade(this->pev->iuser4, MASK_BUFFED))
{
const float kStimpackSpeedMultiplier = 1 + BALANCE_VAR(kCatalystSpeedIncrease);
theBaseSpeed *= kStimpackSpeedMultiplier;
theUnencumberedSpeed *= kStimpackSpeedMultiplier;
}
}
else if(this->GetClassType() == AVH_CLASS_TYPE_ALIEN)
{
int theSpeedUpgradeLevel = 0;
if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_4))
{
theSpeedUpgradeLevel = 1;
if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_12))
{
theSpeedUpgradeLevel = 2;
}
else if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_13))
{
theSpeedUpgradeLevel = 3;
}
}
// When gestating
float theAlienBaseSpeed = 0;
int theSpeedUpgradeAmount = BALANCE_VAR(kAlienCelerityBonus);
const float kChargingFactor = 2.0f;
switch(this->pev->iuser3)
{
// Take into account speed upgrade, if this player has it
case AVH_USER3_ALIEN_PLAYER1:
theAlienBaseSpeed = BALANCE_VAR(kSkulkBaseSpeed);
break;
case AVH_USER3_ALIEN_PLAYER2:
theAlienBaseSpeed = BALANCE_VAR(kGorgeBaseSpeed);
break;
case AVH_USER3_ALIEN_PLAYER3:
theAlienBaseSpeed = BALANCE_VAR(kLerkBaseSpeed);
// Compensate for airpseed multiplier, so lerk gets proper speed increase in main mode of locomotion
//theSpeedUpgradeAmount /= BALANCE_VAR(kAirspeedMultiplier);
break;
case AVH_USER3_ALIEN_PLAYER4:
theAlienBaseSpeed = BALANCE_VAR(kFadeBaseSpeed);
break;
case AVH_USER3_ALIEN_PLAYER5:
//theAlienBaseSpeed = this->mMaxGallopSpeed;
theAlienBaseSpeed = BALANCE_VAR(kOnosBaseSpeed);
// if(GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_MOVEMENT))
// {
// theAlienBaseSpeed *= kChargingFactor;
// theSpeedUpgradeAmount *= kChargingFactor;
// }
break;
}
theUnencumberedSpeed = theBaseSpeed = theAlienBaseSpeed + (theSpeedUpgradeLevel*theSpeedUpgradeAmount);
// Update kMaxPlayerSpeed to make sure footsteps are played correctly
ASSERT(theUnencumberedSpeed <= kMaxGroundPlayerSpeed);
}
outBaseSpeed = theBaseSpeed;
outUnemcumberedSpeed = theUnencumberedSpeed;
// Take into account ensnare
if(this->GetIsEnsnared())
{
float thePercentageComplete = (gpGlobals->time - this->mLastTimeEnsnared)/(this->mTimeToBeUnensnared - this->mLastTimeEnsnared);
thePercentageComplete = max(min(thePercentageComplete, 1.0f), 0.0f);
//ASSERT(thePercentageComplete >= 0.0f);
//ASSERT(thePercentageComplete <= 1.0f);
float theSpeedFactor = min(BALANCE_VAR(kEnsnareMinimumSpeedFactor) + (1.0f - BALANCE_VAR(kEnsnareMinimumSpeedFactor))*thePercentageComplete, BALANCE_VAR(kEnsnareMaximumSpeedFactor));
ASSERT(theSpeedFactor >= BALANCE_VAR(kEnsnareMinimumSpeedFactor));
ASSERT(theSpeedFactor <= 1.0f);
outBaseSpeed *= theSpeedFactor;
outUnemcumberedSpeed *= theSpeedFactor;
}
// Reduce speed a bit when player has carapace
if(this->GetIsAlien())
{
int theCarapaceLevel = AvHPlayerUpgrade::GetArmorUpgrade((AvHUser3)this->pev->iuser3, this->pev->iuser4);
float theSpeedFactor = (1.0f - BALANCE_VAR(kCarapaceSlowFactor)*theCarapaceLevel);
outBaseSpeed *= theSpeedFactor;
outUnemcumberedSpeed *= theSpeedFactor;
}
if(this->GetIsMetabolizing())
{
float theMetabolizeSpeedFactor = BALANCE_VAR(kMetabolizeSpeedFactor);
outBaseSpeed *= theMetabolizeSpeedFactor;
outUnemcumberedSpeed *= theMetabolizeSpeedFactor;
}
if(this->GetIsDigesting())
{
float theDigestingSpeedFactor = BALANCE_VAR(kDigestingSpeedFactor);
outBaseSpeed *= theDigestingSpeedFactor;
outUnemcumberedSpeed *= theDigestingSpeedFactor;
}
if(this->GetIsStunned())
{
// MUHAHAHA!
outBaseSpeed = outUnemcumberedSpeed = 0.0f;
}
// If we're a fade using blink, increase our speed massively
// if(this->GetIsBlinking())
// {
// outBaseSpeed *= 10.0f;
// outUnemcumberedSpeed = outBaseSpeed;
// }
// if(GetGameRules()->GetIsCombatMode())
// {
// int theSpeedIncrease = (this->GetExperienceLevel() - 1)*BALANCE_VAR(kCombatLevelSpeedIncrease);
//
// outBaseSpeed += theSpeedIncrease;
// outUnemcumberedSpeed += theSpeedIncrease;
// }
if(GetGameRules()->GetIsCheatEnabled(kcHighSpeed))
{
outBaseSpeed = outUnemcumberedSpeed = kMaxGroundPlayerSpeed*.7f;
}
}
int AvHPlayer::GetMaxWalkSpeed() const
{
return this->mMaxWalkSpeed;
}
void AvHPlayer::PickSkin()
{
int theSkin = 0;
if(this->GetIsMarine())
{
theSkin = RANDOM_LONG(0, 1);
}
this->SetSkin(theSkin);
}
void AvHPlayer::SetSkin(int inSkin)
{
if(this->pev)
{
this->pev->skin = inSkin;
}
}
void AvHPlayer::GiveOrderToSelection(AvHOrder& inOrder)
{
// Set the player list as the selected players
EntityListType theReceivers = this->mSelected;
// Set the order for the team
AvHTeam* theTeam = this->GetTeamPointer();
ASSERT(theTeam);
int theOrderID=inOrder.GetOrderID();
if(GetGameRules()->GetIsCheatEnabled(kcOrderSelf))
{
AvHOrder theOrder=inOrder;
int theSelfIndex = this->entindex();
theOrder.SetReceiver(theSelfIndex);
theOrder.SetOrderID();
theTeam->SetOrder(theOrder);
}
if ( this->mSelected.size() > 0 ) {
for(EntityListType::iterator theIter = theReceivers.begin(); theIter != theReceivers.end(); theIter++)
{
if ( inOrder.GetTargetIndex() != *theIter )
{
AvHOrder theOrder=inOrder;
theOrder.SetReceiver(*theIter);
theOrder.SetOrderID();
theTeam->SetOrder(theOrder);
}
}
}
// set this to true to indicate they don't need help
this->mHasGivenOrder = true;
}
bool AvHPlayer::GiveOrderToSelection(AvHOrderType inOrder, Vector inNormRay)
{
bool theSuccess = false;
AvHOrder theNewOrder;
Vector theOrigin = this->GetVisualOrigin();
// #ifdef DEBUG
// vec3_t theStartPoint;
// VectorMA(theOrigin, kSelectionStartRange, inNormRay, theStartPoint);
//
// vec3_t theEndPoint;
// VectorMA(theOrigin, kSelectionEndRange, inNormRay, theEndPoint);
//
// vec3_t theValidOrigin;
// AvHSHUServerGetFirstNonSolidPoint(theStartPoint, theEndPoint, theValidOrigin);
//
// theValidOrigin.z -= BALANCE_VAR(kBiteDamage);
//
// CBaseEntity* pEnt = CBaseEntity::Create(kwsDebugEntity, theValidOrigin, Vector(0, 0, 0));
// ASSERT(pEnt);
// pEnt->pev->movetype = MOVETYPE_FLY;
// pEnt->pev->solid = SOLID_NOT;
// #endif
if(AvHCreateSpecificOrder((AvHTeamNumber)(this->pev->team), theOrigin, inOrder, inNormRay, theNewOrder))
{
this->GiveOrderToSelection(theNewOrder);
theSuccess = true;
}
return theSuccess;
}
void AvHPlayer::HolsterCurrent()
{
if(this->m_pActiveItem)
{
CBasePlayerWeapon* theCurrentWeapon = (CBasePlayerWeapon *)this->m_pActiveItem->GetWeaponPtr();
if(theCurrentWeapon && theCurrentWeapon->CanHolster())
{
theCurrentWeapon->Holster();
}
}
}
void AvHPlayer::Kick()
{
char theCommandBuffer[256];
sprintf(theCommandBuffer, "kick \"%s\"\n", STRING(this->pev->netname));
SERVER_COMMAND(theCommandBuffer);
}
void AvHPlayer::ImpulseCommands()
{
AvHMessageID theMessage = (AvHMessageID)pev->impulse;
this->PlayerUse();
bool theHandledMessage = false;
if(theMessage != MESSAGE_NULL)
{
// These things can only happen during play, not in ready room, reinforcement or observer modes
if(this->GetPlayMode() == PLAYMODE_PLAYING || (GetGameRules()->GetCheatsEnabled()))
{
if(this->ExecuteMessage(theMessage))
{
theHandledMessage = true;
}
}
if(!theHandledMessage)
{
CBasePlayer::ImpulseCommands();
}
// Very important I found out
this->pev->impulse = 0;
}
}
void AvHPlayer::ItemPostFrame(void)
{
// Check if player tried to do something while we were in the ready room. If so, display tutorial message.
if(this->GetPlayMode() == PLAYMODE_READYROOM)
{
// +movement: Removed case for +attack2
if((this->pev->button & IN_ATTACK) /* || (this->pev->button & IN_ATTACK2) */ || (this->pev->button & IN_RELOAD))
{
this->SendMessageOnce(kReadyRoomMessage, TOOLTIP);
}
}
if(!this->GetIsBeingDigested())
{
CBasePlayer::ItemPostFrame();
}
this->UpdateAmbientSounds();
}
void AvHPlayer::Init()
{
int i;
// Copy the server variables from the game rules.
AvHGamerules* theGameRules = GetGameRules();
mServerVariableList.clear();
mLastUpdateTime = -1;
for (i = 0; i < theGameRules->GetNumServerVariables(); ++i)
{
mServerVariableList.push_back(ServerVariable());
mServerVariableList.back().mName = theGameRules->GetServerVariable(i);
}
// Reset to default team
strcpy(this->m_szTeamName, kUndefinedTeam);
this->mResources = 0;
this->mScore = 0;
this->mSavedCombatFrags = 0;
this->mLastModelIndex = -1;
#ifdef USE_OLDAUTH
this->mCachedAuthenticationMask = -1;
#endif
this->mFirstUpdate = true;
this->mNewMap = true;
this->mHasSeenTeamA = false;
this->mHasSeenTeamB = false;
this->mPendingCommand = NULL;
this->mIsSpeaking = false;
this->mOrdersRequested = false;
this->mOrderAcknowledged = false;
this->mEnemySighted = false;
this->mTriggerUncloak = false;
this->mTimeOfLastSaying = 0;
this->mLastSaying = MESSAGE_NULL;
this->mTimeOfLastEnemySighting = 0;
this->mClientInTopDownMode = false;
this->mInTopDownMode = false;
this->mClientCommander = -1;
vec3_t theOrigin( 0, 0, 0 );
VectorCopy(theOrigin, this->mPositionBeforeTopDown);
VectorCopy(theOrigin, this->mAnglesBeforeTopDown);
VectorCopy(theOrigin, this->mViewAnglesBeforeTopDown);
VectorCopy(theOrigin, this->mViewOfsBeforeTopDown);
VectorCopy(theOrigin, this->mLastGallopViewDirection);
this->mAnimExtensionBeforeTopDown = "";
this->mTimeStartedTopDown = -1;
if(this->pev)
{
this->pev->team = TEAM_IND;
this->ClearUserVariables();
this->pev->rendermode = kRenderNormal;
this->pev->renderfx = kRenderFxNone;
this->pev->renderamt = 0;
this->pev->skin = 0;
this->pev->solid = SOLID_NOT;
this->pev->frags = 0;
this->m_iDeaths = 0;
this->pev->playerclass = PLAYMODE_UNDEFINED;
}
this->mClientInOverwatch = false;
this->mInOverwatch = false;
this->mOverwatchEnabled = true;
this->mOverwatchTarget = -1;
this->mTimeOfLastOverwatchPreventingAction = -1;
this->mTimeLastSeenOverwatchTarget = 0;
this->mOverwatchFiredThisThink = false;
// tankefugl: 0000953
this->mTimeLastJoinTeam = -1;
// tankefugl
// alien upgrades
this->mTimeOfLastRegeneration = -1;
this->mTimeOfLastPheromone = -1;
this->mMaxGallopSpeed = 0;
this->mClientResearchingTech = MESSAGE_NULL;
this->mAttackOneDown = false;
this->mAttackTwoDown = false;
this->mPlacingBuilding = false;
this->mPreviousUser3 = AVH_USER3_NONE;
this->mSavedJetpackEnergy = 0;
this->mHasBeenSpectator = false;
this->mHasLeftReadyRoom = false;
this->mQueuedThinkMessage = "";
// Clear out tech nodes
this->mClientTechNodes.Clear();
this->mClientTechSlotList.clear();
// Clear out map locations
this->mClientInfoLocations.clear();
// Clear out hive info
this->mClientHiveInfo.clear();
this->mClientGamma = kDefaultMapGamma;
// Clear sent message list
// Don't clear this message, don't keep sending tutorial messages
//this->mSentMessageList.clear();
//this->mClientBlipList.clear();
//this->mBlipList.clear();
this->mAlienSightActive = false;
this->mNumHives = 0;
this->mSpecialPASOrigin.x = this->mSpecialPASOrigin.y = this->mSpecialPASOrigin.z = 0.0f;
this->mClientSpecialPASOrigin.x = this->mClientSpecialPASOrigin.y = this->mClientSpecialPASOrigin.z = 0.0f;
this->mTimeOfLastPASUpdate = -1;
// puzl: 984
// record the last time the player attempted to go to the readyroom
this->mTimeOfLastF4 = -1.0f;
this->mTimeOfLastTeleport = -1;
this->mTimeOfLastHelpText = -1;
this->mTimeOfLastUse = -1;
this->mTimeLeapEnd = -1;
this->mTimeOfLastRedeem = -1;
memset(&this->mClientDebugCSPInfo, 0, sizeof(weapon_data_t));
memset(&this->mDebugCSPInfo, 0, sizeof(weapon_data_t));
this->mClientNextAttack = 0;
this->mMaxWalkSpeed = CBasePlayer::GetMaxWalkSpeed();
this->mTimeToBeUnensnared = -1;
this->mLastTimeEnsnared = -1;
this->mLastTimeStartedPlaying = -1;
this->mTimeToBeFreeToMove = -1;
this->mTimeToEndCatalyst = -1;
this->mLastTimeInCommandStation = -1;
this->mLastTimeRedemptionTriggered = -1;
this->mLastTimeCheckedRedemption = -1;
this->mJetpackEnergy = 1.0f;
this->mJetpacking = false;
this->mLastPowerArmorThink = -1;
this->mLastInventoryThink = -1;
this->mTimeLastPlaying = -1;
this->mTimeGestationStarted = -1;
this->mEvolution = MESSAGE_NULL;
this->mHealthPercentBefore = -1;
this->mArmorPercentBefore = -1;
this->mTimeStartedScream = -1;
//this->mNumParticleTemplatesSent = 0;
//this->mTimeOfLastParticleTemplateSending = -1;
this->mEnemyBlips.Clear();
this->mFriendlyBlips.Clear();
this->mSelected.clear();
this->mClientTrackingEntity = this->mTrackingEntity = 0;
this->mClientSelectAllGroup.clear();
for(i = 0; i < kNumHotkeyGroups; i++)
{
this->mClientGroups[i].clear();
this->mClientGroupAlerts[i] = ALERT_NONE;
}
this->mProgressBarEntityIndex = -1;
// Don't set this, must propagate
//this->mClientProgressBarEntityIndex = -1;
this->mProgressBarParam = -1;
this->mTimeProgressBarTriggered = -1;
this->mTimeOfLastFogTrigger = -1;
this->mFogExpireTime = -1;
this->mCurrentFogEntity = -1;
// Don't set this, we need to propagate lack of fog
//this->mClientCurrentFogEntity = -1;
//this->mUpgrades.clear();
this->mClientOrders.clear();
this->mMouseWorldPos = this->mAttackOnePressedWorldPos = this->mAttackTwoPressedWorldPos = theOrigin;
this->mClientEntityHierarchy.Clear();
this->mKilledX = this->mKilledY = -1;
this->mHasGivenOrder = false;
this->mTimeOfLastSignificantCommanderAction = -1;
this->mPreThinkTicks = 0;
this->mDesiredNetName = "";
#ifdef USE_OLDAUTH
this->mAuthCheatMask = 0;
this->mAllowAuth = true;
#endif
this->mTimeOfLastClassAndTeamUpdate = -1;
this->mEffectivePlayerClassChanged = false;
this->mNeedsTeamUpdate = false;
this->mSendTeamUpdate = false;
this->mSendSpawnScreenFade = false;
this->mClientMenuTechSlots = 0;
memset(&this->mClientRequests, 0, sizeof(int)*kNumRequestTypes);
this->mDigestee = 0;
this->mTimeOfLastDigestDamage = 0;
this->mTimeOfLastCombatThink = 0;
this->mDesiredRoomType = this->mClientDesiredRoomType = 0;
this->mTimeOfLastConstructUseAnimation = 0;
this->mTimeOfLastConstructUse = -1;
this->mTimeOfLastResupply = 0;
this->mTimeOfMetabolizeEnd = -1;
this->m_DefaultSpectatingMode = OBS_IN_EYE;
this->m_DefaultSpectatingTarget = 1;
this->mLastSelectEvent = MESSAGE_NULL;
VectorCopy(g_vecZero, this->mPositionBeforeLastGotoGroup);
this->mTimeOfLastSporeDamage = -1;
this->mTimeOfLastTouchDamage = -1;
this->mLastMessageSent = "";
this->mExperience = 0.0;
this->mExperienceLevelsSpent = 0;
this->mClientExperienceLevelsSpent = 0;
this->mClientPercentToNextLevel = 0.0f;
this->mCombatNodes.Clear();
this->mPurchasedCombatUpgrades.clear();
this->mGiveCombatUpgrades.clear();
this->mMarineHUDUpgrades=0;
}
void AvHPlayer::InitializeFromTeam(float inHealthPercentage, float inArmorPercentage)
{
// Set base health and armor
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
this->pev->health = this->pev->max_health = max(theMaxHealth*inHealthPercentage,1.0f);//voogru: prevent bug with players evolving down from higher lifeform from getting negative health but still "alive"
this->pev->armorvalue = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3)*inArmorPercentage;
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
ASSERT(theTeam);
// If he's already on the team, this does nothing (his rank keeps over death)
if(theTeam->AddPlayer(this->entindex()))
{
GetGameRules()->RecalculateHandicap();
float theStartingPoints = theTeam->GetInitialPlayerPoints(this->edict());
if(this->GetIsMarine())
{
theStartingPoints += this->GetResources();
}
// Set starting resources
this->SetResources(theStartingPoints, false);
// If we're in combat mode, we use our own tech nodes instead of our team's
if(GetGameRules()->GetIsCombatMode())
{
AvHTechTree theInitialTechNodes = theTeam->GetTechNodes();
// Removed restoration of previous levels and experience due to abuse
//// Restore saved/spent nodes if we've already been playing
//AvHServerPlayerData* theServerPlayerData = this->GetServerPlayerData();
//if(theServerPlayerData)
//{
// AvHTechNodes theSavedTechNodes = theServerPlayerData->GetCombatNodes();
// if(theSavedTechNodes.GetNumNodes() > 0)
// {
// theInitialTechNodes = theSavedTechNodes;
// }
//
// int a = this->mExperienceLevelsSpent;
//
// this->SetExperienceLevelsSpent(theServerPlayerData->GetExperienceLevelsSpent());
//
// MessageIDListType thePurchasedCombatUpgrades = theServerPlayerData->GetPurchasedCombatUpgrades();
// for(MessageIDListType::iterator theIter = thePurchasedCombatUpgrades.begin(); theIter != thePurchasedCombatUpgrades.end(); theIter++)
// {
// this->PurchaseCombatUpgrade(*theIter);
// }
//}
// Save it
this->SetCombatNodes(theInitialTechNodes);
// Set initial experience (restore old experience if player disconnected)
this->mExperience = theTeam->GetInitialExperience(this->edict());
}
// Random multiracial marines
this->PickSkin();
}
}
}
bool AvHPlayer::GetIsRelevant(bool inIncludeSpectating) const
{
bool theIsRelevant = false;
// Use ourself unless we're spectating
const AvHPlayer* thePlayer = this;
if(inIncludeSpectating)
{
const CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
const AvHPlayer* theSpectatingPlayer = dynamic_cast<const AvHPlayer*>(theSpectatingEntity);
if(theSpectatingPlayer)
{
thePlayer = theSpectatingPlayer;
}
}
}
if(thePlayer->IsAlive() && (thePlayer->GetPlayMode() == PLAYMODE_PLAYING) && !thePlayer->GetIsSpectator() && (thePlayer->pev->team != TEAM_IND))
{
theIsRelevant = true;
}
return theIsRelevant;
}
bool AvHPlayer::GetCanBeAffectedByEnemies() const
{
bool theCanBeAffected = this->GetIsRelevant(false) && !this->GetIsTemporarilyInvulnerable() && !this->GetIsBeingDigested() && !this->GetIsInTopDownMode();
return theCanBeAffected;
}
float AvHPlayer::GetOpacity() const
{
float theOpacity = AvHCloakable::GetOpacity();
//if(this->GetIsBlinking())
//{
// theOpacity = .05f;
//}
return theOpacity;
}
bool AvHPlayer::GetIsMetabolizing() const
{
bool theIsMetabolizing = false;
if((gpGlobals->time < this->mTimeOfMetabolizeEnd) && (this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER4))
{
theIsMetabolizing = true;
}
return theIsMetabolizing;
}
void AvHPlayer::SetTimeOfMetabolizeEnd(float inTime)
{
this->mTimeOfMetabolizeEnd = inTime;
}
bool AvHPlayer::GetIsSelected(int inEntityIndex) const
{
bool theIsSelected = false;
for(EntityListType::const_iterator theIter = this->mSelected.begin(); theIter != this->mSelected.end(); theIter++)
{
if(*theIter == inEntityIndex)
{
theIsSelected = true;
break;
}
}
return theIsSelected;
}
bool AvHPlayer::RemoveSelection(int inEntityIndex)
{
bool theSuccess = false;
EntityListType::iterator theFoundIter = std::find(this->mSelected.begin(), this->mSelected.end(), (EntityInfo)inEntityIndex);
if(theFoundIter != this->mSelected.end())
{
this->mSelected.erase(theFoundIter);
theSuccess = true;
}
return theSuccess;
}
void AvHPlayer::SetSelection(int inEntityIndex, bool inClearPreviousSelection)
{
if(inClearPreviousSelection)
{
this->mSelected.clear();
}
this->mSelected.push_back(inEntityIndex);
}
bool AvHPlayer::GetIsSpectator() const
{
return (this->pev->flags & FL_PROXY);
}
void AvHPlayer::SetIsSpectator()
{
this->pev->flags |= FL_PROXY;
}
float AvHPlayer::GetLastTimeInCommandStation() const
{
return this->mLastTimeInCommandStation;
}
// Clears the player's impulse and sends "invalid action" notification if not a legal action
void AvHPlayer::ValidateClientMoveEvents()
{
// If player tries to execute an alien ability that they don't have, stop them dead.
AvHMessageID theMessageID = (AvHMessageID)this->mCurrentCommand.impulse;
if(theMessageID != MESSAGE_NULL)
{
// Assume it's invalid
bool theIsValid = false;
this->mCurrentCommand.impulse = MESSAGE_NULL;
this->pev->impulse = MESSAGE_NULL;
const float kMinSayingInterval = 3.0f;
// Process universally allowable impulses
switch(theMessageID)
{
case COMM_CHAT_PUBLIC:
case COMM_CHAT_TEAM:
case COMM_CHAT_NEARBY:
case IMPULSE_FLASHLIGHT:
case IMPULSE_SPRAYPAINT:
case IMPULSE_DEMORECORD:
case ADMIN_READYROOM:
theIsValid = true;
break;
case SAYING_1:
case SAYING_2:
case SAYING_3:
case SAYING_4:
case SAYING_5:
case SAYING_6:
case SAYING_7:
case SAYING_8:
case SAYING_9:
// tankefugl: 0000008
// preventing spamming of request and ack
case ORDER_REQUEST:
case ORDER_ACK:
// :tankefugl
if(GetGameRules()->GetCheatsEnabled() || (gpGlobals->time > (this->mTimeOfLastSaying + kMinSayingInterval)))
{
theIsValid = true;
}
else
{
int a = 0;
}
break;
default:
if(GetGameRules()->GetIsCombatMode())
{
theIsValid = true;
}
break;
}
// If not universally allowable
AvHTeam* theTeam = this->GetTeamPointer();
if(!theIsValid && theTeam)
{
AvHUser3 theUser3 = this->GetUser3();
// Marines
if(this->GetIsMarine())
{
// Soldiers
if((theUser3 == AVH_USER3_MARINE_PLAYER) && this->IsAlive())
{
switch(theMessageID)
{
// Validate orders
// tankefugl: 0000008
// preventing spamming of request and ack
//case ORDER_REQUEST:
//case ORDER_ACK:
// :tankefugl
// Validate weapon switches
case WEAPON_NEXT:
case WEAPON_RELOAD:
case WEAPON_DROP:
// Only soldiers can vote
case ADMIN_VOTEDOWNCOMMANDER:
theIsValid = true;
}
}
// Commanders
else if(theUser3 == AVH_USER3_COMMANDER_PLAYER)
{
switch(theMessageID)
{
// Validate commander movement and input
case COMMANDER_MOUSECOORD:
case COMMANDER_MOVETO:
case COMMANDER_SCROLL:
case COMMANDER_DEFAULTORDER:
case COMMANDER_SELECTALL:
case COMMANDER_REMOVESELECTION:
// Validate hotgroup creation and selection
case GROUP_CREATE_1:
case GROUP_CREATE_2:
case GROUP_CREATE_3:
case GROUP_CREATE_4:
case GROUP_CREATE_5:
case GROUP_SELECT_1:
case GROUP_SELECT_2:
case GROUP_SELECT_3:
case GROUP_SELECT_4:
case GROUP_SELECT_5:
// Special impulses to handle requests
case COMMANDER_NEXTIDLE:
case COMMANDER_NEXTAMMO:
case COMMANDER_NEXTHEALTH:
theIsValid = true;
}
// Validate research and building
// Make sure it comes from the commander and also that the right building is selected and not busy
if(AvHSHUGetIsBuildTech(theMessageID) || AvHSHUGetIsResearchTech(theMessageID) || AvHSHUGetDoesTechCostEnergy(theMessageID) || (theMessageID == BUILD_RECYCLE))
{
if(theTeam->GetTechNodes().GetIsMessageAvailableForSelection(theMessageID, this->mSelected))
{
// Make sure only one structure is selected, and make sure the structure that's selected allows this action
if(AvHSHUGetIsBuildTech(theMessageID)) //doesn't depend on selection
{
theIsValid = true;
}
else
{
if(this->mSelected.size() == 1)
{
int theEntityIndex = *this->mSelected.begin();
CBaseEntity* theEntity = CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(theEntityIndex));
AvHBaseBuildable* theBaseBuildable = dynamic_cast<AvHBaseBuildable*>(theEntity);
if(theBaseBuildable && theBaseBuildable->GetIsTechnologyAvailable(theMessageID))
{
theIsValid = true;
}
}
}
}
}
}
}
// Aliens
else if(this->GetIsAlien() && this->IsAlive())
{
//AvHAlienAbilityWeapon* theAlienAbilityWeapon = dynamic_cast<AvHAlienAbilityWeapon*>(this->m_pActiveItem);
switch(theMessageID)
{
// Validate all alien abilities
case ALIEN_ABILITY_LEAP:
case ALIEN_ABILITY_CHARGE:
// Check that weapon is active
if(this->GetHasActiveAlienWeaponWithImpulse(theMessageID))
{
// Check that we have enough energy, and that it's enabled
//if(theAlienAbilityWeapon->IsUseable())
//{
// Deduct energy cost
//ItemInfo theItemInfo;
//theAlienAbilityWeapon->GetItemInfo(&theItemInfo);
//float theEnergy = AvHMUGetEnergyCost(theItemInfo.iId);
theIsValid = true;
//}
}
// Removed by mmcguire.
// This code is no longer needed because the pm_shared code
// now ignores ability impulses that come from the console.
/*
else
{
// If players are trying to use the abilities when not equipped via the impulse key, stop them dead (or worse?)
//VectorCopy(g_vecZero, this->pev->velocity);
//this->pev->fixangle = TRUE;
this->Killed(this->pev, 1);
char theString[512];
sprintf(theString, "Player \"%s\" executed impulse %d illegally and was killed.\n", STRING(this->pev->netname), theMessageID);
ALERT(at_logged, theString);
}
*/
break;
case ALIEN_ABILITY_BLINK:
theIsValid = true;
break;
// Validate lifeform changes and evolutions
case ALIEN_EVOLUTION_ONE:
case ALIEN_EVOLUTION_TWO:
case ALIEN_EVOLUTION_THREE:
case ALIEN_EVOLUTION_SEVEN:
case ALIEN_EVOLUTION_EIGHT:
case ALIEN_EVOLUTION_NINE:
case ALIEN_EVOLUTION_TEN:
case ALIEN_EVOLUTION_ELEVEN:
case ALIEN_EVOLUTION_TWELVE:
case ALIEN_LIFEFORM_ONE:
case ALIEN_LIFEFORM_TWO:
case ALIEN_LIFEFORM_THREE:
case ALIEN_LIFEFORM_FOUR:
case ALIEN_LIFEFORM_FIVE:
case ALIEN_BUILD_RESOURCES:
case ALIEN_BUILD_OFFENSE_CHAMBER:
case ALIEN_BUILD_DEFENSE_CHAMBER:
case ALIEN_BUILD_SENSORY_CHAMBER:
case ALIEN_BUILD_MOVEMENT_CHAMBER:
case ALIEN_BUILD_HIVE:
// Validate weapon switches
case WEAPON_NEXT:
theIsValid = true;
break;
}
}
}
if(theIsValid)
{
this->mCurrentCommand.impulse = theMessageID;
this->pev->impulse = theMessageID;
}
}
}
bool AvHPlayer::GetHasActiveAlienWeaponWithImpulse(AvHMessageID inMessageID) const
{
bool theHasWeapon = false;
for(int i = 0; (i < MAX_ITEM_TYPES) && !theHasWeapon; i++)
{
CBasePlayerItem* theCurrentItem = this->m_rgpPlayerItems[i];
if(theCurrentItem)
{
AvHAlienAbilityWeapon* theBaseWeapon = dynamic_cast<AvHAlienAbilityWeapon*>(theCurrentItem);
while(theBaseWeapon && !theHasWeapon)
{
if((theBaseWeapon->GetAbilityImpulse() == inMessageID) && theBaseWeapon->GetEnabledState() && theBaseWeapon->IsUseable())
{
theHasWeapon = true;
}
theCurrentItem = theCurrentItem->m_pNext;
theBaseWeapon = dynamic_cast<AvHAlienAbilityWeapon*>(theCurrentItem);
}
}
}
return theHasWeapon;
}
bool AvHPlayer::GetHasSeenATeam()
{
return (this->mHasSeenTeamA || this->mHasSeenTeamB);
}
bool AvHPlayer::GetHasSeenTeam(AvHTeamNumber inNumber) const
{
bool theHasBeenOnTeam = false;
if(inNumber == GetGameRules()->GetTeamA()->GetTeamNumber())
{
theHasBeenOnTeam = this->mHasSeenTeamA;
}
else if(inNumber == GetGameRules()->GetTeamB()->GetTeamNumber())
{
theHasBeenOnTeam = this->mHasSeenTeamB;
}
return theHasBeenOnTeam;
}
void AvHPlayer::SetHasSeenTeam(AvHTeamNumber inNumber)
{
if(inNumber == GetGameRules()->GetTeamA()->GetTeamNumber())
{
this->mHasSeenTeamA = true;
}
else if(inNumber == GetGameRules()->GetTeamB()->GetTeamNumber())
{
this->mHasSeenTeamB = true;
}
}
float AvHPlayer::GetTimeOfLastSporeDamage() const
{
return this->mTimeOfLastSporeDamage;
}
void AvHPlayer::SetTimeOfLastSporeDamage(float inTime)
{
this->mTimeOfLastSporeDamage = inTime;
}
// Assumes that the current input is legal
void AvHPlayer::HandleTopDownInput()
{
// From CBasePlayer::PreThink():
bool theAttackOneDown = FBitSet(this->mCurrentCommand.buttons, IN_ATTACK);
// +movement: Removed case for +attack2
bool theAttackTwoDown = false; //FBitSet(this->mCurrentCommand.buttons, IN_ATTACK2);
bool theJumpHit = FBitSet(this->mCurrentCommand.buttons, IN_JUMP);
bool theCrouchDown = FBitSet(this->mCurrentCommand.buttons, IN_DUCK);
// If we are a commander
if(this->pev->iuser3 == AVH_USER3_COMMANDER_PLAYER)
{
this->mLastTimeInCommandStation = gpGlobals->time;
// Fetch world x and world y from move.
// Note: there is some inaccuracy here for two reasons. One, these values were propagated
// over the network and were optimized for it, and two, the selection assumes that the rays
// emanate from the CURRENT view origin. When clicking on one unit, this is right. When
// dragging a box around units, the view origin could have moved substantially. This inaccuracy
// should really only be noticed when drag selecting huge groups of units. Send view origin as well?
this->mMouseWorldPos.x = this->mCurrentCommand.upmove/kSelectionNetworkConstant;
this->mMouseWorldPos.y = this->mCurrentCommand.sidemove/kSelectionNetworkConstant;
this->mMouseWorldPos.z = this->mCurrentCommand.forwardmove/kSelectionNetworkConstant;
// Fetch potential world location of mouse click. worldx -> cmd->upmove, worldy -> cmd->sidemove
//if(this->pev->impulse == COMMANDER_MOUSECOORD)
AvHMessageID theMessageID = (AvHMessageID)this->mCurrentCommand.impulse;
bool theIsBuildTech = AvHSHUGetIsBuildTech(theMessageID);
bool theIsResearchTech = AvHSHUGetIsResearchTech(theMessageID);
bool theIsRecycleMessage = (theMessageID == BUILD_RECYCLE);
if(AvHSHUGetIsGroupMessage(theMessageID))
{
this->GroupMessage(theMessageID);
}
else if(theMessageID == COMMANDER_SELECTALL)
{
bool theClearSelection = true;
// Run through all players on our team
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*)
if(theEntity->GetTeam() == this->GetTeam())
{
if(/*theEntity->GetIsRelevant() &&*/ (theEntity != this))
{
this->SetSelection(theEntity->entindex(), theClearSelection);
theClearSelection = false;
}
}
END_FOR_ALL_ENTITIES(kAvHPlayerClassName)
// Set "selectall" group
//this->mSelectAllGroup = this->mSelected;
this->GetTeamPointer()->SetSelectAllGroup(this->mSelected);
}
else if(theMessageID == COMMANDER_REMOVESELECTION)
{
// TODO:
gSelectionHelper.ClearSelection();
this->mTrackingEntity = 0;
}
else if((theMessageID == COMMANDER_NEXTIDLE) || (theMessageID == COMMANDER_NEXTAMMO) || (theMessageID == COMMANDER_NEXTHEALTH) || theJumpHit)
{
// Jump to first idle soldier and remove from list
bool theSuccess = false;
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
if(theJumpHit)
{
theMessageID = MESSAGE_NULL;
}
AvHAlert theAlert;
if(theTeam->GetLastAlert(theAlert, true, theJumpHit, &theMessageID))
{
int theEntityIndex = theAlert.GetEntityIndex();
CBaseEntity* theBaseEntity = CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(theEntityIndex));
if(theBaseEntity)
{
VectorCopy(theBaseEntity->pev->origin, this->pev->origin);
switch(theMessageID)
{
case COMMANDER_NEXTIDLE:
case COMMANDER_NEXTAMMO:
case COMMANDER_NEXTHEALTH:
// Set selection to player
this->mSelected.clear();
this->mSelected.push_back(theEntityIndex);
break;
}
}
}
}
}
else if((theMessageID == COMMANDER_SCROLL) || (theMessageID == COMMANDER_MOVETO))
{
// Stop tracking
this->mTrackingEntity = 0;
// Commander didn't make a mistake when going to hotgroup, so clear event (so next group select will go to group, not to last position before)
this->mLastSelectEvent = MESSAGE_NULL;
}
else if((theMessageID == COMMANDER_MOUSECOORD) || theIsBuildTech || theIsResearchTech || theIsRecycleMessage /*|| (theMessageID == COMMANDER_DEFAULTORDER)*/)
{
bool theAttackOnePressed = (theAttackOneDown && !this->mAttackOneDown);
bool theAttackTwoPressed = (theAttackTwoDown && !this->mAttackTwoDown);
bool theAttackOneReleased = (!theAttackOneDown && this->mAttackOneDown);
bool theAttackTwoReleased = !theAttackTwoDown && this->mAttackTwoDown;
// if left button is now down and it wasn't previously down
if(theAttackOnePressed)
{
// Save world location of press
this->mAttackOnePressedWorldPos = this->mMouseWorldPos;
}
// if right button just down
if(theAttackTwoPressed)
{
// remember it's world location
this->mAttackTwoPressedWorldPos = this->mMouseWorldPos;
}
Vector theReleasePosition;
if(theAttackOneReleased || theAttackTwoReleased)
{
// Save world position of release.
theReleasePosition = this->mMouseWorldPos;
//char theMessage[256];
//sprintf(theMessage, "LMB released, selecting\n");
//ClientPrint(this->pev, HUD_PRINTTALK, theMessage);
}
// Watch for selection events
if(theMessageID == COMMANDER_MOUSECOORD)
{
if(theAttackOneReleased)
{
if(!this->mPlacingBuilding)
{
if(!AvHToggleUseable(this, this->GetVisualOrigin(), this->mAttackOnePressedWorldPos))
{
// Clear existing selection
//this->mSelected.clear();
//this->mClientSelected.clear();
gSelectionHelper.QueueSelection(this->GetVisualOrigin(), this->mAttackOnePressedWorldPos, theReleasePosition, (AvHTeamNumber)this->pev->team);
}
this->mTimeOfLastSignificantCommanderAction = gpGlobals->time;
}
this->mPlacingBuilding = false;
}
else if(theAttackTwoReleased)
{
// Check to be sure that players are selected
if(this->mSelected.size() > 0)
{
bool theNonPlayerSelected = false;
for(EntityListType::iterator theIter = this->mSelected.begin(); theIter != this->mSelected.end(); theIter++)
{
AvHPlayer* thePlayer = NULL;
AvHSUGetEntityFromIndex(*theIter, thePlayer);
if(!thePlayer)
{
theNonPlayerSelected = true;
break;
}
}
if(!theNonPlayerSelected || GetGameRules()->GetIsCheatEnabled(kcOrderSelf))
{
// Check if right-clicked a useable thing
// Trace to see if the commander clicked a useable thing
//if(!AvHToggleUseable(this, this->pev->origin, this->mAttackTwoPressedWorldPos))
//{
if(!this->GiveOrderToSelection(ORDERTYPEL_DEFAULT, this->mAttackTwoPressedWorldPos))
{
// This location better be off the map or something, default orders should nearly always go through
this->SendMessage(kInvalidOrderGiven, TOOLTIP);
}
else
{
this->mTimeOfLastSignificantCommanderAction = gpGlobals->time;
}
}
}
//}
}
}
// Issue movement order to selection
//else if(theMessageID == COMMANDER_DEFAULTORDER)
//{
// // Build order
// AvHOrder theOrder;
// theOrder.SetOrderType(ORDERTYPEL_DEFAULT);
//
// float theWorldX = this->mCurrentCommand.upmove*kWorldPosNetworkConstant;
// float theWorldY = this->mCurrentCommand.sidemove*kWorldPosNetworkConstant;
// theOrder.SetLocation(vec3_t(theWorldX, theWorldY, 0.0f));
//
// this->GiveOrderToSelection(theOrder);
//}
// Watch for build events
else if(theIsBuildTech)
{
this->mPlacingBuilding = true;
if(GetGameRules()->GetGameStarted())
{
// 551 puzl
// Hack to stop free scans. This should be reworked as a generic solution for energy based build events.
bool theCanBuild=true;
if(this->mSelected.size() > 0)
{
int theEntityForResearch = *this->mSelected.begin();
CBaseEntity* theEntity = AvHSUGetEntityFromIndex(theEntityForResearch);
AvHObservatory* theObs = dynamic_cast<AvHObservatory*>(theEntity);
if ( theObs )
{
if ( (theObs->GetIsTechnologyAvailable(BUILD_SCAN)) == false && (theMessageID == BUILD_SCAN) )
theCanBuild = false;
}
if(!theObs && theMessageID == BUILD_SCAN) {
theCanBuild = false;
}
}
if ( theCanBuild ) {
// puzl
this->BuildTech(theMessageID, this->mAttackOnePressedWorldPos);
this->mTimeOfLastSignificantCommanderAction = gpGlobals->time;
// tankefugl: 0001014
// // If player(s) selected when something built, give default order to it (assumes that players can't be selected along with other non-players)
// if(AvHSHUGetIsBuilding(theMessageID))
// {
// if(this->mSelected.size() > 0)
// {
// int theFirstEntitySelected = *this->mSelected.begin();
// if((theFirstEntitySelected >= 1) && (theFirstEntitySelected <= gpGlobals->maxClients))
// {
// if(!this->GiveOrderToSelection(ORDERTYPEL_DEFAULT, this->mAttackOnePressedWorldPos))
// {
// this->SendMessage(kInvalidOrderGiven, true);
// }
// }
// }
// }
// :tankefugl
}
}
}
else if(theIsResearchTech)
{
if(GetGameRules()->GetGameStarted())
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam && (theTeam->GetResearchManager().GetIsMessageAvailable(theMessageID)))
{
if(this->mSelected.size() == 1)
{
int theEntityForResearch = *this->mSelected.begin();
this->Research(theMessageID, theEntityForResearch);
this->mTimeOfLastSignificantCommanderAction = gpGlobals->time;
}
}
}
}
// Check for recycling action
else if(theIsRecycleMessage)
{
for(EntityListType::iterator theIter = this->mSelected.begin(); theIter != this->mSelected.end(); theIter++)
{
AvHBaseBuildable* theBuildable = NULL;
AvHSUGetEntityFromIndex(*theIter, theBuildable);
if(theBuildable)
{
if((theBuildable->pev->team == this->pev->team) && (theBuildable->pev->team != 0))
{
theBuildable->StartRecycle();
this->mTimeOfLastSignificantCommanderAction = gpGlobals->time;
const char* theBuildableName = STRING(theBuildable->pev->classname);
this->LogPlayerAction("recycle", theBuildableName);
}
}
}
}
// Update
this->mAttackOneDown = theAttackOneDown;
this->mAttackTwoDown = theAttackTwoDown;
}
}
else
{
// TODO: Make sure we think this player is a commander to prevent cheating on client
// TODO: If they sent COMMANDER_MOUSECOORD and they aren't a commander, there's trouble
}
// Process selections if waiting
gSelectionHelper.ProcessPendingSelections();
if(gSelectionHelper.SelectionResultsWaiting())
{
EntityListType theNewSelection;
gSelectionHelper.GetAndClearSelection(theNewSelection);
//string theMessage = "selecting: ";
// If crouch is down
bool theToggleSelection = theCrouchDown;
if(theToggleSelection)
{
bool theNewSelectionIsAllPlayers = true;
// Check to make sure the whole new selection is all players
for(EntityListType::iterator theIterator = theNewSelection.begin(); theIterator != theNewSelection.end(); theIterator++)
{
if(*theIterator > gpGlobals->maxClients)
{
theNewSelectionIsAllPlayers = false;
}
}
// If not, clear the current selection and clear crouch
if(!theNewSelectionIsAllPlayers)
{
theToggleSelection = false;
}
}
if(!theToggleSelection)
{
this->mSelected.clear();
}
// Get player for each one, make sure it's selectable by this player
for(EntityListType::iterator theIterator = theNewSelection.begin(); theIterator != theNewSelection.end(); /* no increment*/)
{
// If this player is selectable
AvHPlayer* thePlayer = NULL;
AvHSUGetEntityFromIndex(*theIterator, thePlayer);
if(!thePlayer || GetGameRules()->GetIsPlayerSelectableByPlayer(thePlayer, this))
{
// Add to debug message
//char theNumber[16];
//sprintf(theNumber, "%d ", *theIterator);
//theMessage += string(theNumber);
int theCurrentEntity = *theIterator;
// Toggle selection of this player
if(theToggleSelection)
{
// Is entity is already selected?
EntityListType::iterator theFindIter = std::find(this->mSelected.begin(), this->mSelected.end(), theCurrentEntity);
bool theEntityIsSelected = (theFindIter != this->mSelected.end());
if(theEntityIsSelected)
{
this->mSelected.erase(theFindIter);
}
else
{
this->SetSelection(theCurrentEntity, false);
}
}
else
{
this->SetSelection(theCurrentEntity, false);
}
// Increment
theIterator++;
this->mTrackingEntity = 0;
}
else
{
// erase it from the list
theIterator = theNewSelection.erase(theIterator);
}
}
// If we selected at least one unit, display debug message
// if(this->mSelected.size() > 0)
// {
// theMessage += "\n";
// ClientPrint(this->pev, HUD_PRINTTALK, theMessage.c_str());
// }
}
// Temporary for fun
// AvHTeam* theTeam = this->GetTeamPointer();
// if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN))
// {
// if(theAttackOneDown && (this->GetUser3() != ROLE_ALIEN2) && (this->GetUser3() != ROLE_ALIEN5))
// {
// this->PlayRandomRoleSound(kPlayerLevelAttackSoundList, CHAN_WEAPON);
// }
// }
}
void AvHPlayer::Jump()
{
Vector vecWallCheckDir;// direction we're tracing a line to find a wall when walljumping
Vector vecAdjustedVelocity;
Vector vecSpot;
TraceResult tr;
if (FBitSet(pev->flags, FL_WATERJUMP))
return;
if (pev->waterlevel >= 2)
{
return;
}
// If this isn't the first frame pressing the jump button, break out.
if ( !FBitSet( m_afButtonPressed, IN_JUMP ) )
return; // don't pogo stick
if ( !(pev->flags & FL_ONGROUND) || !pev->groundentity )
{
return;
}
// many features in this function use v_forward, so makevectors now.
UTIL_MakeVectors (pev->angles);
SetAnimation( PLAYER_JUMP );
if ( FBitSet(pev->flags, FL_DUCKING ) || FBitSet(m_afPhysicsFlags, PFLAG_DUCKING) )
{
//if ( m_fLongJump && (pev->button & IN_DUCK) && gpGlobals->time - m_flDuckTime < 1 && pev->velocity.Length() > 50 )
if ( (pev->button & IN_DUCK) && gpGlobals->time - m_flDuckTime < 1 && pev->velocity.Length() > 50 )
{// If jump pressed within a second of duck while moving, long jump!
SetAnimation( PLAYER_SUPERJUMP );
}
}
// If you're standing on a conveyor, add it's velocity to yours (for momentum)
entvars_t *pevGround = VARS(pev->groundentity);
if ( pevGround && (pevGround->flags & FL_CONVEYOR) )
{
pev->velocity = pev->velocity + pev->basevelocity;
}
}
void AvHPlayer::Killed( entvars_t *pevAttacker, int iGib )
{
if(this->GetUser3() != AVH_USER3_COMMANDER_PLAYER)
{
// Save death position
this->mKilledX = this->pev->origin.x;
this->mKilledY = this->pev->origin.y;
this->mIsScreaming = false;
this->PackDeadPlayerItems();
//this->StopDigestion(false);
this->ResetBehavior(false);
if(GetGameRules()->GetIsCombatMode())
{
this->ProcessCombatDeath();
}
// Fade out already performed when we start being digested
bool theFadeOut = !this->GetIsBeingDigested();
if(!this->GetIsBeingDigested())
{
if(theFadeOut)
{
Vector theFadeColor;
theFadeColor.x = 0;
theFadeColor.y = 0;
theFadeColor.z = 0;
UTIL_ScreenFade(this, theFadeColor, kTransitionFadeTime, 0.0f, 255, FFADE_OUT | FFADE_STAYOUT);
}
}
else
{
SetUpgradeMask(&this->pev->iuser4, MASK_DIGESTING, false);
}
// This line caused a dynamic_cast failure when a shotty killed a flier, then it set it's ensnare state to false,
// which tried to deploy his current weapon, which got the most recent weapon it used, which was garbage for some reason
//this->SetEnsnareState(false);
SetUpgradeMask(&this->pev->iuser4, MASK_ENSNARED, false);
this->mTimeToBeUnensnared = -1;
this->mLastTimeEnsnared = -1;
this->mAlienSightActive = false;
this->mEvolution = MESSAGE_NULL;
this->SetUsedKilled(false);
this->PlayRandomRoleSound(kPlayerLevelDieSoundList);
this->Uncloak();
int thePriority = 0;
bool theIsDramatic = false;
switch(this->GetUser3(false))
{
case AVH_USER3_ALIEN_EMBRYO:
thePriority = kGestateDeathPriority;
break;
case AVH_USER3_ALIEN_PLAYER2:
thePriority = kGorgeDeathPriority;
break;
case AVH_USER3_ALIEN_PLAYER3:
thePriority = kLerkDeathPriority;
break;
case AVH_USER3_ALIEN_PLAYER4:
thePriority = kFadeDeathPriority;
break;
case AVH_USER3_ALIEN_PLAYER5:
thePriority = kOnosDeathPriority;
theIsDramatic = true;
break;
}
if(this->GetHasHeavyArmor())
{
thePriority = kHeavyDeathPriority;
}
if(thePriority > 0)
{
GetGameRules()->MarkDramaticEvent(thePriority, this->entindex(), theIsDramatic, (entvars_t*)pevAttacker);
}
// Added this to make player models fade out
this->pev->solid = SOLID_NOT;
// We can only die when we're playing, not in ready room, reinforcement mode or observation
// (see ClientKill)
CBasePlayer::Killed(pevAttacker, iGib);
//voogru: remove parasite flag if they are a marine.
if(this->GetIsMarine())
SetUpgradeMask(&this->pev->iuser4, MASK_PARASITED, false);
this->EffectivePlayerClassChanged();
GetGameRules()->TriggerAlert((AvHTeamNumber)this->pev->team, ALERT_PLAYER_DIED, this->entindex());
// // Print death anim
// const char* theMainSequence = this->LookupSequence(this->pev->sequence);
// if(!theMainSequence)
// {
// theMainSequence = "None";
// }
//
// const char* theGaitSequence = this->LookupSequence(this->pev->gaitsequence);
// if(!theGaitSequence)
// {
// theGaitSequence = "None";
// }
//
// char theMessage[128];
// sprintf(theMessage, "Death animation: %s, %s", theMainSequence, theGaitSequence);
// UTIL_SayText(theMessage, this);
}
}
//Activity AvHPlayer::GetDeathActivity (void)
//{
// Activity theDeathActivity = ACT_DIESIMPLE;
//
// switch(RANDOM_LONG(0, 3))
// {
// case 1:
// theDeathActivity = ACT_DIEBACKWARD;
// break;
// case 2:
// theDeathActivity = ACT_DIEFORWARD;
// break;
// case 3:
// theDeathActivity = ACT_DIEVIOLENT;
// break;
// }
//
// return theDeathActivity;
//}
void AvHPlayer::NextWeapon()
{
if(this->GetIsAbleToAct())
{
CBasePlayerWeapon* theActiveWeapon = dynamic_cast<CBasePlayerWeapon*>(this->m_pActiveItem);
if(theActiveWeapon)
{
CBasePlayerWeapon* theNextWeapon = dynamic_cast<CBasePlayerWeapon*>(theActiveWeapon->m_pNext);
if(theNextWeapon)
{
this->m_pLastItem = this->m_pActiveItem;
this->m_pActiveItem->Holster();
this->m_pActiveItem = theNextWeapon;
this->m_pActiveItem->Deploy();
}
else
{
int theSlot = theActiveWeapon->iItemSlot();
for(int i = 0; i < MAX_ITEM_TYPES; i++)
{
theNextWeapon = dynamic_cast<CBasePlayerWeapon*>(this->m_rgpPlayerItems[(theSlot + i + 1) % MAX_ITEM_TYPES]);
if(theNextWeapon)
{
this->m_pLastItem = this->m_pActiveItem;
this->m_pActiveItem->Holster();
this->m_pActiveItem = theNextWeapon;
this->m_pActiveItem->Deploy();
break;
}
}
}
}
}
}
void AvHPlayer::ObserverModeIllegal()
{
// Blackout screen or something if observers aren't allowed
UTIL_ScreenFade( CBaseEntity::Instance(this->edict()), Vector(128,0,0), 6, 15, 255, FFADE_OUT | FFADE_MODULATE );
this->SendMessage(kNoSpectating);
}
void AvHPlayer::PackDeadPlayerItems(void)
{
//to do - drop everything that's not in the standard loadout + LMG.
this->DropItem(kwsMachineGun);
this->DropItem(kwsShotGun);
this->DropItem(kwsHeavyMachineGun);
this->DropItem(kwsGrenadeGun);
this->DropItem(kwsMine);
this->DropItem(kwsWelder);
}
void AvHPlayer::PlayRandomRoleSound(string inSoundListName, int inChannel, float inVolume)
{
char theListName[256];
AvHUser3 theUser3 = this->GetUser3();
if(theUser3 != AVH_USER3_NONE)
{
sprintf(theListName, inSoundListName.c_str(), (int)theUser3);
gSoundListManager.PlaySoundInList(theListName, this, inChannel, inVolume);
}
}
float AvHPlayer::GetTimeOfLastConstructUse() const
{
return this->mTimeOfLastConstructUse;
}
void AvHPlayer::SetTimeOfLastConstructUse(float inTime)
{
this->mTimeOfLastConstructUse = inTime;
}
void AvHPlayer::PlayerConstructUse()
{
AvHTeam* theTeam = this->GetTeamPointer();
ASSERT(theTeam);
if(theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE)
{
this->HolsterWeaponToUse();
}
else if(theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN)
{
const float kConstructAnimationInterval = 1.1f;
const int kConstructAnimationIndex = 4;
if((this->mTimeOfLastConstructUseAnimation == 0) || (gpGlobals->time > (this->mTimeOfLastConstructUseAnimation + kConstructAnimationInterval)))
{
// Play special builder animation
PLAYBACK_EVENT_FULL(0, this->edict(), gWeaponAnimationEventID, 0, this->pev->origin, (float *)&g_vecZero, 0.0, 0.0, 0, kConstructAnimationIndex, 0, 0 );
// Delay idle
if(this->m_pActiveItem)
{
CBasePlayerWeapon* theCurrentWeapon = (CBasePlayerWeapon *)this->m_pActiveItem->GetWeaponPtr();
if(theCurrentWeapon)
{
theCurrentWeapon->m_flTimeWeaponIdle += 2.0f;
}
}
this->mTimeOfLastConstructUseAnimation = gpGlobals->time;
}
}
}
bool AvHPlayer::PlaySaying(AvHMessageID inMessageID)
{
bool thePlayedSaying = false;
char theSoundList[256];
memset(theSoundList, 0, 256*sizeof(char));
if(this->pev->iuser3 == AVH_USER3_MARINE_PLAYER)
{
switch(inMessageID)
{
case SAYING_1:
case SAYING_2:
case SAYING_3:
case SAYING_4:
case SAYING_5:
case SAYING_6:
case SAYING_7:
case SAYING_8:
case SAYING_9:
sprintf(theSoundList, kSoldierSayingList, (inMessageID - SAYING_1 + 1));
break;
case ORDER_REQUEST:
strcpy(theSoundList, kSoldierOrderRequestList);
break;
case ORDER_ACK:
strcpy(theSoundList, kSoldierOrderAckList);
break;
}
}
else if(this->pev->iuser3 == AVH_USER3_COMMANDER_PLAYER)
{
switch(inMessageID)
{
case SAYING_1:
case SAYING_2:
case SAYING_3:
case SAYING_4:
case SAYING_5:
case SAYING_6:
case SAYING_7:
case SAYING_8:
case SAYING_9:
sprintf(theSoundList, kCommanderSayingList, (inMessageID - SAYING_1 + 1));
break;
}
}
else if(this->GetIsAlien())
{
switch(inMessageID)
{
case SAYING_1:
case SAYING_2:
case SAYING_3:
case SAYING_4:
case SAYING_5:
case SAYING_6:
case SAYING_8:
// puzl: 0001088 added SAYING_8
sprintf(theSoundList, kAlienSayingList, (inMessageID - SAYING_1 + 1));
break;
}
}
if(strcmp(theSoundList, ""))
{
if(inMessageID == ORDER_REQUEST)
{
this->mOrdersRequested = true;
this->mOrderAcknowledged = false;
this->mIsSpeaking = false;
}
else if(inMessageID == ORDER_ACK)
{
this->mOrderAcknowledged = true;
this->mIsSpeaking = false;
this->mOrdersRequested = false;
}
else
{
this->mIsSpeaking = true;
this->mOrderAcknowledged = false;
this->mOrdersRequested = false;
}
this->mTimeOfLastSaying = gpGlobals->time;
this->mLastSaying = inMessageID;
//int pitch = 95;// + RANDOM_LONG(0,29);
//EMIT_SOUND_DYN(ENT(pev), CHAN_VOICE, theSaying, 1, ATTN_NORM, 0, pitch);
gSoundListManager.PlaySoundInList(theSoundList, this, CHAN_VOICE, 1.0f);
thePlayedSaying = true;
}
return thePlayedSaying;
}
bool AvHPlayer::PlayHUDSound(AvHHUDSound inSound) const
{
return this->PlayHUDSound( inSound, pev->origin[0], pev->origin[1] );
}
bool AvHPlayer::PlayHUDSound(AvHHUDSound inSound, float x, float y) const
{
bool theSuccess = false;
if((inSound > HUD_SOUND_INVALID) && (inSound < HUD_SOUND_MAX))
{
NetMsg_PlayHUDNotification( this->pev, 0, inSound, x, y );
theSuccess = true;
}
return theSuccess;
}
void AvHPlayer::PlayHUDStructureNotification(AvHMessageID inMessageID, const Vector& inLocation)
{
if(GetGameRules()->GetGameStarted())
{
// This player built a structure. Tell all his teammates
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*)
if(theEntity->GetIsRelevant() && !theEntity->GetIsBeingDigested())
{
// Don't send our own messages to ourself unless cheats are enabled
if(GetGameRules()->GetCheatsEnabled() || (this != theEntity))
{
bool theShowNotification = false;
// Show to friendlies...
if(theEntity->pev->team == this->pev->team)
{
theShowNotification = true;
}
if(theShowNotification)
{
NetMsg_PlayHUDNotification( theEntity->pev, 1, inMessageID, inLocation.x, inLocation.y );
}
}
}
END_FOR_ALL_ENTITIES(kAvHPlayerClassName);
}
}
void AvHPlayer::ProcessEvolution()
{
this->SaveHealthArmorPercentages();
UTIL_EmitAmbientSound(ENT(this->pev), this->pev->origin, kGestationSound, 1.0f, .5, SND_STOP, 100);
// Go back to the role we were before, unless we're morphing into a new lifeform
AvHUser3 theNewUser3 = this->mPreviousUser3;
bool theChangedLifeforms = false;
switch(this->mEvolution)
{
case ALIEN_LIFEFORM_ONE:
theNewUser3 = AVH_USER3_ALIEN_PLAYER1;
theChangedLifeforms = true;
break;
case ALIEN_LIFEFORM_TWO:
theNewUser3 = AVH_USER3_ALIEN_PLAYER2;
theChangedLifeforms = true;
break;
case ALIEN_LIFEFORM_THREE:
theNewUser3 = AVH_USER3_ALIEN_PLAYER3;
theChangedLifeforms = true;
break;
case ALIEN_LIFEFORM_FOUR:
theNewUser3 = AVH_USER3_ALIEN_PLAYER4;
theChangedLifeforms = true;
break;
case ALIEN_LIFEFORM_FIVE:
theNewUser3 = AVH_USER3_ALIEN_PLAYER5;
theChangedLifeforms = true;
break;
}
// This shouldn't be needed now that SetUser3 takes into account moving the origin.
// Position player a bit higher off the ground so he doesn't get stuck
//this->pev->origin.z += AvHMUGetOriginOffsetForUser3(theNewUser3);
if(theChangedLifeforms)
{
if(GetGameRules()->GetIsCombatMode())
this->SetLifeformCombatNodesAvailable(false); // Only allow one lifeform change
else
{
int iUpgrades = MASK_UPGRADE_1
| MASK_UPGRADE_2
| MASK_UPGRADE_3
| MASK_UPGRADE_4
| MASK_UPGRADE_5
| MASK_UPGRADE_6
| MASK_UPGRADE_7
| MASK_UPGRADE_8
| MASK_UPGRADE_9
| MASK_UPGRADE_10
| MASK_UPGRADE_11
| MASK_UPGRADE_12
| MASK_UPGRADE_13
| MASK_UPGRADE_14
| MASK_UPGRADE_15;
SetUpgradeMask(&this->pev->iuser4, (AvHUpgradeMask)iUpgrades, false);
}
}
this->SetUser3(theNewUser3, true);
//this->mPreviousUser3 = User3_UNDEFINED;
// int theMaxArmor = 0;
// float theArmorPercentage = 0;
// int theNewMaxArmor = 0;
switch(this->mEvolution)
{
case ALIEN_EVOLUTION_ONE:
case ALIEN_EVOLUTION_TWO:
case ALIEN_EVOLUTION_THREE:
case ALIEN_EVOLUTION_SEVEN:
case ALIEN_EVOLUTION_EIGHT:
case ALIEN_EVOLUTION_NINE:
case ALIEN_EVOLUTION_TEN:
case ALIEN_EVOLUTION_ELEVEN:
case ALIEN_EVOLUTION_TWELVE:
// If it's the exoskeleton upgrade, upgrade now
// theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, this->mUser3);
// theArmorPercentage = this->pev->armorvalue/theMaxArmor;
ProcessGenericUpgrade(this->pev->iuser4, this->mEvolution);
// If player has already decided on the direction to upgrade, spend any extra upgrade levels in that category.
// This has to happen immediately, or else spamming the upgrade button can get another evolution before this is called in InternalAlienUpgradesThink.
AvHTeam* theTeamPointer = this->GetTeamPointer();
ASSERT(theTeamPointer);
AvHAlienUpgradeListType theUpgrades = theTeamPointer->GetAlienUpgrades();
AvHAddHigherLevelUpgrades(theUpgrades, this->pev->iuser4);
//this->PlayHUDSound(HUD_SOUND_ALIEN_UPGRADECOMPLETE);
// if(this->mEvolution == ALIEN_EVOLUTION_TWO)
// {
// theNewMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, this->mUser3);
// this->pev->armorvalue = theArmorPercentage*theNewMaxArmor;
// }
break;
}
this->PlayRandomRoleSound(kPlayerLevelSpawnSoundList, CHAN_ITEM, this->GetAlienAdjustedEventVolume());
this->pev->fuser3 = 1.0f*kNormalizationNetworkFactor;
this->RevertHealthArmorPercentages();
this->mEvolution = MESSAGE_NULL;
}
void AvHPlayer::RevertHealthArmorPercentages()
{
// Preserve armor and health percentages
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
this->pev->health = max(this->mHealthPercentBefore*theMaxHealth,1.0f);//voogru: prevent bug with players evolving down from higher lifeform from getting negative health but still "alive"
int theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
this->pev->armorvalue = max(this->mArmorPercentBefore*theMaxArmor, 0.0f);
// Assumes a push/pop kind of deal
this->mHealthPercentBefore = this->mArmorPercentBefore = 1.0f;
}
void AvHPlayer::SaveHealthArmorPercentages()
{
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
this->mHealthPercentBefore = this->pev->health/(float)theMaxHealth;
this->mHealthPercentBefore = min(max(0.0f, this->mHealthPercentBefore), 1.0f);
int theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
this->mArmorPercentBefore = this->pev->armorvalue/(float)theMaxArmor;
this->mArmorPercentBefore = min(max(0.0f, this->mArmorPercentBefore), 1.0f);
}
void AvHPlayer::ProcessResourceAdjustment(AvHMessageID inMessageID)
{
// TODO: Mark cheater if this isn't the case?
if(this->pev->iuser3 == AVH_USER3_COMMANDER_PLAYER)
{
// Make sure we have resource tower selected
if(this->mSelected.size() == 1)
{
int theEntIndex = *this->mSelected.begin();
AvHFuncResource* theFuncResource;
if(AvHSUGetEntityFromIndex(theEntIndex, theFuncResource))
{
// Get particle system
int theParticleSystemIndex = theFuncResource->GetParticleSystemIndex();
AvHParticleSystemEntity* theParticleSystemEntity = NULL;
if(AvHSUGetEntityFromIndex(theParticleSystemIndex, theParticleSystemEntity))
{
// Adjust particle system for resource tower
// Get custom data
uint16 theCustomData = theParticleSystemEntity->GetCustomData();
// Adjust custom data by inMessageID
ASSERT(false);
//switch(inMessageID)
//{
//case RESOURCE_ADJUST_ONE_UP:
// theCustomData = min(theCustomData+1, 0xF);
// break;
//case RESOURCE_ADJUST_ONE_DOWN:
// theCustomData = max(theCustomData-1, 0);
// break;
//}
// Set custom data again
theParticleSystemEntity->SetCustomData(theCustomData);
}
}
}
}
}
void AvHPlayer::Evolve(AvHMessageID inMessageID, bool inInstantaneous)
{
// TODO: Put in a waiting time or some other effects?
if(this->GetTeamPointer()->GetTeamType() == AVH_CLASS_TYPE_ALIEN)
{
this->PlayHUDSound(HUD_SOUND_POINTS_SPENT);
int theDramaticPriority = 0;
switch(inMessageID)
{
case ALIEN_LIFEFORM_ONE:
theDramaticPriority = kEvolveLevelOnePriority;
break;
case ALIEN_LIFEFORM_TWO:
theDramaticPriority = kEvolveLevelTwoPriority;
break;
case ALIEN_LIFEFORM_THREE:
theDramaticPriority = kEvolveLevelThreePriority;
break;
case ALIEN_LIFEFORM_FOUR:
theDramaticPriority = kEvolveLevelFourPriority;
break;
case ALIEN_LIFEFORM_FIVE:
theDramaticPriority = kEvolveLevelFivePriority;
break;
default:
theDramaticPriority = kEvolveUpgradePriority;
break;
}
if(!inInstantaneous)
{
GetGameRules()->MarkDramaticEvent(theDramaticPriority, this);
this->mTimeGestationStarted = gpGlobals->time;
this->mPreviousUser3 = (AvHUser3)this->pev->iuser3;
this->mEvolution = inMessageID;
this->SetUser3(AVH_USER3_ALIEN_EMBRYO);
this->BecomePod();
}
else
{
this->mPreviousUser3 = (AvHUser3)this->pev->iuser3;
this->mEvolution = inMessageID;
this->ProcessEvolution();
}
}
}
void AvHPlayer::LogEmitRoleChange()
{
const char* theUser3Name = AvHSHUGetClassNameFromUser3((AvHUser3)this->pev->iuser3);
if(theUser3Name != NULL)
{
UTIL_LogPrintf("%s changed role to \"%s\"\n",
GetLogStringForPlayer( this->edict() ).c_str(),
theUser3Name
);
}
}
void AvHPlayer::LogPlayerAction(const char* inActionDescription, const char* inActionData)
{
if(inActionDescription && inActionData)
{
UTIL_LogPrintf("%s triggered \"%s\" (type \"%s\")\n",
GetLogStringForPlayer( this->edict() ).c_str(),
inActionDescription,
inActionData);
}
}
void AvHPlayer::LogPlayerActionPlayer(CBasePlayer* inActionPlayer, const char* inAction)
{
if(inAction)
{
UTIL_LogPrintf("%s triggered \"%s\" against %s\n",
GetLogStringForPlayer( inActionPlayer->edict() ).c_str(),
inAction,
GetLogStringForPlayer( this->edict() ).c_str()
);
}
}
void AvHPlayer::LogPlayerAttackedPlayer(CBasePlayer* inAttackingPlayer, const char* inWeaponName, float inDamage)
{
if(inWeaponName)
{
edict_t* theAttacker = inAttackingPlayer->edict();
edict_t* theReceiver = this->edict();
bool theLogAttack = false;
int theLogDetail = CVAR_GET_FLOAT(kvLogDetail);
if((theLogDetail > 0) && (theAttacker->v.team != theReceiver->v.team))
{
theLogAttack = true;
}
else if((theLogDetail > 1) && (theAttacker->v.team == theReceiver->v.team))
{
theLogAttack = true;
}
else if(theLogDetail > 2)
{
theLogAttack = true;
}
if(theLogAttack)
{
// Remove "weapon_" prefix
string theKillerName(inWeaponName);
AvHSHUMakeViewFriendlyKillerName(theKillerName);
int theDamage = (int)inDamage;
UTIL_LogPrintf("%s attacked %s with \"%s\" (damage \"%d\")\n",
GetLogStringForPlayer( theAttacker ).c_str(),
GetLogStringForPlayer( theReceiver ).c_str(),
theKillerName.c_str(),
theDamage
);
}
}
}
void AvHPlayer::LogPlayerKilledPlayer(CBasePlayer* inAttackingPlayer, const char* inWeaponName)
{
if(inWeaponName)
{
edict_t* theAttacker = inAttackingPlayer->edict();
edict_t* theReceiver = this->edict();
bool theLogAttack = false;
int theLogDetail = CVAR_GET_FLOAT(kvLogDetail);
if((theLogDetail > 0) && (theAttacker->v.team != theReceiver->v.team))
{
theLogAttack = true;
}
else if((theLogDetail > 1) && (theAttacker->v.team == theReceiver->v.team))
{
theLogAttack = true;
}
else if(theLogDetail > 2)
{
theLogAttack = true;
}
if(theLogAttack)
{
// Remove "weapon_" prefix
string theKillerName(inWeaponName);
AvHSHUMakeViewFriendlyKillerName(theKillerName);
UTIL_LogPrintf("%s killed %s with \"%s\"\n",
GetLogStringForPlayer( theAttacker ).c_str(),
GetLogStringForPlayer( theReceiver ).c_str(),
theKillerName.c_str()
);
}
}
}
void AvHPlayer::Research(AvHMessageID inUpgrade, int inEntityIndex)
{
if(this->GetUser3() == AVH_USER3_COMMANDER_PLAYER)
{
bool theIsResearchable;
int theResearchCost;
float theResearchTime;
AvHTeam* theTeam = this->GetTeamPointer();
CBaseEntity* theEntity = AvHSUGetEntityFromIndex(inEntityIndex);
if(theEntity && theTeam && (theEntity->pev->team == this->pev->team))
{
AvHResearchManager& theResearchManager = theTeam->GetResearchManager();
if(inUpgrade == MESSAGE_CANCEL)
{
// Remember research tech and time done
float thePercentageComplete = 0.0f;
AvHMessageID theCancelledTechnology = MESSAGE_NULL;
if(theResearchManager.CancelResearch(inEntityIndex, thePercentageComplete, theCancelledTechnology))
{
if(theResearchManager.GetResearchInfo(theCancelledTechnology, theIsResearchable, theResearchCost, theResearchTime))
{
ASSERT(thePercentageComplete >= 0.0f);
ASSERT(thePercentageComplete < 1.0f);
const float kRefundFactor = 1.0f;
float theRefund = kRefundFactor*(1.0f - thePercentageComplete)*theResearchCost;
theRefund = min(theRefund, (float)theResearchCost);
this->SetResources(this->GetResources() + theRefund);
}
char* theResearchName = NULL;
if(AvHSHUGetResearchTechName(inUpgrade, theResearchName))
{
this->LogPlayerAction("research_cancel", theResearchName);
}
}
}
else if(theResearchManager.GetResearchInfo(inUpgrade, theIsResearchable, theResearchCost, theResearchTime))
{
// Look up cost, deduct from player
if(this->GetResources() >= theResearchCost)
{
// Remember research tech and time done
if(theResearchManager.SetResearching(inUpgrade, inEntityIndex))
{
this->PayPurchaseCost(theResearchCost);
// Tell everyone on this team about research and time research done, so their
// HUD is updated, and so they can't then walk up to station and start researching something else
this->PlayHUDStructureNotification(inUpgrade, theEntity->pev->origin);
char* theResearchName = NULL;
if(AvHSHUGetResearchTechName(inUpgrade, theResearchName))
{
this->LogPlayerAction("research_start", theResearchName);
}
}
}
else
{
this->PlayHUDSound(HUD_SOUND_MARINE_MORE);
}
}
}
}
}
// Digestion functions
int AvHPlayer::GetDigestee() const
{
return this->mDigestee;
}
void AvHPlayer::SetDigestee(int inPlayerID)
{
this->mDigestee = inPlayerID;
}
void AvHPlayer::StartDigestion(int inDigestee)
{
AvHPlayer* theDigestee = dynamic_cast<AvHPlayer*>(CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(inDigestee)));
if(theDigestee)
{
theDigestee->SetBeingDigestedMode(true);
SetUpgradeMask(&this->pev->iuser4, MASK_DIGESTING, true);
this->SetDigestee(inDigestee);
EMIT_SOUND(ENT(this->pev), CHAN_VOICE, kDevourSwallowSound, this->GetAlienAdjustedEventVolume(), ATTN_NORM);
this->mTimeOfLastDigestDamage = gpGlobals->time;
}
}
void AvHPlayer::StopDigestion(bool inDigested)
{
// Play digest complete sound and stop digestion
int theDigesteeIndex = this->GetDigestee();
this->SetDigestee(0);
AvHPlayer* theDigestee = dynamic_cast<AvHPlayer*>(CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(theDigesteeIndex)));
if(theDigestee)
{
if(inDigested)
{
this->SetDesiredRoomType(0, true);
EMIT_SOUND(ENT(this->pev), CHAN_VOICE, kDevourCompleteSound, this->GetAlienAdjustedEventVolume(), ATTN_NORM);
}
else
{
theDigestee->SetBeingDigestedMode(false);
EMIT_SOUND(ENT(this->pev), CHAN_VOICE, kDevourCancelSound, this->GetAlienAdjustedEventVolume(), ATTN_NORM);
}
}
SetUpgradeMask(&this->pev->iuser4, MASK_DIGESTING, false);
}
void AvHPlayer::InternalDigestionThink()
{
// If we have a digestee
int theDigesteeIndex = this->GetDigestee();
AvHPlayer* theDigestee = dynamic_cast<AvHPlayer*>(CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(theDigesteeIndex)));
if(theDigestee)
{
bool thePlayerWasDigested = false;
// If digestee is alive and still in the game (hasn't disconnected or switched teams)
if(theDigestee->GetIsRelevant())
{
if(RANDOM_LONG(0, 110) == 0)
{
// Play digesting sound occasionally
EMIT_SOUND(this->edict(), CHAN_AUTO, kDigestingSound, this->GetAlienAdjustedEventVolume(), ATTN_NORM);
}
// Do damage to digestee
float theTimePassed = gpGlobals->time - this->mTimeOfLastDigestDamage;
if(theTimePassed > 1.0f)
{
// Find devour weapon if we have one, so death message appears properly
entvars_t* theInflictor = this->pev;
CBasePlayerItem* theDevourWeapon = this->HasNamedPlayerItem(kwsDevour);
if(theDevourWeapon)
{
theInflictor = theDevourWeapon->pev;
}
const float theCombatModeScalar = GetGameRules()->GetIsCombatMode() ? BALANCE_VAR(kCombatModeTimeScalar) : 1.0f;
theDigestee->pev->takedamage = DAMAGE_YES;
float theDamage = theTimePassed*BALANCE_VAR(kDevourDamage)*(1.0f/theCombatModeScalar);
theDigestee->TakeDamage(theInflictor, this->pev, theDamage, DMG_DROWN);
theDigestee->pev->takedamage = DAMAGE_NO;
// Get health back too
this->Heal(theDamage, false);
this->mTimeOfLastDigestDamage = gpGlobals->time;
}
// Set the digestee's position to our own
VectorCopy(this->pev->origin, theDigestee->pev->origin);
VectorCopy(this->pev->angles, theDigestee->pev->angles);
// Set status bar estimating how long before player will be digested (for both digestee and digester)
theDigestee->TriggerProgressBar(theDigesteeIndex, 3);
this->TriggerProgressBar(theDigesteeIndex, 3);
// Set fuser3 appropriately
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(theDigestee->pev->iuser4, theDigestee->GetUser3(), this->GetExperienceLevel());
float theDigestingScalar = (((float)theMaxHealth - theDigestee->pev->health)/(float)theMaxHealth);
/*this->pev->fuser3 =*/ theDigestee->pev->fuser3 = theDigestingScalar*kNormalizationNetworkFactor;
// Set sound effects as player gets more and more digested
int theDesiredRoomType = 26; // strange sounds right before you die
if(theDigestingScalar < .33f)
{
// Water 1
theDesiredRoomType = 14;
}
else if(theDigestingScalar < .66f)
{
// Water 2
theDesiredRoomType = 15;
}
else if(theDigestingScalar < .9f)
{
// Water 3
theDesiredRoomType = 16;
}
theDigestee->SetDesiredRoomType(theDesiredRoomType);
if(theDigestee->pev->health <= 0)
{
thePlayerWasDigested = true;
}
}
// If digestee is dead and no longer relevant
if(!theDigestee->IsAlive() || !theDigestee->GetIsRelevant() || (theDigestee->GetTeam() == this->GetTeam()))
{
this->StopDigestion(thePlayerWasDigested);
}
}
}
bool AvHPlayer::GetDoesCurrentStateStopOverwatch() const
{
AvHBasePlayerWeapon* theWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
const AvHTeam* theTeam = this->GetTeamPointer();
bool theCurrentStateStopsOverwatch = false;
// bool theCurrentStateStopsOverwatch = ((this->pev->button != 0 && !this->mOverwatchFiredThisThink) || /*(this->pev->velocity.Length() > kOverwatchBreakingVelocity) ||*/ !theWeapon || ((this->pev->iuser3 != AVH_USER3_MARINE_PLAYER) && !GetHasUpgrade(this->pev->iuser4, MASK_MARINE_OVERWATCH)));
//
// // Overwatch not valid for alien players
// if(theTeam && (theTeam->GetTeamType() != AVH_CLASS_TYPE_MARINE))
// {
// theCurrentStateStopsOverwatch = true;
// }
return theCurrentStateStopsOverwatch;
}
bool AvHPlayer::GetIsEntityInSight(CBaseEntity* inEntity)
{
bool theSuccess = false;
AvHTeam* theTeam = this->GetTeamPointer();
// Elven - we don't want marines who are being digested to be able to sight anything.
if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE) && GetHasUpgrade(this->pev->iuser4, MASK_DIGESTING))
{
return theSuccess;
}
// Check if the entitiy is in the player's view frustum.
UTIL_MakeVectors ( pev->v_angle );
Vector center = inEntity->pev->origin + (inEntity->pev->maxs + inEntity->pev->mins) / 2;
Vector sightLine = center - (pev->origin + pev->view_ofs);
Vector hSightLine = sightLine - DotProduct(sightLine, gpGlobals->v_up) * gpGlobals->v_up;
Vector vSightLine = sightLine - DotProduct(sightLine, gpGlobals->v_right) * gpGlobals->v_right;
float hDot = DotProduct(hSightLine.Normalize(), gpGlobals->v_forward);
float vDot = DotProduct(vSightLine.Normalize(), gpGlobals->v_forward);
float hAngle = acosf(hDot) * 180 / M_PI;
float vAngle = acosf(vDot) * 180 / M_PI;
if (hAngle > 180) hAngle -= 360;
if (vAngle > 180) vAngle -= 360;
float aspect = 1.333; // 640/480
if (fabs(hAngle) <= pev->fov / 2 && fabs(vAngle) <= pev->fov / (2 * aspect))
{
if (FVisible(inEntity))
{
theSuccess = true;
}
}
// if(GET_RUN_CODE(4096))
// {
// if (!theSuccess)
// {
//
// UTIL_MakeVectors ( pev->v_angle );
//
// const float kMaxDistanceForSighting = 10000;
//
// // Trace a ray in the direction the player is aiming.
//
// Vector theStart = EyePosition();
// Vector theEnd;
// VectorMA(theStart, kMaxDistanceForSighting, gpGlobals->v_forward, theEnd);
//
// TraceResult tr;
// UTIL_TraceLine(theStart, theEnd, dont_ignore_monsters, ignore_glass, ENT(pev), &tr);
//
// if (tr.flFraction != 1 && tr.pHit == ENT(inEntity->pev))
// {
// int a = 0;
// theSuccess = true;
// }
//
// }
// }
// TODO: Make this better so we can see edges of things? What about big aliens?
return theSuccess;
}
char* AvHPlayer::GetPlayerModelKeyName()
{
// Default to marine in ready room
char* theModelKeyName = kSoldierName;
AvHUser3 theUser3 = this->GetUser3();
switch(theUser3)
{
case AVH_USER3_MARINE_PLAYER:
theModelKeyName = kSoldierName;
if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_13))
{
theModelKeyName = kHeavyName;
}
break;
case AVH_USER3_COMMANDER_PLAYER:
theModelKeyName = kCommanderName;
break;
case AVH_USER3_ALIEN_PLAYER1:
theModelKeyName = kAlien1Name;
break;
case AVH_USER3_ALIEN_PLAYER2:
theModelKeyName = kAlien2Name;
break;
case AVH_USER3_ALIEN_PLAYER3:
theModelKeyName = kAlien3Name;
break;
case AVH_USER3_ALIEN_PLAYER4:
theModelKeyName = kAlien4Name;
break;
case AVH_USER3_ALIEN_PLAYER5:
theModelKeyName = kAlien5Name;
break;
case AVH_USER3_ALIEN_EMBRYO:
theModelKeyName = kAlienGestationName;
break;
}
return theModelKeyName;
}
void AvHPlayer::HandleOverwatch(void)
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE))
{
this->mOverwatchFiredThisThink = false;
// If in overwatch
if(this->mInOverwatch)
{
// do we have a target we're firing at?
if(this->mOverwatchTarget != -1)
{
// is target around?
edict_t* theTargetEdict = INDEXENT(this->mOverwatchTarget);
if(!FNullEnt(theTargetEdict))
{
// if so, move aim toward it but not farther than x degrees from starting pos
CBaseEntity* theTarget = CBaseEntity::Instance(theTargetEdict);
this->TurnOverwatchTowardsTarget(theTarget);
// is it within our weapon range?
float theTargetDistance = (theTarget->pev->origin - this->pev->origin).Length();
AvHBasePlayerWeapon* theWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
if(theWeapon)
{
if(theWeapon->GetRange() >= theTargetDistance)
{
// do we have a clear line of sight to it?
bool theCanSeeTarget = this->GetIsEntityInSight(theTarget);
if(theCanSeeTarget || (gpGlobals->time - this->mTimeLastSeenOverwatchTarget < kOverwatchKeepFiringAfterMissingTargetTime))
{
// do we have ammo left?
if(!theWeapon->UsesAmmo() || (theWeapon->m_iClip > 0))
{
// FIRE!
this->pev->button |= IN_ATTACK;
this->mOverwatchFiredThisThink = true;
}
// Update last time we saw our target
if(theCanSeeTarget)
{
this->mTimeLastSeenOverwatchTarget = gpGlobals->time;
// Playback event to increase tension, but keep network usage down (once or twice a second?)
if(this->pev->fuser2 != -1)
{
//if(RANDOM_LONG(0, 100) == 0)
//{
//PLAYBACK_EVENT_FULL(0, this->edict(), gTensionOverwatchEventID, 0, this->pev->origin, (float *)&g_vecZero, 0.0, 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
//}
}
}
}
// if not, has it been a long time since we've seen it, or did we see it die? Reset.
if( (gpGlobals->time - this->mTimeLastSeenOverwatchTarget > kOverwatchLostTargetTime) ||
(theCanSeeTarget && !theTarget->IsAlive()))
{
this->ResetOverwatch();
}
}
}
else
{
// This can happen when dropping weapons or switching roles
this->TurnOffOverwatch();
}
}
else
{
this->ResetOverwatch();
}
}
// is there a new target in range?
else
{
this->AcquireOverwatchTarget();
}
// see if we moved so we're out
if(this->GetDoesCurrentStateStopOverwatch())
{
this->TurnOffOverwatch();
}
}
else
{
if(this->GetDoesCurrentStateStopOverwatch())
{
this->mTimeOfLastOverwatchPreventingAction = gpGlobals->time;
}
// if overwatch is enabled, see if we've been still long enough to put us into it
if(this->mOverwatchEnabled && !(this->pev->flags & FL_FAKECLIENT))
{
if(this->mTimeOfLastOverwatchPreventingAction != -1)
{
if(gpGlobals->time - this->mTimeOfLastOverwatchPreventingAction >= kOverwatchAcquireTime)
{
// if so, set overwatch on, make sure to set the current weapon into overwatch
this->TurnOnOverwatch();
}
}
}
}
}
}
void AvHPlayer::InternalAlienUpgradesThink()
{
// If we're an alien player
AvHTeam* theTeam = this->GetTeamPointer();
if(this->GetIsAlien())
{
if(theTeam && !GetGameRules()->GetIsCombatMode())
{
// Preserve health and armor percentages as we get and remove upgrades
float theHealthPercentage = this->pev->health/AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
float theArmorPercentage = this->pev->armorvalue/AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
ASSERT(theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN);
AvHAlienUpgradeListType theUpgrades = theTeam->GetAlienUpgrades();
// If player has already decided on the direction to upgrade, spend any extra upgrade levels in that category
AvHAddHigherLevelUpgrades(theUpgrades, this->pev->iuser4);
// If we have more upgrades then we should, remove one randomly
int theNumRemoved = AvHRemoveExcessUpgrades(theUpgrades, this->pev->iuser4);
if(theNumRemoved > 0)
{
// Play a sound indicating this has happened
this->PlayHUDSound(HUD_SOUND_ALIEN_UPGRADELOST);
}
// If we're cloaked, and we no longer have any sensory upgrades, trigger uncloak
//int theNumSensoryUpgrades = AvHGetNumUpgradesInCategoryInList(theUpgrades, ALIEN_UPGRADE_CATEGORY_SENSORY);
//if(GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_CLOAKED) && (theNumSensoryUpgrades == 0))
//{
// this->TriggerUncloak();
//}
this->pev->health = theHealthPercentage*AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
this->pev->armorvalue = theArmorPercentage*AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
}
if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN) && this->IsAlive())
{
// If we have cloaking, update our cloak state
this->InternalAlienUpgradesCloakingThink();
// If we have regeneration, heal us
this->InternalAlienUpgradesRegenerationThink();
// If we have pheromones, update them
//this->InternalAlienUpgradesPheromonesThink();
}
}
// Update ensnare here
if(this->GetIsEnsnared())
{
if(gpGlobals->time > this->mTimeToBeUnensnared)
{
this->SetEnsnareState(false);
}
}
// Update stun
if(this->GetIsStunned())
{
if(gpGlobals->time > this->mTimeToBeFreeToMove)
{
this->SetIsStunned(false);
}
}
}
bool AvHPlayer::GetIsCloaked() const
{
bool theIsCloaked = false;
if( (this->GetOpacity() < 0.1f))
{
theIsCloaked = true;
}
return theIsCloaked;
}
bool AvHPlayer::GetIsPartiallyCloaked() const
{
bool theIsCloaked = false;
if( (this->GetOpacity() < 0.6f))
{
theIsCloaked = true;
}
return theIsCloaked;
}
bool AvHPlayer::GetRandomGameStartedTick(float inApproximateFrameRate)
{
bool theTimeToTick = false;
if(GetGameRules()->GetGameStarted() && (inApproximateFrameRate > 0))
{
ASSERT(this->mPreThinkFrameRate > 0);
int theUpperBound = (int)(this->mPreThinkFrameRate/inApproximateFrameRate);
if(RANDOM_LONG(1, theUpperBound) == 1)
{
theTimeToTick = true;
}
}
return theTimeToTick;
}
void AvHPlayer::TriggerUncloak()
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam && (theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN) && this->IsAlive())
{
this->mTriggerUncloak = true;
SetUpgradeMask(&this->pev->iuser4, MASK_SENSORY_NEARBY, false);
this->Uncloak();
}
}
void AvHPlayer::InternalAlienUpgradesPheromonesThink()
{
const float kPheromoneUpdateInterval = .4f;
const float kPheromoneBaseRange = 600;
const int kMaxPheromonePuffs = 3;
if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_8))
{
int theRange = kPheromoneBaseRange;
if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_14))
{
theRange *= 2;
}
else if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_15))
{
theRange *= 3;
}
if(this->mTimeOfLastPheromone == -1 || (gpGlobals->time > this->mTimeOfLastPheromone + kPheromoneUpdateInterval))
{
typedef std::map<EntityInfo, float> PlayerDistanceListType;
PlayerDistanceListType thePlayerDistanceList;
// Look for players in range to draw pheromones of
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*)
// Find nearest distance to friendly and relevant player
if(theEntity->GetIsRelevant() && (theEntity->pev->team != this->pev->team) && !GetHasUpgrade(theEntity->pev->iuser4, MASK_TOPDOWN) && (theEntity != this) /*&& !this->GetIsEntityInSight(theEntity)*/)
{
double theDistance = VectorDistance(theEntity->pev->origin, this->pev->origin);
if(theDistance < theRange)
{
// Choose nearest x players to emit from
thePlayerDistanceList[theEntity->entindex()] = theDistance;
}
}
END_FOR_ALL_ENTITIES(kAvHPlayerClassName)
for(int theNumPheromonePuffs = 0; (theNumPheromonePuffs < kMaxPheromonePuffs) && (thePlayerDistanceList.size() > 0); theNumPheromonePuffs++)
{
// Find the nearest entity
PlayerDistanceListType::iterator theClosestIter = thePlayerDistanceList.begin();
float theClosestDistance = sqrt(kMaxMapDimension*kMaxMapDimension + kMaxMapDimension*kMaxMapDimension);
for(PlayerDistanceListType::iterator theIter = thePlayerDistanceList.begin(); theIter != thePlayerDistanceList.end(); theIter++)
{
float theCurrentRange = theIter->second;
if(theCurrentRange < theClosestDistance)
{
theClosestDistance = theCurrentRange;
theClosestIter = theIter;
}
}
// Play a puff for it
int theCurrentEntityIndex = theClosestIter->first;
CBaseEntity* theCurrentEntity = (CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(theCurrentEntityIndex)));
ASSERT(theCurrentEntity);
AvHSUPlayParticleEvent(kpsPheromoneEffect, this->edict(), theCurrentEntity->pev->origin, FEV_HOSTONLY);
// Delete that entity
thePlayerDistanceList.erase(theClosestIter);
}
this->mTimeOfLastPheromone = gpGlobals->time;
}
}
}
float AvHPlayer::GetCloakTime() const
{
float theCloakTime = AvHCloakable::GetCloakTime();
if(this->GetIsAlien())
{
// If we have cloaking upgrade, we cloak faster
int theCloakingLevel = AvHGetAlienUpgradeLevel(this->pev->iuser4, MASK_UPGRADE_7);
if(theCloakingLevel > 0)
{
theCloakTime = BALANCE_VAR(kCloakTime)/theCloakingLevel;
}
}
return theCloakTime;
}
void AvHPlayer::InternalAlienUpgradesCloakingThink()
{
// joev:
// 0000342 - Cloaking no longer depends on speed.
// For some reason the lerk moves faster when turning
//const float kWalkSpeedFactor = (this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER3) ? .95f : .7f;
//const int kMaxSpeed = this->pev->maxspeed;
//elven - needs to update the speed factor here to count for celerity - see bug 342
//int theBaseSpeed, theUnencumberedSpeed;
//this->GetSpeeds(theBaseSpeed, theUnencumberedSpeed);
//const int kMaxSpeed = this->pev->maxspeed;
//const int kMaxSpeed = (theUnencumberedSpeed > this->pev->maxspeed ) ? theUnencumberedSpeed : this->pev->maxspeed;
//// If player moving too fast or trigger uncloak is set
//if( ( (this->pev->velocity.Length() > kMaxSpeed*kWalkSpeedFactor) && !GetHasUpgrade(this->pev->iuser4, MASK_SENSORY_NEARBY)) || this->mTriggerUncloak)
if(this->mTriggerUncloak)
{
//Uncloak
this->Uncloak();
}
// :joev
else
{
// If we have cloaking upgrade
int theCloakingLevel = AvHGetAlienUpgradeLevel(this->pev->iuser4, MASK_UPGRADE_7);
if(theCloakingLevel > 0)
{
// If time needed to cloak has passed
// puzl: 864
float theMaxWalkSpeed=this->pev->maxspeed * ((this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER3) ? .95f : .7f);
float theSpeed=AvHGetAlienUpgradeLevel(this->pev->iuser4, MASK_SENSORY_NEARBY) ? 0.0f : this->pev->velocity.Length();
this->SetSpeeds(theSpeed, this->pev->maxspeed*1.05, theMaxWalkSpeed);
// if ( this->pev->velocity.Length() < theMaxWalkSpeed )
// {
// ALERT(at_console, "Cloaking\n");
this->Cloak();
// }
}
}
this->mTriggerUncloak = false;
}
bool AvHPlayer::Redeem()
{
bool theSuccess = false;
// Can only be pulled back if we have at least one active hive
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam && (theTeam->GetNumActiveHives() > 0))
{
vector<int> theSafeHives;
vector<int> theAttackedHives;
// Bring player back
if(this->GetTeamPointer()->GetNumActiveHives() > 0)
{
// Loop through the hives for this team, look for the farthest one (hives under attack take precedence)
FOR_ALL_ENTITIES(kesTeamHive, AvHHive*)
if(theEntity->pev->team == this->pev->team)
{
bool theHiveIsUnderAttack = GetGameRules()->GetIsEntityUnderAttack(theEntity->entindex());
// allow teleport to any built hive, or unbuilt hives under attack.
if(!theEntity->GetIsSpawning())
{
if ( theHiveIsUnderAttack )
theAttackedHives.push_back(theEntity->entindex());
else
theSafeHives.push_back(theEntity->entindex());
}
}
END_FOR_ALL_ENTITIES(kesTeamHive)
}
vector<int> *tmpPtr=&theSafeHives;
if ( theSafeHives.size() == 0 )
tmpPtr=&theAttackedHives;
int teleportHiveIndex=-1;
if ( tmpPtr->size() > 0 ) {
int index=RANDOM_LONG(0, tmpPtr->size()-1);
teleportHiveIndex=(*tmpPtr)[index];
}
// If we have a valid hive index, jump the player to it
if(teleportHiveIndex != -1)
{
// Play redeem effect where it happened so attackers know it happened
PLAYBACK_EVENT_FULL(0, this->edict(), gStartCloakEventID, 0, this->pev->origin, (float *)&g_vecZero, this->GetAlienAdjustedEventVolume(), 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
// Move him to it!
AvHHive* theHive = NULL;
AvHSUGetEntityFromIndex(teleportHiveIndex, theHive);
if(theHive)
{
CBaseEntity* theSpawnEntity = GetGameRules()->GetRandomHiveSpawnPoint(this, theHive->pev->origin, theHive->GetMaxSpawnDistance());
if(theSpawnEntity)
{
Vector theMinSize;
Vector theMaxSize;
this->GetSize(theMinSize, theMaxSize);
int theOffset = AvHMUGetOriginOffsetForUser3(AvHUser3(this->pev->iuser3));
Vector theOriginToSpawn = theSpawnEntity->pev->origin;
theOriginToSpawn.z += theOffset;
if(AvHSUGetIsEnoughRoomForHull(theOriginToSpawn, AvHMUGetHull(false, this->pev->iuser3), this->edict()))
{
this->SetPosition(theOriginToSpawn);
this->pev->velocity = Vector(0, 0, 0);
if(this->GetIsDigesting())
{
this->StopDigestion(false);
}
mTimeOfLastRedeem = gpGlobals->time;
theSuccess = true;
// Play teleport sound before and after
PLAYBACK_EVENT_FULL(0, this->edict(), gStartCloakEventID, 0, this->pev->origin, (float *)&g_vecZero, this->GetAlienAdjustedEventVolume(), 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 ); }
}
}
}
}
return theSuccess;
}
void AvHPlayer::InternalAlienUpgradesRegenerationThink()
{
// puzl - 0000856 - Add innate regeneration for all alien players.
// Add a small and silent innate health and armor regeneration for
// all alien players, similar to the innate regeneration of all alien
// chambers. If a player chooses the regeneration upgrade, it replaces
// the innate regeneration rather than adding to it.
// If enough time has elapsed to heal
if((this->mTimeOfLastRegeneration == -1) || (gpGlobals->time - this->mTimeOfLastRegeneration > BALANCE_VAR(kAlienRegenerationTime)))
{
int theRegenLevel = AvHGetAlienUpgradeLevel(this->pev->iuser4, MASK_UPGRADE_2);
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
float theRegenAmount = 0.0f;
float theRegenPercentage = BALANCE_VAR(kAlienInnateRegenerationPercentage);
// If we have regeneration upgrade, multiply the amount by the regen level
if(theRegenLevel > 0)
{
theRegenPercentage = BALANCE_VAR(kAlienRegenerationPercentage);
theRegenAmount = (theRegenPercentage*theMaxHealth)*theRegenLevel;
}
// Innate regeneration is at a fixed rate
else {
theRegenAmount = theRegenPercentage*(float)theMaxHealth;
}
// Always do at least 1 health of innate regeneration
theRegenAmount=max(theRegenAmount, 1.0f);
// Don't play a sound for innate regeneration
this->Regenerate(theRegenAmount, theRegenLevel > 0 );
}
// Do we have the redemption?
int theRedemptionLevel = AvHGetAlienUpgradeLevel(this->pev->iuser4, MASK_UPGRADE_3);
if(theRedemptionLevel > 0)
{
// Is the player really hurting?
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
if((this->pev->health < theMaxHealth*BALANCE_VAR(kRedemptionThreshold)) && this->IsAlive())
{
const float kPullBackTime = 20.0f;
if((this->mLastTimeRedemptionTriggered == -1) || (gpGlobals->time > (this->mLastTimeRedemptionTriggered + kPullBackTime)))
{
// Chance per second
float theRedemptionChance = theRedemptionLevel*BALANCE_VAR(kRedemptionChance);
// How many times is this being called per second?
float theThinkInterval = 1.0f;
if(this->mLastTimeCheckedRedemption > 0)
{
// The longer the time between checks, the higher the chance of being redeemed
theThinkInterval = (gpGlobals->time - this->mLastTimeCheckedRedemption);
}
// Chance per second times seconds elapsed
if(RANDOM_FLOAT(0, 1.0f) <= theRedemptionChance*theThinkInterval)
{
if(this->Redeem())
{
this->mLastTimeRedemptionTriggered = gpGlobals->time;
}
}
}
}
}
this->mLastTimeCheckedRedemption = gpGlobals->time;
}
void AvHPlayer::ProcessEntityBlip(CBaseEntity* inEntity)
{
// puzl: 982
// Make alien hivesight range a balance var
const float kAlienFriendlyBlipRange = BALANCE_VAR(kHiveSightRange);
// Is player alien?
bool theIsAlien = this->GetIsAlien(true);
// Is player marine?
bool theIsMarine = this->GetIsMarine(true);
ASSERT(theIsAlien || theIsMarine);
// Friendly?
bool theIsFriendly = ((inEntity->pev->team == 0) || (inEntity->pev->team == this->GetTeam())) ;
CBaseEntity* theSpectatingEntity = this->GetSpectatingEntity();
if(theSpectatingEntity)
{
theIsFriendly = ((theSpectatingEntity->pev->team == 0) || (theSpectatingEntity->pev->team == inEntity->pev->team));
}
// If this player in an alien
if(theIsAlien && this->IsAlive(true) && !GetHasUpgrade(inEntity->pev->iuser4, MASK_TOPDOWN))
{
// elven added - don't let digesting players get rendered on parasite
bool theEntityIsDigesting = GetHasUpgrade(inEntity->pev->iuser4, MASK_DIGESTING);
bool theEntityIsParasited = (theEntityIsDigesting) ? false : GetHasUpgrade(inEntity->pev->iuser4, MASK_PARASITED);
bool theEntityIsNearSensory = (theEntityIsDigesting) ? false : GetHasUpgrade(inEntity->pev->iuser4, MASK_SENSORY_NEARBY);
bool theEntityIsInHiveSight = (theEntityIsNearSensory || theEntityIsParasited || (GetHasUpgrade(inEntity->pev->iuser4, MASK_VIS_SIGHTED) && inEntity->IsPlayer()) || theIsFriendly || (inEntity->pev->iuser3 == AVH_USER3_HIVE));
//bool theHasHiveSightUpgrade = true;//GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_UPGRADE_11) || GetGameRules()->GetIsTesting();
bool theEntityIsInSight = this->GetIsEntityInSight(inEntity);
// If we're processing a relevant player
AvHPlayer* theOtherPlayer = dynamic_cast<AvHPlayer*>(inEntity);
bool theIsSpectatingEntity = this->GetIsSpectatingPlayer(inEntity->entindex());
if(theOtherPlayer && (theOtherPlayer != this) && !theIsSpectatingEntity && theOtherPlayer->GetIsRelevant())
{
// Calculate angle and distance to player
Vector theVectorToEntity = inEntity->pev->origin - this->pev->origin;
float theDistanceToEntity = theVectorToEntity.Length();
// If friendly
if(theEntityIsInHiveSight && theIsFriendly && !theEntityIsInSight)
{
int8 theBlipStatus = kFriendlyBlipStatus;
if(GetGameRules()->GetIsEntityUnderAttack(inEntity->entindex()))
{
theBlipStatus = kVulnerableFriendlyBlipStatus;
}
if(theOtherPlayer->GetUser3() == AVH_USER3_ALIEN_PLAYER2)
{
theBlipStatus = kGorgeBlipStatus;
}
//if(theOtherPlayer->GetEnemySighted())
//{
// theBlipStatus = 1;
//}
if(theBlipStatus || (theDistanceToEntity < kAlienFriendlyBlipRange))
{
// Info is the player index by default
int theBlipInfo = theOtherPlayer->entindex();
this->mFriendlyBlips.AddBlip(inEntity->pev->origin.x, inEntity->pev->origin.y, inEntity->pev->origin.z, theBlipStatus, theBlipInfo);
}
}
// else if enemy
else
{
// If it's in range
//if(theDistanceToEntity < kAlienEnemyBlipRange)
//{
int8 theBlipStatus = kEnemyBlipStatus;
bool theDrawBlip = false;
if(!theIsFriendly && !theEntityIsInSight)
{
// If they're parasited
if(theEntityIsParasited)
{
theDrawBlip = true;
}
// If we have scent of fear upgrade - don't render if being eaten. changed by elven.
if( (theEntityIsNearSensory || GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_9)) && !theEntityIsParasited && !GetHasUpgrade(inEntity->pev->iuser4, MASK_DIGESTING))
{
int theRange = BALANCE_VAR(kScentOfFearRadiusPerLevel);
if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_14))
{
theRange *= 2;
}
else if(GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_15))
{
theRange *= 3;
}
// ...and blip is within range
if( (theRange > theDistanceToEntity) || theEntityIsNearSensory )
{
// int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(theOtherPlayer->pev->iuser4, theOtherPlayer->GetUser3());
//
// // ...and blip is under attack or weak
// if(GetGameRules()->GetIsEntityUnderAttack(theOtherPlayer->entindex()) || (theOtherPlayer->pev->health < (theMaxHealth/3)))
// {
theBlipStatus = kVulnerableEnemyBlipStatus;
theDrawBlip = true;
// }
}
}
}
// Add it if it's in hive sight, or if we have scent of fear and he's marked
if(theDrawBlip)
{
this->mEnemyBlips.AddBlip(inEntity->pev->origin.x, inEntity->pev->origin.y, inEntity->pev->origin.z, theBlipStatus);
}
//}
}
}
// else if we're processing a generic buildable
else
{
AvHBaseBuildable* theBaseBuildable = dynamic_cast<AvHBaseBuildable*>(inEntity);
if(theBaseBuildable)
{
// If friendly hive
if(theIsFriendly)
{
// If not visible
if(!theEntityIsInSight)
{
// If it's being hurt
bool theDrawBlip=false;
AvHTeam* theTeam = this->GetTeamPointer();
AvHAlertType theAlertType = ALERT_NONE;
int8 theBlipStatus = kFriendlyBlipStatus;
bool theIsUnderAttack = GetGameRules()->GetIsEntityUnderAttack(theBaseBuildable->entindex());
if(theIsUnderAttack)
{
theBlipStatus = kVulnerableFriendlyBlipStatus;
theDrawBlip=true;
}
// Add it if relevant
AvHHive* theHive = dynamic_cast<AvHHive*>(inEntity);
if(theHive )
{
theDrawBlip=true;
if(!theIsUnderAttack)
{
theBlipStatus = kHiveBlipStatus;
}
if ( theHive && !theHive->GetIsBuilt() )
{
theBlipStatus=kUnbuiltHiveBlipsStatus;
}
}
if ( theDrawBlip == true )
{
// Info is the player index or the kBaseBlipInfoOffset + the iuser3
int theBlipInfo = kBaseBlipInfoOffset + inEntity->pev->iuser3;
if(theBlipInfo >= sizeof(char)*128)
{
ASSERT(false);
}
this->mFriendlyBlips.AddBlip(inEntity->pev->origin.x, inEntity->pev->origin.y, inEntity->pev->origin.z, theBlipStatus, theBlipInfo);
}
}
}
// If enemy
else
{
// If it's in hive sight, but not visible
if(!theEntityIsInSight && theEntityIsInHiveSight)
{
// If within range
//if(theDistanceToEntity < kAlienEnemyBlipRange)
//{
// Add it
this->mEnemyBlips.AddBlip(inEntity->pev->origin.x, inEntity->pev->origin.y, inEntity->pev->origin.z, kEnemyStructureBlipsStatus);
//}
}
}
}
}
}
// else if this player is a marine
else if(theIsMarine && this->IsAlive(true))
{
bool theTeamHasMotionTracking = GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_8);
// If player is commander (or is any marine with motion tracking?)
if((this->GetUser3() == AVH_USER3_COMMANDER_PLAYER) || (theTeamHasMotionTracking))
{
// If enemy, add it to enemy list if "detected" // Elven - we don't want motion blips on aliens visible to us.
// && !(inEntity->pev->iuser4 & MASK_VIS_SIGHTED) <- this won't work as if I see an alien, other marines won't see MT if they're in another room
bool visibleToThis = this->GetIsEntityInSight(inEntity);
if(!theIsFriendly && (inEntity->pev->iuser4 & MASK_VIS_DETECTED) && !visibleToThis && inEntity->IsAlive())//voogru: make sure there alive
{
this->mEnemyBlips.AddBlip(inEntity->pev->origin.x, inEntity->pev->origin.y, inEntity->pev->origin.z, kMarineEnemyBlipStatus);
}
}
}
}
void AvHPlayer::ClearBlips()
{
this->mEnemyBlips.Clear();
this->mFriendlyBlips.Clear();
}
void AvHPlayer::ClientDisconnected()
{
this->InitBalanceVariables();
this->ResetGameNewMap();
this->ResetEntity();
}
void AvHPlayer::ResetGameNewMap()
{
this->mNumParticleTemplatesSent = 0;
this->mTimeOfLastParticleTemplateSending = -1;
this->mClientGamma = kDefaultMapGamma;
this->mNewMap = true;
}
void AvHPlayer::InternalCommanderThink()
{
// Finally, if we are commander, set the special PAS origin to use
if(this->GetUser3() == AVH_USER3_COMMANDER_PLAYER)
{
// Only update every so often
const float kUpdatePASInterval = 1.0f;
float theTime = gpGlobals->time;
if((this->mTimeOfLastPASUpdate == -1) || (theTime > (this->mTimeOfLastPASUpdate + kUpdatePASInterval)))
{
// Default to our last known "real-world" origin, in case there are no other players?
VectorCopy(this->mPositionBeforeTopDown, this->mSpecialPASOrigin);
// Max map size is 8012x8012
double theShortestDistance = 64192144;
// Loop through all players
FOR_ALL_ENTITIES(kAvHPlayerClassName, AvHPlayer*)
// Find nearest distance to friendly and relevant player
if(theEntity->GetIsRelevant() && (theEntity->pev->team == this->pev->team) && (theEntity != this))
{
double theDistance = VectorDistance(theEntity->pev->origin, this->pev->origin);
if(theDistance < theShortestDistance)
{
VectorCopy(theEntity->pev->origin, this->mSpecialPASOrigin);
theShortestDistance = theDistance;
//SET_VIEW(ENT(this->pev), ENT(theEntity->pev));
}
}
END_FOR_ALL_ENTITIES(kAvHPlayerClassName)
this->mTimeOfLastPASUpdate = theTime;
}
}
}
void AvHPlayer::InternalBoundResources()
{
// Aliens have a max resource amount, put any that overflows back into the team
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam && this->GetIsAlien())
{
float theMaxResources = theTeam->GetMaxResources((AvHUser3)this->pev->iuser3);
float theExtraResources = this->mResources - theMaxResources;
if(theExtraResources > 0)
{
theTeam->SetTeamResources(theTeam->GetTeamResources() + theExtraResources);
}
this->mResources = min(this->mResources, theMaxResources);
}
if(GetGameRules()->GetIsCombatMode())
{
this->mResources = 0;
}
}
bool AvHPlayer::QueryEnemySighted(CBaseEntity* inEntity)
{
bool theSuccess = false;
if((this->pev->team != inEntity->pev->team) && (this->pev->team != TEAM_IND) && (inEntity->pev->team != TEAM_IND))
{
if(inEntity != this)
{
if(inEntity->IsAlive() && this->GetIsEntityInSight(inEntity))
{
AvHPlayer* thePlayer = dynamic_cast<AvHPlayer*>(inEntity);
//this->CompareToPlayer(inEntity); //voogru: WTF?
if(!thePlayer || (thePlayer->GetIsRelevant() && !thePlayer->GetIsCloaked()))
{
this->mEnemySighted = true;
this->mTimeOfLastEnemySighting = gpGlobals->time;
theSuccess = true;
}
}
}
}
return theSuccess;
}
void AvHPlayer::InternalAlienThink()
{
if(this->GetIsAlien() && this->IsAlive())
{
// Are we gestating? Has enough time passed so we are something new?
if(this->pev->iuser3 == AVH_USER3_ALIEN_EMBRYO)
{
float theCurrentTime = gpGlobals->time;
float theFullTimeToGestate = GetGameRules()->GetBuildTimeForMessageID(this->mEvolution);
if(GetGameRules()->GetIsTesting())
{
theFullTimeToGestate = 1.0f;
}
this->TriggerProgressBar(this->entindex(), 3);
// If changing this, make sure to change spectator behavior in InternalCommonThink
float theTimeElapsed = theCurrentTime - this->mTimeGestationStarted;
this->pev->fuser3 = (theTimeElapsed/theFullTimeToGestate)*kNormalizationNetworkFactor;
if(theTimeElapsed >= theFullTimeToGestate)
{
this->ProcessEvolution();
// Set ourselves crouching so we have a smaller chance of getting stuck
if(AvHMUGetCanDuck(this->pev->iuser3))
{
SetBits(this->pev->flags, FL_DUCKING);
}
}
}
// Has enough time passed since we started screaming?
if(this->mIsScreaming)
{
if(gpGlobals->time > (this->mTimeStartedScream + BALANCE_VAR(kPrimalScreamDuration)))
{
this->mIsScreaming = false;
}
}
// Uncloak if we are charging, leaping or blinking
if(GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_MOVEMENT))
{
this->TriggerUncloak();
}
}
}
void AvHPlayer::InternalCommonThink()
{
if(GetGameRules()->GetGameStarted())
{
this->mPreThinkTicks++;
float theTimePassed = gpGlobals->time - GetGameRules()->GetTimeGameStarted();
this->mPreThinkFrameRate = this->mPreThinkTicks/theTimePassed;
// if(RANDOM_LONG(0, 125) == 0)
// {
// char theMessage[128];
// sprintf(theMessage, "Num ents: %d\n", GetGameRules()->GetNumEntities());
// //sprintf(theMessage, "Time passed: %f, ticks: %d, rate: %f, %d\n", theTimePassed, this->mPreThinkTicks, this->mPreThinkFrameRate, gNumFullPackCalls);
// UTIL_SayText(theMessage, this);
// }
}
else
{
this->mPreThinkTicks = 0;
this->mPreThinkFrameRate = 30;
}
// If we're not in the ready room, and the a victor has just been determined, see if it's time to reset us
// This has to be done at a different time per player, to avoid overflows
if((GetGameRules()->GetVictoryTeam() != TEAM_IND) && (this->GetPlayMode() != PLAYMODE_READYROOM))
{
// Get victory time, see if it's time to reset us
int thePlayerIndex = this->entindex();
int theSecondToReset = kVictoryIntermission - 1 - kMaxPlayers/kResetPlayersPerSecond + (thePlayerIndex - 1)/kResetPlayersPerSecond;
//ASSERT(theSecondToReset >= 0);
//ASSERT(theSecondToReset < kVictoryIntermission);
// NOTE: This depends on gamerules not allong players join a team after victory has been declared
float theVictoryTime = GetGameRules()->GetVictoryTime();
if(gpGlobals->time > (theVictoryTime + theSecondToReset))
{
this->SetPlayMode(PLAYMODE_READYROOM, true);
}
}
// Must be called every frame to prevent exploiting
this->SetModelFromState();
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
// Clear existing upgrades for marines. Aliens have their own individual upgrades.
if(this->GetIsMarine() && !GetGameRules()->GetIsCombatMode())
{
int theInvertedUpgradeMask = ~kUpgradeBitMask;
this->pev->iuser4 &= theInvertedUpgradeMask;
}
// Set the current upgrades
this->pev->iuser4 |= theTeam->GetTeamWideUpgrades();
}
// Update active and inactive inventory
const float kUpdateInventoryInterval = .5f;
if(gpGlobals->time > (this->mLastInventoryThink + kUpdateInventoryInterval))
{
this->UpdateInventoryEnabledState(this->mNumHives, true);
this->mLastInventoryThink = gpGlobals->time;
}
// Remember last time we were playing
if(this->GetPlayMode() == PLAYMODE_PLAYING)
{
this->mTimeLastPlaying = gpGlobals->time;
}
this->InternalBoundResources();
// Players keep their health in fuser2
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
int theCurrentHealth = max(0.0f, this->pev->health);
int theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
int theCurrentArmor = max(0.0f, this->pev->armorvalue);
// Draw ring to take into account health and armor for aliens,
// Send armor and health for marines
if(this->GetIsMarine())
{
// puzl: 991 use a composite for marine armour and health
int theCurrentArmorPercent=(theCurrentArmor*100)/theMaxArmor;
int theCurrentHealthPercent=(theCurrentHealth*100)/theMaxHealth;
this->pev->fuser2= (float)( ((theCurrentArmorPercent&0x7F) << 7 ) + (theCurrentHealthPercent & 0x7F));
}
else
{
float theScalar = (float)(theCurrentHealth + theCurrentArmor)/(float)(theMaxHealth + theMaxArmor);
this->pev->fuser2 = theScalar*kNormalizationNetworkFactor;
}
//float theRandomAngle = RANDOM_FLOAT(0, 50);
//this->pev->v_angle.x = theRandomAngle;
//VectorCopy(this->pev->angles, this->mAnglesBeforeTopDown);
//VectorCopy(this->pev->v_angle, this->mViewAnglesBeforeTopDown);
if(GetGameRules()->GetCountdownStarted() && !GetGameRules()->GetGameStarted() && (GetPlayMode() == PLAYMODE_PLAYING) && !GetGameRules()->GetCheatsEnabled())
{
this->pev->flags |= FL_FROZEN;
DROP_TO_FLOOR(this->edict());
}
else
{
this->pev->flags &= ~FL_FROZEN;
}
// If we have a different desired name, change to it
if(GetGameRules()->GetArePlayersAllowedToJoinImmediately())
{
// If our desired net name hasn't been set, set it now
if(this->mDesiredNetName != "")
{
char* theInfoBuffer = g_engfuncs.pfnGetInfoKeyBuffer(this->edict());
char theBuffer[256];
strcpy(theBuffer, this->mDesiredNetName.c_str());
g_engfuncs.pfnSetClientKeyValue(this->entindex(), theInfoBuffer, "name", theBuffer);
this->mDesiredNetName = "";
}
}
// If we're spectating a target, set our health and armorvalue to theirs
// For spectators that are tracking players, move to their location, to prevent problems outside of the PVS
AvHPlayer* theEntity = NULL;
if(AvHSUGetEntityFromIndex(this->pev->iuser2, theEntity))
{
// If entity is no longer spectatable, jump to the next target
if(theEntity->GetIsInTopDownMode() || (theEntity->GetPlayMode() != PLAYMODE_PLAYING))
{
this->Observer_FindNextPlayer();
}
else
{
this->pev->health = theEntity->pev->health;
this->pev->armorvalue = theEntity->pev->armorvalue;
this->pev->fuser3 = theEntity->pev->fuser3;
if((theEntity->GetUser3() == AVH_USER3_ALIEN_EMBRYO) || (theEntity->GetIsBeingDigested()))
{
this->TriggerProgressBar(theEntity->entindex(), 3);
}
// Hacky way to make sure player is in PVS of target
float theDistance = VectorDistance(theEntity->pev->origin, this->pev->origin);
if(theDistance > 700)
{
VectorCopy(theEntity->pev->origin, this->pev->origin);
}
}
}
// If we are digesting a player, process him (process even for marines in case we're testing)
this->InternalDigestionThink();
// Update FOV and view height every tick, needed for first-person spectating
this->SetViewForUser3();
PropagateServerVariables();
}
void AvHPlayer::PropagateServerVariables()
{
if ( gpGlobals->time > (mLastUpdateTime + 0.5f) )
{
for (int i = 0; i < (signed)mServerVariableList.size(); ++i)
{
std::string theValue = CVAR_GET_STRING( mServerVariableList[i].mName.c_str() );
if ( mServerVariableList[i].mLastValueSent != theValue)
{
NetMsg_ServerVar( this->pev, mServerVariableList[i].mName, theValue );
mServerVariableList[i].mLastValueSent = theValue;
break; // Only send one message per tick to avoid overflow.
}
}
mLastUpdateTime = gpGlobals->time;
}
}
void AvHPlayer::InternalMarineThink()
{
if(this->GetIsMarine() && (this->pev->iuser3 != AVH_USER3_COMMANDER_PLAYER))
{
// Slowly heal power armor over time
if(this->GetHasPowerArmor())
{
if(this->mLastPowerArmorThink != -1)
{
float theTimePassed = gpGlobals->time - this->mLastPowerArmorThink;
if(theTimePassed > 0.0f)
{
// Key regen to velocity to enhance leap-frogging, ala Halo
float theVelocity = this->pev->velocity.Length();
float theRegenFactor = .5f;
if(theVelocity > 0)
{
//theRegenFactor = .3f - .3f*(theVelocity/kMaxGroundPlayerSpeed);
theRegenFactor = 0.1f;
}
theRegenFactor = max(min(theRegenFactor, 1.0f), 0.0f);
const float kPowerRegenRate = theRegenFactor*2.0f;
int theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
if(this->pev->armorvalue < theMaxArmor)
{
this->pev->armorvalue = min((float)theMaxArmor, this->pev->armorvalue + kPowerRegenRate*theTimePassed);
}
}
}
this->mLastPowerArmorThink = gpGlobals->time;
}
// Update buffed
if(this->GetIsCatalysted())
{
if(gpGlobals->time > this->mTimeToEndCatalyst)
{
this->SetIsCatalysted(false);
}
}
}
}
void AvHPlayer::InternalPreThink()
{
PROFILE_START()
this->InternalCommonThink();
PROFILE_END(kPlayerCommonThink);
PROFILE_START()
this->InternalAlienThink();
PROFILE_END(kPlayerAlienThink)
PROFILE_START()
this->InternalMarineThink();
PROFILE_END(kPlayerMarineThink)
PROFILE_START()
this->InternalCommanderThink();
PROFILE_END(kPlayerCommanderThink)
PROFILE_START()
this->InternalAlienUpgradesThink();
PROFILE_END(kPlayerAlienUpgradesThink)
PROFILE_START()
this->InternalCombatThink();
PROFILE_END(kPlayerCombatThink)
//this->InternalEnemySightedPreThink();
PROFILE_START()
this->InternalSpeakingThink();
PROFILE_END(kPlayerSpeakingThink)
PROFILE_START()
this->InternalProgressBarThink();
PROFILE_END(kPlayerProgressBarThink)
PROFILE_START()
this->InternalFogThink();
PROFILE_END(kPlayerFogThink)
PROFILE_START()
this->InternalHUDThink();
PROFILE_END(kPlayerHUDThink)
this->InternalMovementThink();
}
// Charge pushback
void AvHPlayer::InternalMovementThink()
{
// Ensure that we do leap damage
if(GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_MOVEMENT) && (GetUser3() == AVH_USER3_ALIEN_PLAYER1))
this->StartLeap();
// Check whether we're in a charge
if(GetHasUpgrade(this->pev->iuser4, MASK_ALIEN_MOVEMENT) && (GetUser3() == AVH_USER3_ALIEN_PLAYER5))
{
CBaseEntity* theEntity = NULL;
float radius = (float)BALANCE_VAR(kChargePushbackRadius);
float maxpushbackspeedfactor = (float)BALANCE_VAR(kChargeMaxPushbackSpeedFactor);
float pushbackfactor = (float)BALANCE_VAR(kChargeMaxPushbackForce);
// Find all entities around the onos
while((theEntity = UTIL_FindEntityInSphere(theEntity, this->pev->origin, radius)) != NULL)
{
if (theEntity->IsPlayer() && theEntity->IsAlive())
{
float distance = VectorDistance(this->pev->origin, theEntity->pev->origin);
if (distance > 0.0f)
{
float factor = pushbackfactor / (radius / distance);
vec3_t direction, heading;
VectorSubtract(theEntity->pev->origin, this->pev->origin, direction);
VectorNormalize(direction);
VectorCopy(this->pev->velocity, heading);
VectorNormalize(heading);
float dot = DotProduct(heading, direction);
if (dot > 0.0f)
{
direction[0] = 50;
VectorScale(direction, factor * dot, direction);
VectorAdd(theEntity->pev->velocity, direction, theEntity->pev->velocity);
if (Length(theEntity->pev->velocity) > theEntity->pev->maxspeed * maxpushbackspeedfactor)
{
VectorNormalize(theEntity->pev->velocity);
VectorScale(theEntity->pev->velocity, theEntity->pev->maxspeed * maxpushbackspeedfactor, theEntity->pev->velocity);
}
}
}
}
}
}
}
void AvHPlayer::InternalFogThink()
{
if((this->mTimeOfLastFogTrigger != -1) && (gpGlobals->time > this->mTimeOfLastFogTrigger + this->mFogExpireTime))
{
this->mCurrentFogEntity = -1;
}
}
void AvHPlayer::InternalHUDThink()
{
// Pull weapon out if we're done using something
if(this->mTimeOfLastUse != -1)
{
const float kUseTime = .5f;
if(gpGlobals->time > this->mTimeOfLastUse + kUseTime)
{
// Don't deploy while you're a commander (happens if you finish building something next to command station and immediately jump in)
if(!this->GetIsInTopDownMode() && !this->GetIsBeingDigested())
{
this->DeployCurrent();
this->mTimeOfLastUse = -1;
}
}
}
// Don't hide chat by default
int theHideHUD = HIDEHUD_WEAPONS | HIDEHUD_FLASHLIGHT | HIDEHUD_HEALTH;
// Use local player or player we're spectating
AvHPlayer* theVisiblePlayer = this;
if(this->pev->iuser1 == OBS_IN_EYE)
{
AvHPlayer* theEntity = NULL;
if(AvHSUGetEntityFromIndex(this->pev->iuser2, theEntity))
{
theVisiblePlayer = theEntity;
}
}
AvHPlayMode thePlayMode = theVisiblePlayer->GetPlayMode();
AvHUser3 theUser3 = theVisiblePlayer->GetUser3();
if((thePlayMode == PLAYMODE_READYROOM) || (thePlayMode == PLAYMODE_REINFORCING) || (thePlayMode == PLAYMODE_AWAITINGREINFORCEMENT))
{
theHideHUD = HIDEHUD_FLASHLIGHT | HIDEHUD_WEAPONS | HIDEHUD_HEALTH;
// Only hide health when not following a target
if((thePlayMode == PLAYMODE_AWAITINGREINFORCEMENT) && (this->pev->iuser1 == OBS_IN_EYE))
{
//theHideHUD &= ~HIDEHUD_WEAPONS;
theHideHUD &= ~HIDEHUD_HEALTH;
}
}
else if(thePlayMode == PLAYMODE_PLAYING)
{
theHideHUD = 0;
if(!GetGameRules()->FAllowFlashlight())
{
theHideHUD |= HIDEHUD_FLASHLIGHT;
}
if(!theVisiblePlayer->pev->viewmodel)
{
//theHideHUD |= HIDEHUD_WEAPONS | HIDEHUD_FLASHLIGHT;
}
if(theUser3 == AVH_USER3_ALIEN_EMBRYO)
{
//theHideHUD |= HIDEHUD_HEALTH;
}
if(GetHasUpgrade(theVisiblePlayer->pev->iuser4, MASK_TOPDOWN) || theVisiblePlayer->GetIsBeingDigested())
{
theHideHUD |= HIDEHUD_WEAPONS | HIDEHUD_FLASHLIGHT | HIDEHUD_HEALTH;
}
else
{
// If we have no other weapons, hide ammo
if(!HasWeapons())
{
theHideHUD |= HIDEHUD_WEAPONS;
theHideHUD |= HIDEHUD_FLASHLIGHT;
}
if(theUser3 == AVH_USER3_ALIEN_EMBRYO)
{
theHideHUD |= HIDEHUD_WEAPONS;
theHideHUD |= HIDEHUD_FLASHLIGHT;
}
}
}
else if(thePlayMode == PLAYMODE_OBSERVER)
{
theHideHUD = HIDEHUD_WEAPONS | HIDEHUD_FLASHLIGHT | HIDEHUD_HEALTH;
// Only hide health when not following a target
if((this->pev->iuser1 == OBS_IN_EYE))
{
//theHideHUD &= ~HIDEHUD_WEAPONS;
theHideHUD &= ~HIDEHUD_HEALTH;
}
}
else
{
int a = 0;
}
this->m_iHideHUD = theHideHUD;
}
void AvHPlayer::InternalProgressBarThink()
{
// If some time has passed since the progress bar was triggered, send down a message to kill it
const float kProgressBarTimeOut = .2f;
if(this->mTimeProgressBarTriggered != -1)
{
if(gpGlobals->time > this->mTimeProgressBarTriggered + kProgressBarTimeOut)
{
this->mTimeProgressBarTriggered = -1;
this->mProgressBarEntityIndex = -1;
this->mProgressBarParam = -1;
}
}
}
void AvHPlayer::InternalSpeakingThink()
{
if((this->mIsSpeaking || this->mOrderAcknowledged || this->mOrdersRequested) && (gpGlobals->time - this->mTimeOfLastSaying >= kSpeakingTime))
{
this->mIsSpeaking = false;
this->mOrderAcknowledged = false;
this->mOrdersRequested = false;
}
}
void AvHPlayer::PreThink( void )
{
// Get play mode
AvHPlayMode thePlayMode = this->GetPlayMode();
bool theRunThink = ((thePlayMode == PLAYMODE_READYROOM) && GET_RUN_CODE(8)) ||
((thePlayMode == PLAYMODE_OBSERVER) && (GET_RUN_CODE(16))) ||
(this->GetIsAlien() && GET_RUN_CODE(32)) ||
(this->GetIsMarine() && GET_RUN_CODE(64));
if(theRunThink)
{
PROFILE_START()
CBasePlayer::PreThink();
PROFILE_END(kCBasePlayerPreThink)
PROFILE_START()
this->InternalPreThink();
PROFILE_END(kPlayerInternalPreThink)
PROFILE_START()
this->ValidateClientMoveEvents();
PROFILE_END(kValidateClientMoveEvents)
PROFILE_START()
this->HandleTopDownInput();
PROFILE_END(kHandleTopDownInput)
PROFILE_START()
this->RecalculateSpeed();
PROFILE_END(kRecalculateSpeed)
PROFILE_START()
if(this->mQueuedThinkMessage != "")
{
this->SendMessage(this->mQueuedThinkMessage.c_str(), TOOLTIP);
this->mQueuedThinkMessage = "";
}
if(this->mPendingCommand)
{
// Is this bad?
GetGameRules()->ClientCommand(this, this->mPendingCommand);
this->mPendingCommand = NULL;
}
PROFILE_END(kPlayerPreThinkMisc)
}
}
bool AvHPlayer::PayPurchaseCost(int inCost)
{
bool theSuccess = false;
if(GetGameRules()->GetIsCombatMode())
{
if(inCost <= (this->GetExperienceLevel() - this->GetExperienceLevelsSpent() - 1))
{
this->SetExperienceLevelsSpent(this->GetExperienceLevelsSpent() + inCost);
theSuccess = true;
}
}
else
{
if(inCost <= this->GetResources())
{
this->SetResources(this->GetResources() - inCost);
theSuccess = true;
}
}
return theSuccess;
}
void AvHPlayer::RecalculateSpeed(void)
{
// Look at inventory and set speed from weight
int theRelevantWeight = this->GetRelevantWeight();
int theMaxWeight = GetGameRules()->GetMaxWeight();
int theBaseSpeed, theUnencumberedSpeed;
this->GetSpeeds(theBaseSpeed, theUnencumberedSpeed);
this->mMaxWalkSpeed = theUnencumberedSpeed*.75f;
// Calculate the max speed
int theMaxSpeed = theUnencumberedSpeed - (theRelevantWeight/(float)theMaxWeight)*(theUnencumberedSpeed - theBaseSpeed);
theMaxSpeed = max(theMaxSpeed, theBaseSpeed);
theMaxSpeed = min(theMaxSpeed, theUnencumberedSpeed);
// Set it but only if it changed (just in case there's a hidden performance or network cost)
if(this->pev->maxspeed != theMaxSpeed)
{
//char theMessage[256];
//sprintf(theMessage, "New weight is %d, setting speed to %d.\n", theRelevantWeight, theMaxSpeed);
//ClientPrint(this->pev, HUD_PRINTTALK, theMessage);
this->pev->maxspeed = theMaxSpeed;
g_engfuncs.pfnSetClientMaxspeed( ENT(this->pev), theMaxSpeed);
}
}
void AvHPlayer::ReloadWeapon(void)
{
if(this->m_pActiveItem)
{
CBasePlayerWeapon* theGun = (CBasePlayerWeapon *)this->m_pActiveItem->GetWeaponPtr();
if(theGun)
{
//SetAnimation(PLAYER_RELOAD);
theGun->Reload();
}
}
}
// Reset stuff
void AvHPlayer::ResetEntity(void)
{
CBasePlayer::ResetEntity();
this->mHasSeenTeamA = false;
this->mHasSeenTeamB = false;
this->ResetBehavior(true);
this->UpdateTopDownMode();
// Preserve items we want to survive init
bool theSavedNewMap = this->mNewMap;
string theSavedDesiredNetName = this->mDesiredNetName;
AvHBaseInfoLocationListType theSavedClientInfoLocations = this->mClientInfoLocations;
this->Init();
this->mNewMap = theSavedNewMap;
this->mDesiredNetName = theSavedDesiredNetName;
this->mClientInfoLocations = theSavedClientInfoLocations;
this->mMarineHUDUpgrades=0;
}
void AvHPlayer::ResetOverwatch()
{
// clear target
this->mOverwatchTarget = -1;
this->pev->fuser1 = -1;
this->pev->fuser2 = -1;
// Set facing back to original facing
VectorCopy(this->mOverwatchFacing, this->pev->angles);
this->pev->fixangle = TRUE;
}
#include "engine/studio.h"
void AvHPlayer::SetModelFromState()
{
// Default to marine in ready room
char* theModelName = NULL;
AvHUser3 theUser3 = this->GetUser3();
switch(theUser3)
{
case AVH_USER3_MARINE_PLAYER:
theModelName = kMarineSoldierModel;
if(this->GetHasHeavyArmor())
{
theModelName = kHeavySoldierModel;
}
break;
case AVH_USER3_COMMANDER_PLAYER:
theModelName = kMarineCommanderModel;
break;
case AVH_USER3_ALIEN_PLAYER1:
theModelName = kAlienLevelOneModel;
break;
case AVH_USER3_ALIEN_PLAYER2:
theModelName = kAlienLevelTwoModel;
break;
case AVH_USER3_ALIEN_PLAYER3:
theModelName = kAlienLevelThreeModel;
break;
case AVH_USER3_ALIEN_PLAYER4:
theModelName = kAlienLevelFourModel;
break;
case AVH_USER3_ALIEN_PLAYER5:
theModelName = kAlienLevelFiveModel;
break;
case AVH_USER3_ALIEN_EMBRYO:
theModelName = kAlienGestateModel;
break;
default:
if(this->GetPlayMode() == PLAYMODE_READYROOM)
{
theModelName = kReadyRoomModel;
}
break;
}
//SET_MODEL(ENT(pev), theModelName);
// Alternative method of setting the model on the server:
if(theModelName)
{
pev->model = MAKE_STRING(theModelName);
this->mLastModelIndex = MODEL_INDEX(theModelName);
}
if(this->mLastModelIndex != 1)
{
pev->modelindex = mLastModelIndex;
}
// Set body group for marine armor
this->pev->body = 0;
if(this->GetHasJetpack())
{
this->pev->body = 1;
}
}
void AvHPlayer::SetMoveTypeForUser3()
{
switch(this->pev->iuser3)
{
case AVH_USER3_ALIEN_EMBRYO:
// this->pev->movetype = MOVETYPE_PUSH;
// break;
case AVH_USER3_NONE:
case AVH_USER3_MARINE_PLAYER:
case AVH_USER3_ALIEN_PLAYER1:
case AVH_USER3_ALIEN_PLAYER2:
case AVH_USER3_ALIEN_PLAYER3:
case AVH_USER3_ALIEN_PLAYER4:
case AVH_USER3_ALIEN_PLAYER5:
this->pev->movetype = MOVETYPE_WALK;
break;
}
}
void AvHPlayer::GetSize(Vector& outMinSize, Vector& outMaxSize) const
{
bool theIsDucking = FBitSet(this->pev->flags, FL_DUCKING);
AvHSHUGetSizeForPlayerUser3((AvHUser3)this->pev->iuser3, outMinSize, outMaxSize, theIsDucking);
}
void AvHPlayer::SetWeaponsForUser3()
{
AvHUser3 theUser3 = (AvHUser3)this->pev->iuser3;
bool theTeamHasGrenades = false;
if(!GetGameRules()->GetIsCombatMode())
{
AvHTeam* theTeamPointer = this->GetTeamPointer(false);
if(theTeamPointer)
{
theTeamHasGrenades = theTeamPointer->GetResearchManager().GetTechNodes().GetIsTechResearched(TECH_RESEARCH_GRENADES);
}
}
switch(theUser3)
{
case AVH_USER3_NONE:
break;
case AVH_USER3_MARINE_PLAYER:
if(this->mPreviousUser3 != AVH_USER3_COMMANDER_PLAYER)
{
this->GiveNamedItem(kwsMachineGun);
this->GiveNamedItem(kwsPistol);
this->GiveNamedItem(kwsKnife);
if(theTeamHasGrenades)
{
this->GiveNamedItem(kwsGrenade);
}
}
break;
case AVH_USER3_COMMANDER_PLAYER:
break;
// NOTE: When moving weapons/abilities around, be sure to change AvHBasePlayerWeapon::GetAnimationExtension(), and AvHSHUGetIsWeaponFocusable().
case AVH_USER3_ALIEN_PLAYER1:
this->DestroyAllItems(FALSE);
this->GiveNamedItem(kwsBiteGun);
this->GiveNamedItem(kwsParasiteGun);
this->GiveNamedItem(kwsLeap);
this->GiveNamedItem(kwsDivineWind);
this->SwitchWeapon(kwsBiteGun);
break;
case AVH_USER3_ALIEN_PLAYER2:
this->DestroyAllItems(FALSE);
this->GiveNamedItem(kwsHealingSpray);
this->GiveNamedItem(kwsSpitGun);
this->GiveNamedItem(kwsBileBombGun);
this->GiveNamedItem(kwsWebSpinner);
this->SwitchWeapon(kwsSpitGun);
break;
case AVH_USER3_ALIEN_PLAYER3:
this->DestroyAllItems(FALSE);
this->GiveNamedItem(kwsBite2Gun);
//this->GiveNamedItem(kwsSpikeGun);
this->GiveNamedItem(kwsSporeGun);
this->GiveNamedItem(kwsUmbraGun);
this->GiveNamedItem(kwsPrimalScream);
this->SwitchWeapon(kwsBite2Gun);
break;
case AVH_USER3_ALIEN_PLAYER4:
this->DestroyAllItems(FALSE);
this->GiveNamedItem(kwsSwipe);
this->GiveNamedItem(kwsBlinkGun);
this->GiveNamedItem(kwsAcidRocketGun);
this->GiveNamedItem(kwsMetabolize);
this->SwitchWeapon(kwsSwipe);
break;
case AVH_USER3_ALIEN_PLAYER5:
this->DestroyAllItems(FALSE);
this->GiveNamedItem(kwsClaws);
this->GiveNamedItem(kwsDevour);
this->GiveNamedItem(kwsStomp);
this->GiveNamedItem(kwsCharge);
this->SwitchWeapon(kwsClaws);
break;
case AVH_USER3_ALIEN_EMBRYO:
this->DestroyAllItems(FALSE);
break;
}
}
void AvHPlayer::SetSizeForUser3()
{
AvHUser3 theUser3 = (AvHUser3)this->pev->iuser3;
Vector theMinSize;
Vector theMaxSize;
// Use our previous User3 if we're back in the ready room after a game
if(this->GetPlayMode() == PLAYMODE_READYROOM)
{
if(this->mPreviousUser3 != AVH_USER3_NONE)
{
theUser3 = this->mPreviousUser3;
}
}
this->GetSize(theMinSize, theMaxSize);
UTIL_SetSize(this->pev, theMinSize, theMaxSize);
UTIL_SetOrigin(this->pev, this->pev->origin );
}
void AvHPlayer::GetViewForUser3(AvHUser3 inUser3, bool inIsDucking, float& outFOV, float& outOffset) const
{
switch(inUser3)
{
case AVH_USER3_NONE:
case AVH_USER3_MARINE_PLAYER:
case AVH_USER3_COMMANDER_PLAYER:
case AVH_USER3_ALIEN_PLAYER4:
default:
outFOV = 90;
outOffset = inIsDucking ? kDuckingViewHeightPercentage*HULL1_MAXZ: kStandingViewHeightPercentage*HULL0_MAXZ;
break;
case AVH_USER3_ALIEN_PLAYER1:
outFOV = 105;
outOffset = 0;
break;
case AVH_USER3_ALIEN_EMBRYO:
case AVH_USER3_ALIEN_PLAYER2:
outFOV = 100;
outOffset = 10;
break;
case AVH_USER3_ALIEN_PLAYER3:
outFOV = 90;
outOffset = 10;
break;
case AVH_USER3_ALIEN_PLAYER5:
outFOV = 90;
outOffset = inIsDucking ? kDuckingViewHeightPercentage*HULL0_MAXZ: kStandingViewHeightPercentage*HULL3_MAXZ;
break;
}
}
void AvHPlayer::SetViewForUser3()
{
AvHUser3 theEndUser3 = this->GetUser3(true);
bool theIsDucking = FBitSet(this->pev->flags, FL_DUCKING);
if (theEndUser3 == AVH_USER3_ALIEN_EMBRYO)
{
bool theEndIsDucking = true;
switch(GetEvolution(true))
{
case ALIEN_LIFEFORM_ONE:
theEndUser3 = AVH_USER3_ALIEN_PLAYER1;
break;
case ALIEN_LIFEFORM_TWO:
theEndUser3 = AVH_USER3_ALIEN_PLAYER2;
break;
case ALIEN_LIFEFORM_THREE:
theEndUser3 = AVH_USER3_ALIEN_PLAYER3;
break;
case ALIEN_LIFEFORM_FOUR:
theEndUser3 = AVH_USER3_ALIEN_PLAYER4;
break;
case ALIEN_LIFEFORM_FIVE:
theEndUser3 = AVH_USER3_ALIEN_PLAYER5;
break;
default:
// For upgrades.
theEndUser3 = GetPreviousUser3(true);
break;
}
// Linearly interpolate between the previous lifeform and the new lifeform.
float theStartFOV;
float theStartOffset;
float theEndFOV;
float theEndOffset;
float amount = pev->fuser3 / kNormalizationNetworkFactor;
AvHUser3 theStartUser3 = GetPreviousUser3(true);
GetViewForUser3(theStartUser3, theIsDucking, theStartFOV, theStartOffset);
GetViewForUser3(theEndUser3, true, theEndFOV, theEndOffset);
// Take into account that the origin will change for the offset.
Vector theStartMinSize;
Vector theStartMaxSize;
AvHSHUGetSizeForPlayerUser3(this->GetUser3(true), theStartMinSize, theStartMaxSize, theIsDucking);
Vector theEndMinSize;
Vector theEndMaxSize;
AvHSHUGetSizeForPlayerUser3(theEndUser3, theEndMinSize, theEndMaxSize, true);
theEndOffset += theStartMinSize.z - theEndMinSize.z;
pev->fov = theStartFOV + amount * (theEndFOV - theStartFOV);
pev->view_ofs[2] = theStartOffset + amount * (theEndOffset - theStartOffset);
}
else
{
GetViewForUser3(theEndUser3, theIsDucking, pev->fov, pev->view_ofs[2]);
}
}
bool AvHPlayer::SendMessage(const char *pMessage, SHOWMESSAGE_TYPE type)
{
bool theSuccess = false;
int theNumChars = strlen(pMessage);
if((theNumChars > 0) && (theNumChars < kMaxPlayerSendMessageLength))
{
string theMessage(pMessage);
if(theMessage != this->mLastMessageSent)
{
UTIL_ShowMessage2(pMessage, this, type);
this->mLastMessageSent = theMessage;
theSuccess = true;
}
else
{
int a = 0;
}
}
// else
// {
// // Log error to console
// char theErrorMessage[10000];
// sprintf(theErrorMessage, "Can't send message \"%s\" of length %d, max size is %d", pMessage, theNumChars, kMaxPlayerSendMessageLength);
// ALERT(at_logged, theErrorMessage);
// }
return theSuccess;
}
bool AvHPlayer::SendMessageOnce(const char *pMessage, SHOWMESSAGE_TYPE type)
{
bool theSentMessage = false;
string theMessage(pMessage);
// Check if message is in sent list
StringList::iterator theIter = std::find(this->mSentMessageList.begin(), this->mSentMessageList.end(), theMessage);
if(theIter == this->mSentMessageList.end())
{
// If not
// Call SendMessage
theSentMessage = this->SendMessage(pMessage, type);
this->mLastMessageSent = theMessage;
// Add message to list
this->mSentMessageList.push_back(theMessage);
}
return theSentMessage;
}
bool AvHPlayer::SendMessageNextThink(const char* pMessage)
{
this->mQueuedThinkMessage = string(pMessage);
return true;
}
void AvHPlayer::SetCurrentCommand(const struct usercmd_s* inCommand)
{
memcpy(&this->mCurrentCommand, inCommand, sizeof(*inCommand));
}
void AvHPlayer::SetDebugCSP(weapon_data_t* inWeaponData)
{
CBasePlayerWeapon* theCurrentWeapon = dynamic_cast<CBasePlayerWeapon*>(this->m_pActiveItem);
if(theCurrentWeapon && (theCurrentWeapon->m_iId == inWeaponData->m_iId))
{
memcpy(&this->mDebugCSPInfo, inWeaponData, sizeof(weapon_data_t));
}
}
void AvHPlayer::StartObserver( Vector vecPosition, Vector vecViewAngle )
{
Vector theFadeColor;
theFadeColor.x = 0;
theFadeColor.y = 0;
theFadeColor.z = 0;
UTIL_ScreenFade(this, theFadeColor, kTransitionFadeTime, 0.0f, 255, FFADE_IN);
CBasePlayer::StartObserver(vecPosition, vecViewAngle);
}
void AvHPlayer::ResetBehavior(bool inRemoveFromTeam)
{
// remove observer mode if enabled
this->StopObserver();
// Leave top down mode if in it
this->StopTopDownMode();
// Stop digesting if you are
this->StopDigestion(false);
// Stop being digested
this->SetBeingDigestedMode(false);
// Reset room sounds
this->SetDesiredRoomType(0, true);
// remove all equipment, but don't drop it (how to do this?)
this->DestroyAllItems(FALSE);
if(inRemoveFromTeam)
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
theTeam->RemovePlayer(this->entindex());
}
// Clear experience
this->mExperience = 0.0f;
this->mExperienceLevelsSpent = 0;
this->mCombatNodes.Clear();
this->mPurchasedCombatUpgrades.clear();
this->mGiveCombatUpgrades.clear();
}
}
void AvHPlayer::SetPlayMode(AvHPlayMode inPlayMode, bool inForceSpawn)
{
if(this->pev->playerclass != inPlayMode || inForceSpawn)
{
bool theGoingToReadyRoom = (inPlayMode == PLAYMODE_READYROOM);
this->ResetBehavior(theGoingToReadyRoom);
if(!theGoingToReadyRoom)
{
// Clear player
//this->Init();
this->ClearUserVariables();
this->pev->rendermode = kRenderNormal;
this->pev->renderfx = kRenderFxNone;
this->pev->renderamt = 0;
}
// Clear anim
this->m_szAnimExtention[0] = '\0';
// this->mUpgrades.clear();
AvHTeamNumber theTeamNumber = AvHTeamNumber(this->pev->team);
AvHUser3 theUser3 = AVH_USER3_NONE;
bool theSetUser3 = false;
string theMessage;
AvHTeam* theTeam = this->GetTeamPointer(false);
string theTeamName = kUndefinedTeam;
switch(inPlayMode)
{
// Initialize stuff
case PLAYMODE_UNDEFINED:
this->pev->iuser3 = AVH_USER3_NONE;
this->pev->playerclass = PLAYMODE_UNDEFINED;
this->pev->team = TEAM_IND;
respawn(this->pev, FALSE);
break;
case PLAYMODE_READYROOM:
this->pev->playerclass = PLAYMODE_READYROOM;
if(theTeam)
{
theTeam->RemovePlayer(this->entindex());
}
this->pev->frags = 0;
this->mScore = 0;
this->mSavedCombatFrags = 0;
this->m_iDeaths = 0;
this->pev->team = TEAM_IND;
// So we don't have the sideways camera because we're dead
this->pev->health = 100;
this->pev->max_health = pev->health;
this->pev->armorvalue = 0;
respawn(this->pev, FALSE);
if(this->pev->iuser3 == AVH_USER3_ALIEN_EMBRYO)
{
// Stop sound and allow movement, else we're stuck in the ready room
UTIL_EmitAmbientSound(ENT(this->pev), this->pev->origin, kGestationSound, 1.0f, .5, SND_STOP, 100);
SetUpgradeMask(&this->pev->iuser4, MASK_ALIEN_EMBRYO, false);
}
// Else "commander" draws on the scoreboard and it doesn't make sense to have a commander on the ground
else if(this->pev->iuser3 == AVH_USER3_COMMANDER_PLAYER)
{
this->pev->iuser3 = AVH_USER3_MARINE_PLAYER;
}
//else if(this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER5)
//{
// // Set player ducking to improve chances of them not getting stuck
// SetBits( m_afPhysicsFlags, PFLAG_DUCKING );
// SetBits( this->pev->flags, FL_DUCKING );
// this->pev->origin.z += 25;
//}
theMessage = kReadyRoomMessage;
theTeamName = kUndefinedTeam;
break;
case PLAYMODE_PLAYING:
// Don't set playmode if we couldn't respawn
this->pev->playerclass = PLAYMODE_PLAYING;
//respawn(this->pev, FALSE);
// Account for both sides, or to let player choose it somehow
if(this->GetClassType() == AVH_CLASS_TYPE_MARINE)
{
theUser3 = AVH_USER3_MARINE_PLAYER;
}
else
{
theUser3 = AVH_USER3_ALIEN_PLAYER1; // TEMPORARY
// In combat mode, spawn player in as most advanced lifeform so player doesn't get stuck.
if(GetGameRules()->GetIsCombatMode() && !GetGameRules()->GetIsHamboneMode())
{
// Find the most advanced lifeform
for(MessageIDListType::const_iterator theIter = this->mGiveCombatUpgrades.begin(); theIter != this->mGiveCombatUpgrades.end(); theIter++)
{
AvHMessageID theCurrentCombatUpgrade = *theIter;
switch(theCurrentCombatUpgrade)
{
case ALIEN_LIFEFORM_TWO:
theUser3 = AVH_USER3_ALIEN_PLAYER2;
break;
case ALIEN_LIFEFORM_THREE:
theUser3 = AVH_USER3_ALIEN_PLAYER3;
break;
case ALIEN_LIFEFORM_FOUR:
theUser3 = AVH_USER3_ALIEN_PLAYER4;
break;
case ALIEN_LIFEFORM_FIVE:
theUser3 = AVH_USER3_ALIEN_PLAYER5;
break;
}
}
}
}
this->SetUser3(theUser3, true, false);
respawn(this->pev, FALSE);
this->SetWeaponsForUser3();
// In combat mode, add all player upgrades
if(GetGameRules()->GetIsCombatMode())
{
if(GetGameRules()->GetIsHamboneMode())
{
// Reset tech nodes
this->mGiveCombatUpgrades.clear();
this->mCombatNodes = this->GetTeamPointer()->GetTechNodes();
this->mExperienceLevelsSpent = 0;
}
else if(!GetGameRules()->GetIsIronMan())
{
// Respend upgrades
this->GiveCombatUpgradesOnSpawn();
}
}
theTeam = this->GetTeamPointer();
theTeamName = (theTeam ? theTeam->GetTeamName() : kUndefinedTeam);
this->mLastTimeStartedPlaying = gpGlobals->time;
this->mHasLeftReadyRoom = true;
break;
case PLAYMODE_AWAITINGREINFORCEMENT:
//this->mRole = AVH_USER3_NONE;
// Preserve team, we could be brought back in
this->pev->team = theTeamNumber;
this->pev->playerclass = PLAYMODE_AWAITINGREINFORCEMENT;
theTeam = this->GetTeamPointer();
theTeamName = (theTeam ? theTeam->GetTeamName() : kUndefinedTeam);
this->StartObservingIfNotAlready();
theTeam = this->GetTeamPointer();
theTeamName = (theTeam ? theTeam->GetTeamName() : kUndefinedTeam);
theMessage = kReinforcementMessage;
this->mHasLeftReadyRoom = true;
break;
case PLAYMODE_REINFORCING:
this->pev->team = theTeamNumber;
this->pev->playerclass = PLAYMODE_REINFORCING;
theTeam = this->GetTeamPointer();
theTeamName = (theTeam ? theTeam->GetTeamName() : kUndefinedTeam);
this->StartObservingIfNotAlready();
this->SendMessage(kReinforcingMessage, TOOLTIP);
this->mHasLeftReadyRoom = true;
break;
case PLAYMODE_OBSERVER:
if(theTeam)
{
theTeam->RemovePlayer(this->entindex());
}
// Set observer mode
this->StartObservingIfNotAlready();
// Set current team to indeterminate; we aren't allowed to join from spectating
this->pev->team = TEAM_IND;
this->pev->playerclass = PLAYMODE_OBSERVER;
// Remember that we have spectated
this->mHasBeenSpectator = true;
theMessage = kObserverMessage;
theTeamName = kSpectatorTeam;
this->mHasLeftReadyRoom = true;
this->SetHasSeenTeam(GetGameRules()->GetTeamA()->GetTeamNumber());
this->SetHasSeenTeam(GetGameRules()->GetTeamB()->GetTeamNumber());
break;
case PLAYMODE_REINFORCINGCOMPLETE:
this->pev->playerclass = PLAYMODE_REINFORCINGCOMPLETE;
this->SendMessage(kReinforcementComplete, NORMAL);
break;
}
// Force reset of entities because we just respawned
//this->ResetPlayerPVS();
// Inform gamerules of the change
GetGameRules()->ChangePlayerTeam(this, theTeamName.c_str(), false, false);
if(theTeam)
{
this->SetHasSeenTeam(theTeam->GetTeamNumber());
}
// Inform scoreboard
this->EffectivePlayerClassChanged();
if(theMessage != "")
{
// Send instructions to player
this->SendMessageNextThink(theMessage.c_str());
}
}
}
void AvHPlayer::GetNewOrigin(AvHUser3 inNewUser3, bool inCheckDucking, vec3_t& outOrigin) const
{
vec3_t theOldMinSize;
vec3_t theOldMaxSize;
GetSize(theOldMinSize, theOldMaxSize);
vec3_t theNewMinSize;
vec3_t theNewMaxSize;
AvHSHUGetSizeForPlayerUser3(inNewUser3, theNewMinSize, theNewMaxSize, inCheckDucking);
VectorCopy(pev->origin, outOrigin);
outOrigin[2] += theOldMinSize.z - theNewMinSize.z;
}
void AvHPlayer::SetUser3(AvHUser3 inUser3, bool inForceChange, bool inGiveWeapons)
{
if((inUser3 != this->pev->iuser3) || inForceChange)
{
// Make us duck so that it's easier to gestate in small areas.
if(AvHMUGetCanDuck(this->pev->iuser3))
{
SetBits(this->pev->flags, FL_DUCKING);
SetBits( m_afPhysicsFlags, PFLAG_DUCKING );
// Important or the animations will get screwed up!
m_Activity = ACT_RESET;
SetAnimation( PLAYER_IDLE );
}
bool theIsDucking = FBitSet(this->pev->flags, FL_DUCKING);
// Save off the current size of the player so that we can adjust the
// origin later.
Vector theOldMinSize;
Vector theOldMaxSize;
this->GetSize(theOldMinSize, theOldMaxSize);
vec3_t theNewOrigin;
GetNewOrigin(inUser3, true, theNewOrigin);
this->mPreviousUser3 = (AvHUser3)this->pev->iuser3;
// Leave old User3
switch(this->mPreviousUser3)
{
case AVH_USER3_MARINE_PLAYER:
break;
case AVH_USER3_ALIEN_PLAYER5:
this->StopDigestion(false);
break;
case AVH_USER3_COMMANDER_PLAYER:
this->StopTopDownMode();
break;
}
string theMessage;
this->pev->iuser3 = inUser3;
// Drop inventory, clear abilities
//this->DestroyAllItems(FALSE);
bool theSavedAlienSightActive = this->mAlienSightActive;
this->ClearRoleAbilities();
int theSavedUser4 = this->pev->iuser4;
bool theSavedIsSpectator = this->GetIsSpectator();
float theJetpackEnergy = mSavedJetpackEnergy;
mSavedJetpackEnergy = this->pev->fuser3;
this->ClearUserVariables();
switch(inUser3)
{
case AVH_USER3_NONE:
this->pev->team = TEAM_IND;
break;
case AVH_USER3_MARINE_PLAYER:
this->pev->iuser3 = inUser3;
theMessage = kSoldierMessage;
this->pev->movetype = MOVETYPE_WALK;
break;
case AVH_USER3_COMMANDER_PLAYER:
this->pev->iuser3 = inUser3;
this->StartTopDownMode();
theMessage = kCommanderMessage;
break;
// NOTE: When moving weapons/abilities around, be sure to change AvHBasePlayerWeapon::GetAnimationExtension() also
case AVH_USER3_ALIEN_PLAYER1:
this->pev->iuser3 = inUser3;
this->pev->fuser3 = 1.0f*kNormalizationNetworkFactor;
break;
case AVH_USER3_ALIEN_PLAYER2:
this->pev->iuser3 = inUser3;
this->pev->fuser3 = 1.0f*kNormalizationNetworkFactor;
break;
case AVH_USER3_ALIEN_PLAYER3:
this->pev->iuser3 = inUser3;
this->pev->fuser3 = 1.0f*kNormalizationNetworkFactor;
break;
case AVH_USER3_ALIEN_PLAYER4:
this->pev->iuser3 = inUser3;
this->pev->fuser3 = 1.0f*kNormalizationNetworkFactor;
break;
case AVH_USER3_ALIEN_PLAYER5:
this->pev->iuser3 = inUser3;
this->pev->fuser3 = 1.0f*kNormalizationNetworkFactor;
this->mLastGallopViewDirection = gpGlobals->v_forward;
break;
case AVH_USER3_ALIEN_EMBRYO:
this->pev->iuser3 = inUser3;
this->pev->iuser4 |= MASK_ALIEN_EMBRYO;
theMessage = kGestationMessage;
break;
}
// Preserve upgrades on a role change
this->pev->iuser4 |= theSavedUser4;
// All players are selectable
this->pev->iuser4 |= MASK_SELECTABLE;
// Get team-wide upgrades
AvHTeam* theTeamPointer = this->GetTeamPointer();
if(theTeamPointer)
{
this->pev->iuser4 |= theTeamPointer->GetTeamWideUpgrades();
}
if(inGiveWeapons)
{
this->SetWeaponsForUser3();
}
// Adjust the size for the new user3.
this->SetSizeForUser3();
// Adjust the origin of the player so that they are still on the ground.
//Vector theNewMinSize;
//Vector theNewMaxSize;
//this->GetSize(theNewMinSize, theNewMaxSize);
//this->pev->origin[2] += theOldMinSize.z - theNewMinSize.z;
if(this->mPreviousUser3 != AVH_USER3_COMMANDER_PLAYER)
{
UTIL_SetOrigin(this->pev, theNewOrigin);
}
this->SetViewForUser3();
this->SetMoveTypeForUser3();
if(theSavedIsSpectator)
{
this->SetIsSpectator();
}
if(this->pev->playerclass == PLAYMODE_AWAITINGREINFORCEMENT)
{
this->pev->playerclass = PLAYMODE_PLAYING;
}
//this->SetModelFromState();
float theHealthPercentage = 1.0f;
float theArmorPercentage = 1.0f;
if((this->mPreviousUser3 != AVH_USER3_NONE) /*&& (inUser3 != AVH_USER3_ALIEN_EMBRYO)*/)
{
theHealthPercentage = (float)this->pev->health/AvHPlayerUpgrade::GetMaxHealth(theSavedUser4, this->mPreviousUser3, this->GetExperienceLevel());
theArmorPercentage = (float)this->pev->armorvalue/AvHPlayerUpgrade::GetMaxArmorLevel(theSavedUser4, this->mPreviousUser3);
}
if (mPreviousUser3 == AVH_USER3_COMMANDER_PLAYER)
{
// Restore the jetpack energy from when the player went into the CC.
this->pev->fuser3 = theJetpackEnergy;
}
//char theInitializeMessage[128];
//sprintf(theInitializeMessage, "Initializing to user3: %d with health/armor percentages: %f/%f\n", inUser3, theHealthPercentage, theArmorPercentage);
//ClientPrint(this->pev, HUD_PRINTTALK, theInitializeMessage);
this->InitializeFromTeam(theHealthPercentage, theArmorPercentage);
this->SetModelFromState();
if(this->GetIsAlien())
{
this->mAlienSightActive = theSavedAlienSightActive;
}
else
{
this->mAlienSightActive = false;
}
// Update team
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
theTeam->ProcessRankChange(this, this->mPreviousUser3, this->GetUser3());
}
// Update scoreboard
this->EffectivePlayerClassChanged();
if(theMessage != "")
{
// Send instructions to player
this->SendMessageOnce(theMessage.c_str(), TOOLTIP);
}
this->LogEmitRoleChange();
}
}
void AvHPlayer::SetResources(float inResources, bool inPlaySound)
{
if(!GetGameRules()->GetIsCombatMode())
{
if(inResources < 0)
{
inResources = 0;
}
AvHTeam* theTeam = this->GetTeamPointer();
ASSERT(theTeam != NULL);
if(this->GetIsMarine())
{
if(inPlaySound)
{
this->PlayHUDSound(HUD_SOUND_MARINE_POINTS_RECEIVED);
AvHPlayer* theCommander = this->GetCommander();
if(theCommander)
{
theCommander->PlayHUDSound(HUD_SOUND_MARINE_POINTS_RECEIVED);
}
}
theTeam->SetTeamResources(inResources);
}
else if(this->GetIsAlien())
{
if(inPlaySound)
{
this->PlayHUDSound(HUD_SOUND_ALIEN_POINTS_RECEIVED);
}
if ( this->mResources != inResources )
this->EffectivePlayerClassChanged();
this->mResources = inResources;
}
this->InternalBoundResources();
if(this->GetIsAlien())
{
AvHServerPlayerData* theServerPlayerData = this->GetServerPlayerData();
if(theServerPlayerData)
{
theServerPlayerData->SetResources(this->mResources);
}
}
}
}
void AvHPlayer::Spawn( void )
{
CBasePlayer::Spawn();
//this->PrecacheAndSetPlayerModel();
pev->classname = MAKE_STRING(kAvHPlayerClassName);
this->mSendSpawnScreenFade = true;
if(this->pev->playerclass != PLAYMODE_READYROOM)
{
this->PlayRandomRoleSound(kPlayerLevelSpawnSoundList, CHAN_ITEM, this->GetAlienAdjustedEventVolume());
}
//SET_MODEL(ENT(pev), "models/headcrab.mdl");
//UTIL_SetSize(pev, Vector(-12, -12, 0), Vector(12, 12, 24));
SetTouch(&AvHPlayer::PlayerTouch);
// Stop spectating
this->pev->iuser1 = 0;
}
void AvHPlayer::StartObservingIfNotAlready(void)
{
// Prevent this is the cvar is set
if ( allow_spectators.value )
{
//CBasePlayer *pPlayer = GetClassPtr((CBasePlayer *)pev);
if(!this->IsObserver())
//if(!pPlayer->IsObserver())
{
//if(this->mPlayMode == PLAYMODE_AWAITINGREINFORCEMENT)
//{
// TODO: Start observer mode in chase cam on friendlies
//}
//else
//{
edict_t *pentSpawnSpot = GetGameRules()->SelectSpawnPoint( this /*pPlayer*/ );
if(!FNullEnt(pentSpawnSpot))
{
this->StartObserver( VARS(pentSpawnSpot)->origin, VARS(pentSpawnSpot)->angles);
}
else
{
this->StartObserver( this->pev->origin, this->pev->angles);
}
}
}
else
{
this->ObserverModeIllegal();
}
}
bool AvHPlayer::SetBeingDigestedMode(bool inBeingDigested)
{
bool theSuccess = false;
bool theIsDigesting = this->GetIsBeingDigested();
SetUpgradeMask(&this->pev->iuser4, MASK_DIGESTING, false);
if(inBeingDigested && !theIsDigesting)
{
// Fade player to black
// Vector theFadeColor;
// theFadeColor.x = 0;
// theFadeColor.y = 0;
// theFadeColor.z = 0;
// UTIL_ScreenFade(this, theFadeColor, .7f, 0.0f, 255, FFADE_OUT | FFADE_STAYOUT);
this->HolsterCurrent();
this->pev->solid = SOLID_NOT;
this->pev->effects |= EF_NODRAW;
this->pev->takedamage = DAMAGE_NO;
this->pev->movetype = MOVETYPE_FLY;
this->m_afPhysicsFlags |= PFLAG_OBSERVER;
ClearBits( m_afPhysicsFlags, PFLAG_DUCKING );
ClearBits( this->pev->flags, FL_DUCKING );
VectorCopy(g_vecZero, this->pev->velocity);
theSuccess = true;
}
else if(!inBeingDigested && theIsDigesting)
{
// Fade player up from black
Vector theFadeColor;
theFadeColor.x = 0;
theFadeColor.y = 0;
theFadeColor.z = 0;
UTIL_ScreenFade(this, theFadeColor, 1.0f, 0.0f, 255, FFADE_IN);
this->SetDesiredRoomType(0);
this->DeployCurrent();
// Set physics
this->pev->solid = SOLID_SLIDEBOX;
this->pev->effects &= ~EF_NODRAW;
this->pev->takedamage = DAMAGE_YES;
this->pev->movetype = MOVETYPE_WALK;
ClearBits(this->m_afPhysicsFlags, PFLAG_OBSERVER);
// Set player ducking to improve chances of them not getting stuck
SetBits( m_afPhysicsFlags, PFLAG_DUCKING );
SetBits( this->pev->flags, FL_DUCKING );
VectorCopy(g_vecZero, this->pev->velocity);
this->pev->fixangle = TRUE;
theSuccess = true;
}
if(theSuccess)
{
// Set digesting flag
SetUpgradeMask(&this->pev->iuser4, MASK_DIGESTING, inBeingDigested);
this->EffectivePlayerClassChanged();
}
return theSuccess;
}
void AvHPlayer::StartTopDownMode()
{
if(!this->mInTopDownMode && !this->GetCurrentWeaponCannotHolster())
{
Vector theFadeColor;
theFadeColor.x = 0;
theFadeColor.y = 0;
theFadeColor.z = 0;
UTIL_ScreenFade(this, theFadeColor, kTransitionFadeTime, 0.0f, 255, FFADE_IN);
VectorCopy(this->pev->origin, this->mPositionBeforeTopDown);
VectorCopy(this->pev->angles, this->mAnglesBeforeTopDown);
VectorCopy(this->pev->v_angle, this->mViewAnglesBeforeTopDown);
VectorCopy(this->pev->view_ofs, this->mViewOfsBeforeTopDown);
this->mAnimExtensionBeforeTopDown = this->m_szAnimExtention;
this->HolsterCurrent();
this->mTimeStartedTopDown = gpGlobals->time;
this->mOverwatchEnabled = false;
SetUpgradeMask(&this->pev->iuser4, MASK_TOPDOWN);
this->m_afPhysicsFlags |= PFLAG_OBSERVER;
this->pev->effects |= EF_NODRAW;
this->pev->view_ofs = g_vecZero;
this->pev->gravity = 0;
this->pev->solid = SOLID_NOT;
this->pev->takedamage = DAMAGE_NO;
//this->pev->movetype = MOVETYPE_NOCLIP;
//this->pev->movetype = MOVETYPE_WALK;
this->pev->movetype = MOVETYPE_FLY;
this->m_afPhysicsFlags |= PFLAG_OBSERVER;
ClearBits( m_afPhysicsFlags, PFLAG_DUCKING );
ClearBits( this->pev->flags, FL_DUCKING );
//this->pev->deadflag = DEAD_RESPAWNABLE;
//this->pev->deadflag = DEAD_RESPAWNABLE;
//this->pev->velocity[0] = 0.0f;
//this->pev->velocity[1] = 0.0f;
//this->pev->velocity[2] = -1.0f;
// float theMinViewHeight, theMaxViewHeight;
// float theMinX, theMaxX;
// float theMinY, theMaxY;
// bool theDrawMapBG;
// GetGameRules()->GetMapExtents(theMinViewHeight, theMaxViewHeight, theMinX, theMinY, theMaxX, theMaxY, theDrawMapBG);
this->pev->origin.z = GetGameRules()->GetMapExtents().GetMaxViewHeight();
this->mInTopDownMode = true;
// Cheesy way to make sure player class change is sent to everyone
this->EffectivePlayerClassChanged();
}
}
bool AvHPlayer::GetHasLeftReadyRoom() const
{
return this->mHasLeftReadyRoom;
}
bool AvHPlayer::GetHasJetpack() const
{
return this->GetIsMarine() && GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_7);
}
bool AvHPlayer::GetHasHeavyArmor() const
{
return this->GetIsMarine() && GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_13);
}
bool AvHPlayer::GetHasGivenOrder() const
{
return this->mHasGivenOrder;
}
void AvHPlayer::SetHasGivenOrder(bool inState)
{
this->mHasGivenOrder = inState;
}
float AvHPlayer::GetTimeLastF4() const
{
return this->mTimeOfLastF4;
}
void AvHPlayer::SetTimeLastF4(float inTime)
{
this->mTimeOfLastF4=inTime;
}
float AvHPlayer::GetTimeStartedTopDown() const
{
return this->mTimeStartedTopDown;
}
float AvHPlayer::GetTimeOfLastSignificantCommanderAction() const
{
return this->mTimeOfLastSignificantCommanderAction;
}
bool AvHPlayer::GetHasAvailableUpgrades() const
{
bool theHasPendingUpgrades = false;
AvHTeam* theTeamPointer = this->GetTeamPointer();
if(theTeamPointer)
{
AvHAlienUpgradeListType theUpgrades = theTeamPointer->GetAlienUpgrades();
for(int i = ALIEN_UPGRADE_CATEGORY_INVALID + 1; i < ALIEN_UPGRADE_CATEGORY_MAX_PLUS_ONE; i++)
{
AvHAlienUpgradeCategory theCurrentCategory = AvHAlienUpgradeCategory(i);
// Now make sure we have an unspent upgrade available
if(AvHGetHasFreeUpgradeCategory(theCurrentCategory, theUpgrades, this->pev->iuser4))
{
theHasPendingUpgrades = true;
break;
}
}
}
return theHasPendingUpgrades;
}
bool AvHPlayer::GetHasPowerArmor() const
{
return this->GetIsMarine() && GetHasUpgrade(this->pev->iuser4, MASK_UPGRADE_10);
}
int AvHPlayer::GetHull() const
{
int theHull = AvHMUGetHull(true, this->pev->iuser3);
return theHull;
}
bool AvHPlayer::GetIsTemporarilyInvulnerable() const
{
bool theIsInvulnerable = false;
if(GetGameRules()->GetIsCombatMode() && (this->GetPlayMode() == PLAYMODE_PLAYING))
{
if(this->mLastTimeStartedPlaying != -1)
{
float theInvulnerableTimeSeconds = 0.0f;
#ifdef DEBUG
theInvulnerableTimeSeconds = avh_spawninvulnerabletime.value;
#endif
if(gpGlobals->time < (this->mLastTimeStartedPlaying + theInvulnerableTimeSeconds))
{
theIsInvulnerable = true;
}
}
}
if (mTimeOfLastRedeem != -1 && gpGlobals->time < mTimeOfLastRedeem + kRedeemInvulnerableTime)
{
theIsInvulnerable = true;
}
return theIsInvulnerable;
}
bool AvHPlayer::GetIsEnsnared() const
{
return GetHasUpgrade(this->pev->iuser4, MASK_ENSNARED);
}
bool AvHPlayer::GetIsAbleToAct() const
{
return !GetIsInTopDownMode() && !GetIsBeingDigested() && !GetIsEnsnared() && !GetIsStunned();
}
bool AvHPlayer::SetEnsnareState(bool inState)
{
bool theSuccess = true;
if(inState)
{
// If too ensnared already, don't ensnare further
if(!this->GetIsEnsnared())
{
this->mTimeToBeUnensnared = gpGlobals->time;
}
if(!this->GetIsEnsnared() || ((this->mTimeToBeUnensnared + BALANCE_VAR(kEnsnareTime) - gpGlobals->time) < BALANCE_VAR(kMaxEnsnareTime)))
{
this->mLastTimeEnsnared = gpGlobals->time;
this->mTimeToBeUnensnared += BALANCE_VAR(kEnsnareTime);
// Player is defenseless
this->HolsterCurrent();
SetUpgradeMask(&this->pev->iuser4, MASK_ENSNARED);
}
else
{
theSuccess = false;
}
}
else
{
SetUpgradeMask(&this->pev->iuser4, MASK_ENSNARED, false);
this->mTimeToBeUnensnared = -1;
this->mLastTimeEnsnared = -1;
this->DeployCurrent();
}
return theSuccess;
}
bool AvHPlayer::GetIsStunned() const
{
return GetHasUpgrade(this->pev->iuser4, MASK_PLAYER_STUNNED);
}
bool AvHPlayer::SetIsStunned(bool inState, float inTime)
{
bool theSuccess = false;
// Only able to stun walking players (prevents weird problems with players on ladders, who are treated as flying
if(inState && !this->GetIsStunned() && (this->pev->movetype == MOVETYPE_WALK))
{
SetUpgradeMask(&this->pev->iuser4, MASK_PLAYER_STUNNED);
this->mTimeToBeFreeToMove = gpGlobals->time + inTime;
Vector theFadeColor;
theFadeColor.x = 255;
theFadeColor.y = 255;
theFadeColor.z = 255;
float theFadeTime = .25f;
UTIL_ScreenFade(this, theFadeColor, theFadeTime, 0.0f, 128, FFADE_IN/* | FFADE_MODULATE*/);
theSuccess = true;
// Clear keys so they aren't held down
//this->ClearKeys();
}
else if(!inState && this->GetIsStunned())
{
SetUpgradeMask(&this->pev->iuser4, MASK_PLAYER_STUNNED, false);
this->mTimeToBeFreeToMove = -1;
}
return theSuccess;
}
bool AvHPlayer::GetIsCatalysted() const
{
return this->GetIsMarine() && GetHasUpgrade(this->pev->iuser4, MASK_BUFFED);
}
void AvHPlayer::SetIsCatalysted(bool inState, float inTime)
{
if(this->GetIsMarine())
{
if(inState && !this->GetIsCatalysted())
{
SetUpgradeMask(&this->pev->iuser4, MASK_BUFFED);
this->mTimeToEndCatalyst = gpGlobals->time + inTime;
// Trigger screen effect?
}
else
{
SetUpgradeMask(&this->pev->iuser4, MASK_BUFFED, false);
this->mTimeToEndCatalyst = -1;
}
}
}
bool AvHPlayer::Energize(float inEnergyAmount)
{
bool theSuccess = false;
if(AvHMUGiveAlienEnergy(this->pev->fuser3, inEnergyAmount))
{
theSuccess = true;
}
return theSuccess;
}
bool AvHPlayer::Heal(float inAmount, bool inPlaySound)
{
int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
int theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(this->pev->iuser4, (AvHUser3)this->pev->iuser3);
bool theDidHeal = false;
float theAmount = inAmount;
// If we aren't at full health, heal health
if(this->pev->health < theMaxHealth)
{
int theAmountToGive = theAmount;
theAmount -= (theMaxHealth - this->pev->health); //store relative amount compared to that necessary for complete heal
this->pev->health = min((float)theMaxHealth, this->pev->health + theAmountToGive);
theDidHeal = true;
}
else if(this->pev->armorvalue < theMaxArmor)
{
this->pev->armorvalue = min((float)theMaxArmor, this->pev->armorvalue + theAmount);
theDidHeal = true;
}
// Play regen event
if(theDidHeal)
{
if(inPlaySound)
{
// Play regeneration event
PLAYBACK_EVENT_FULL(0, this->edict(), gRegenerationEventID, 0, this->pev->origin, (float *)&g_vecZero, this->GetAlienAdjustedEventVolume(), 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
}
}
return theDidHeal;
}
bool AvHPlayer::Regenerate(float inRegenerationAmount, bool inPlaySound)
{
bool theDidRegenerate = this->Heal(inRegenerationAmount, inPlaySound);
if(theDidRegenerate)
{
this->mTimeOfLastRegeneration = gpGlobals->time;
}
return theDidRegenerate;
}
bool AvHPlayer::GetCanBeResupplied() const
{
bool theCanBeResupplied = false;
const float theResupplyTime = BALANCE_VAR(kResupplyTime);
if((this->mTimeOfLastResupply == 0) || (gpGlobals->time > (this->mTimeOfLastResupply + theResupplyTime)))
{
if(this->m_pActiveItem)
{
AvHBasePlayerWeapon* theBaseWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem->GetWeaponPtr());
if(theBaseWeapon && theBaseWeapon->CanHolster() && theBaseWeapon->GetCanBeResupplied())
{
theCanBeResupplied = true;
}
}
// If we don't have max health, or we need ammo
if(this->pev->health < this->pev->max_health)
{
theCanBeResupplied = true;
}
}
return theCanBeResupplied;
}
bool AvHPlayer::Resupply(bool inGiveHealth)
{
bool theSuccess = false;
if(this->m_pActiveItem)
{
AvHBasePlayerWeapon* theBaseWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem->GetWeaponPtr());
if(theBaseWeapon && theBaseWeapon->Resupply())
{
theSuccess = true;
}
if(inGiveHealth)
{
// puzl: 1017 armoury gives 10 health per use
if(AvHHealth::GiveHealth(this, BALANCE_VAR(kPointsPerArmouryHealth)))
{
// Play event for each person helped
//PLAYBACK_EVENT_FULL(0, this->edict(), gPhaseInEventID, 0, this->pev->origin, (float *)&g_vecZero, 0.0, 0.0, 0, 0, 0, 0 );
theSuccess = true;
}
}
this->mTimeOfLastResupply = gpGlobals->time;
}
return theSuccess;
}
bool AvHPlayer::GetIsScreaming()
{
return this->mIsScreaming;
}
void AvHPlayer::StartScreaming()
{
this->mIsScreaming = true;
this->mTimeStartedScream = gpGlobals->time;
}
bool AvHPlayer::StopTopDownMode()
{
bool theSuccess = false;
if(this->mInTopDownMode)
{
Vector theFadeColor;
theFadeColor.x = 0;
theFadeColor.y = 0;
theFadeColor.z = 0;
UTIL_ScreenFade(this, theFadeColor, kTransitionFadeTime, 0.0f, 255, FFADE_IN);
this->DeployCurrent();
this->mOverwatchEnabled = true;
this->pev->effects &= ~EF_NODRAW;
this->pev->view_ofs = g_vecZero;
SetUpgradeMask(&this->pev->iuser4, MASK_TOPDOWN, false);
// TODO: Make sure original gravity is 1?
this->pev->gravity = 1;
this->pev->solid = SOLID_SLIDEBOX;
this->pev->takedamage = DAMAGE_YES;
//this->pev->movetype = MOVETYPE_NONE;
this->pev->movetype = MOVETYPE_WALK;
ClearBits(this->m_afPhysicsFlags, PFLAG_OBSERVER);
this->pev->deadflag = DEAD_NO;
VectorCopy(this->mPositionBeforeTopDown, this->pev->origin);
VectorCopy(this->mAnglesBeforeTopDown, this->pev->angles);
VectorCopy(this->mViewAnglesBeforeTopDown, this->pev->v_angle);
VectorCopy(this->mViewOfsBeforeTopDown, this->pev->view_ofs);
strcpy(this->m_szAnimExtention, this->mAnimExtensionBeforeTopDown.c_str());
VectorCopy(g_vecZero, this->pev->velocity);
this->pev->fixangle = TRUE;
this->mInTopDownMode = false;
AvHTeamNumber theStationTeamNumber = (AvHTeamNumber)this->pev->team;
const char* theTarget = (theStationTeamNumber == TEAM_ONE) ? kTargetCommandStationLogoutTeamOne : kTargetCommandStationLogoutTeamTwo;
FireTargets(theTarget, NULL, NULL, USE_TOGGLE, 0.0f);
this->mSelected.clear();
// Need to reset groups when logging out, so they are re-propagated when logging in after another potential commander (see AvHHud::ResetTopDownUI)
for(int i = 0; i < kNumHotkeyGroups; i++)
{
this->mClientGroupAlerts[i] = ALERT_NONE;
this->mClientGroups[i].clear();
}
this->mClientSelectAllGroup.clear();
theSuccess = true;
}
return theSuccess;
}
void AvHPlayer::SetPendingCommand(char* inCommand)
{
this->mPendingCommand = inCommand;
}
void AvHPlayer::TriggerFog(int inFogEntity, float inFogExpireTime)
{
this->mCurrentFogEntity = inFogEntity;
this->mFogExpireTime = inFogExpireTime;
// Allows resetting of fog entity
if(inFogEntity > -1)
{
this->mTimeOfLastFogTrigger = gpGlobals->time;
}
}
void AvHPlayer::TriggerProgressBar(int inEntityID, int inParam)
{
ASSERT(inEntityID >= 0);
this->mProgressBarEntityIndex = inEntityID;
this->mProgressBarParam = inParam;
this->mTimeProgressBarTriggered = gpGlobals->time;
}
float AvHPlayer::GetTimeOfLastTeleport() const
{
return this->mTimeOfLastTeleport;
}
float AvHPlayer::GetTimeLastPlaying() const
{
return this->mTimeLastPlaying;
}
bool AvHPlayer::HolsterWeaponToUse()
{
bool theSuccess = false;
if(!this->GetCurrentWeaponCannotHolster())
{
this->HolsterCurrent();
this->mTimeOfLastUse = gpGlobals->time;
theSuccess = true;
}
return theSuccess;
}
void AvHPlayer::SetTimeOfLastTeleport(float inTime)
{
this->mTimeOfLastTeleport = inTime;
}
void AvHPlayer::BecomePod()
{
//ASSERT(this->mRole != AVH_USER3_ALIEN_EMBRYO);
this->HolsterCurrent();
ClearBits(this->m_afPhysicsFlags, PFLAG_DUCKING);
ClearBits(this->pev->flags, FL_DUCKING);
//EMIT_SOUND_DYN(ENT(this->pev), CHAN_VOICE, kGestationSound, 1, ATTN_NORM, 0, 100);
float flSilenceLevel = this->GetAlienAdjustedEventVolume();
if(flSilenceLevel > 0.0)
UTIL_EmitAmbientSound(ENT(this->pev), this->pev->origin, kGestationSound, flSilenceLevel, 2.0, 0, 100);
}
bool AvHPlayer::SwitchWeapon(const char* inString)
{
bool theSuccess = false;
if(!this->GetIsEnsnared())
{
CBasePlayerWeapon* theCurrentWeapon;
for (int i = 0 ; i < MAX_ITEM_TYPES ; i++ )
{
theCurrentWeapon = dynamic_cast<CBasePlayerWeapon*>(this->m_rgpPlayerItems[i]);
while( theCurrentWeapon )
{
if(FClassnameIs(theCurrentWeapon->pev, inString))
{
// this weapon is from the same category.
if ( theCurrentWeapon->CanDeploy() )
{
theSuccess = CBasePlayer::SwitchWeapon( theCurrentWeapon );
}
break;
}
theCurrentWeapon = dynamic_cast<CBasePlayerWeapon*>(theCurrentWeapon->m_pNext);
}
}
}
return theSuccess;
}
//BOOL AvHPlayer::SwitchWeapon( CBasePlayerItem* inWeapon )
//{
// CBasePlayerWeapon* theCurrentWeapon = dynamic_cast<CBasePlayerWeapon*>(this->m_pActiveItem);
// CBasePlayerItem* theWeapon = inWeapon;
// BOOL theSuccess = TRUE;
//
// if(!inWeapon)
// {
// if(theCurrentWeapon)
// {
// theCurrentWeapon->RetireWeapon();
// }
// else
// {
// theSuccess = FALSE;
// }
// }
// else
// {
// CBasePlayer::SwitchWeapon(theWeapon);
// }
//
// return theSuccess;
//}
void AvHPlayer:: TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType)
{
if ( pev->takedamage && GetCanBeAffectedByEnemies())
{
m_LastHitGroup = ptr->iHitgroup;
// No locational damage in NS.
/*
switch ( ptr->iHitgroup )
{
case HITGROUP_GENERIC:
break;
case HITGROUP_HEAD:
flDamage *= gSkillData.plrHead;
break;
case HITGROUP_CHEST:
flDamage *= gSkillData.plrChest;
break;
case HITGROUP_STOMACH:
flDamage *= gSkillData.plrStomach;
break;
case HITGROUP_LEFTARM:
case HITGROUP_RIGHTARM:
flDamage *= gSkillData.plrArm;
break;
case HITGROUP_LEFTLEG:
case HITGROUP_RIGHTLEG:
flDamage *= gSkillData.plrLeg;
break;
default:
break;
}
*/
// Player's aren't affected by structural damage, so don't create blood
// if that's the damage type.
if (!(bitsDamageType & NS_DMG_STRUCTURAL))
{
SpawnBlood(ptr->vecEndPos, BloodColor(), flDamage);
TraceBleed( flDamage, vecDir, ptr, bitsDamageType );
}
AddMultiDamage( pevAttacker, this, flDamage, bitsDamageType );
}
}
int AvHPlayer::TakeDamage( entvars_t* pevInflictor, entvars_t* pevAttacker, float flDamage, int bitsDamageType)
{
//Even out the damage
//flDamage = ceil(flDamage);
int theReturnValue = 0;
if(GetGameRules()->GetGameStarted() && !this->GetIsTemporarilyInvulnerable())
{
// Take into account handicap
if(!pevAttacker)
{
pevAttacker = pevInflictor;
}
if(!pevInflictor)
{
pevInflictor = pevAttacker;
}
AvHTeam* theTeam = GetGameRules()->GetTeam(AvHTeamNumber(pevAttacker->team));
if(theTeam)
{
float theHandicap = theTeam->GetHandicap();
flDamage *= theHandicap;
}
if(GetGameRules()->GetIsCheatEnabled(kcHighDamage))
{
flDamage *= 10;
}
if(bitsDamageType & NS_DMG_STRUCTURAL)
{
flDamage = 0.0f;
}
// Do half damage to the heavy armor of HA and Onos
if(bitsDamageType & NS_DMG_LIGHT)
{
if(this->GetHasHeavyArmor() || (this->pev->iuser3 == AVH_USER3_ALIEN_PLAYER5))
{
flDamage *= .5f;
}
}
// If we're metabolizing, convert the damage to energy
// if(this->GetIsMetabolizing())
// {
// const float theFactor = BALANCE_VAR(kMetabolizeDamageEnergyFactor);
// float theEnergy = (flDamage/100.f)*theFactor;
// AvHMUGiveAlienEnergy(this->pev->fuser3, theEnergy);
//
// if((this->mTimeOfLastMetabolizeEvent == -1) || (gpGlobals->time > (this->mTimeOfLastMetabolizeEvent + 1.0f)))
// {
// // Playback metabolize success event
// PLAYBACK_EVENT_FULL(0, this->edict(), gMetabolizeSuccessEventID, 0, this->pev->origin, (float *)&g_vecZero, this->GetAlienAdjustedEventVolume(), 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
//
// this->mTimeOfLastMetabolizeEvent = gpGlobals->time;;
// }
//
// theReturnValue = 0;
// }
// else
// {
theReturnValue = CBasePlayer::TakeDamage(pevInflictor, pevAttacker, flDamage, bitsDamageType);
if(theReturnValue > 0)
{
float theSlowDownFactor = .8f;
float theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(this->pev->iuser4, (AvHUser3)this->pev->iuser3, this->GetExperienceLevel());
if(flDamage > theMaxHealth/2.0f)
{
this->PlayRandomRoleSound(kPlayerLevelWoundSoundList, CHAN_BODY, 1.0);
theSlowDownFactor = .3f;
}
else if(flDamage >= theMaxHealth/5.0f)
{
this->PlayRandomRoleSound(kPlayerLevelPainSoundList, CHAN_BODY, .8f);
theSlowDownFactor = .5f;
}
// Slow down when hit
//VectorScale(this->pev->velocity, theSlowDownFactor, this->pev->velocity);
if(!pevAttacker || (this->pev->team != pevAttacker->team) && (pevAttacker->team != 0))
{
GetGameRules()->TriggerAlert((AvHTeamNumber)this->pev->team, ALERT_PLAYER_ENGAGE, this->entindex());
}
if(pevAttacker)
{
CBasePlayer* inAttackingPlayer = dynamic_cast<CBasePlayer*>(CBaseEntity::Instance(ENT(pevAttacker)));
const char* inWeaponName = STRING(pevInflictor->classname);
if(inAttackingPlayer && inWeaponName)
{
this->LogPlayerAttackedPlayer(inAttackingPlayer, inWeaponName, flDamage);
}
}
bool theDrawDamage = (CVAR_GET_FLOAT(kvDrawDamage) > 0);
if(theDrawDamage)
{
this->PlaybackNumericalEvent(kNumericalInfoHealthEvent, (int)(-flDamage));
}
this->Uncloak();
}
// }
}
return theReturnValue;
}
void AvHPlayer::PlaybackNumericalEvent(int inEventID, int inNumber)
{
Vector theMinSize;
Vector theMaxSize;
this->GetSize(theMinSize, theMaxSize);
Vector theStartPos = this->pev->origin;
theStartPos.z += theMaxSize.z;
// Draw for everyone (team = 0 after flDamage parameter)
AvHSUPlayNumericEvent(inNumber, this->edict(), theStartPos, 0, inEventID, this->pev->team);
}
//const char* AvHPlayer::TeamID( void )
//{
// AvHTeam* theTeam = this->GetTeamPointer();
// const char* theTeamName = (theTeam ? theTeam->GetTeamName() : kUndefinedTeam);
// //const char* theTeamName = this->GetPlayerModelKeyName();
// return theTeamName;
//}
void AvHPlayer::TurnOffOverwatch()
{
this->mInOverwatch = false;
AvHBasePlayerWeapon* theWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
if(theWeapon)
{
theWeapon->SetOverwatchState(false);
}
//VectorCopy(this->mOverwatchFacing, this->pev->angles);
//this->pev->fixangle = TRUE;
//ASSERT(this->pev->iuser4 == AVH_USER4_OVERWATCH);
//this->pev->iuser4 &= ~MASK_MARINE_OVERWATCH;
this->pev->fuser1 = -1;
this->pev->fuser2 = -1;
}
void AvHPlayer::TurnOnOverwatch()
{
this->mInOverwatch = true;
// Remember facing when we entered overwatch
VectorCopy(this->pev->angles, this->mOverwatchFacing);
// if so, set overwatch on, make sure to set the current weapon into overwatch
AvHBasePlayerWeapon* theWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_pActiveItem);
ASSERT(theWeapon);
theWeapon->SetOverwatchState(true);
// Flip on overwatch, clear target, it will be acquired in think
this->mOverwatchTarget = -1;
this->pev->fuser1 = -1;
this->pev->fuser2 = -1;
//this->pev->iuser4 |= MASK_MARINE_OVERWATCH;
}
void AvHPlayer::TurnOverwatchTowardsTarget(CBaseEntity* theTarget)
{
// TODO: Take gun offset into account with vecMid?
Vector vecMid = pev->origin + pev->view_ofs;
Vector vecMidEnemy = theTarget->BodyTarget( vecMid );
// Right now just point at enemy
Vector vecDirToEnemy = vecMidEnemy - vecMid;
Vector vec = UTIL_VecToAngles(vecDirToEnemy);
vec.x = -vec.x;
// if (vec.y > 360)
// vec.y -= 360;
//
// if (vec.y < 0)
// vec.y += 360;
VectorCopy(vec, this->pev->angles);
VectorCopy(vec, this->pev->v_angle);
this->pev->fixangle = TRUE;
}
bool AvHPlayer::RunClientScript(const string& inScriptName)
{
// Returns false if client scripts aren't enabled
bool theSuccess = false;
this->mPendingClientScripts.push_back(inScriptName);
theSuccess = true;
return theSuccess;
}
void AvHPlayer::PrintWeaponListToClient(CBaseEntity *theAvHPlayer) {
char msg[1024];
sprintf(msg, "Weapons for %s:\n", this->GetPlayerName());
ClientPrint(theAvHPlayer->pev, HUD_PRINTNOTIFY, msg);
for(int i = 0; i < MAX_ITEM_TYPES; i++)
{
AvHBasePlayerWeapon* theActiveWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_rgpPlayerItems[i]);
while(theActiveWeapon)
{
theActiveWeapon->PrintWeaponToClient(theAvHPlayer);
// Next weapon
theActiveWeapon = dynamic_cast<AvHBasePlayerWeapon*>(theActiveWeapon->m_pNext);
}
}
}
void AvHPlayer::UpdateInventoryEnabledState(int inNumActiveHives, bool inForceUpdate)
{
// Have we not yet updated our weapons with this # of hives?
if((inNumActiveHives != this->mNumHives) || (inForceUpdate))
{
for(int i = 0; i < MAX_ITEM_TYPES; i++)
{
AvHBasePlayerWeapon* theActiveWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_rgpPlayerItems[i]);
while(theActiveWeapon)
{
theActiveWeapon->UpdateInventoryEnabledState(inNumActiveHives);
// Next weapon
theActiveWeapon = dynamic_cast<AvHBasePlayerWeapon*>(theActiveWeapon->m_pNext);
}
}
}
// Save # of hives we've last updated with
this->mNumHives = inNumActiveHives;
}
bool AvHPlayer::GetIsBeingDigested() const
{
bool theIsBeingDigested = false;
if(GetHasUpgrade(this->pev->iuser4, MASK_DIGESTING))
{
if(this->pev->effects & EF_NODRAW)
{
theIsBeingDigested = true;
}
}
return theIsBeingDigested;
}
bool AvHPlayer::GetIsDigesting() const
{
bool theIsDigesting = false;
if(GetHasUpgrade(this->pev->iuser4, MASK_DIGESTING))
{
if(! (this->pev->effects & EF_NODRAW))
{
theIsDigesting = true;
}
}
return theIsDigesting;
}
void AvHPlayer::UpdateAmbientSounds()
{
AvHClassType theClassType = this->GetClassType();
if(theClassType == AVH_CLASS_TYPE_MARINE)
{
}
else if(theClassType == AVH_CLASS_TYPE_ALIEN)
{
// Get role
AvHUser3 theUser3 = this->GetUser3();
int theAlienLevel = theUser3 - AVH_USER3_COMMANDER_PLAYER;
int theVelocity = this->pev->velocity.Length();
bool theIsMoving = theVelocity > 100;
int theSilenceLevel = AvHGetAlienUpgradeLevel(this->pev->iuser4, MASK_UPGRADE_6);
// if(RANDOM_LONG(0, 40) == 0)
// {
// char theVelocityMsg[128];
// sprintf(theVelocityMsg, "alien moving at velocity: %d", theVelocity);
// ClientPrint(this->pev, HUD_PRINTTALK, theVelocityMsg);
// }
// if moving, check chance for playing moving sound
if(!this->GetIsCloaked() && !this->mIsScreaming /*&& !this->GetIsBlinking()*/)
{
float theSilenceVolumeFactor = this->GetAlienAdjustedEventVolume();
if(theIsMoving)
{
int theBaseSpeed, theMaxSpeed;
this->GetSpeeds(theBaseSpeed, theMaxSpeed);
float theAlienSoundFreq = 0.003f;
float theChanceOfPlayingSound = theAlienSoundFreq*(theVelocity/((float)theMaxSpeed));
if(RANDOM_FLOAT(0, 1) < theChanceOfPlayingSound)
{
float theVolume = RANDOM_FLOAT(.5, 1.0)*theSilenceVolumeFactor;
if(theVolume > 0.01f)
{
this->PlayRandomRoleSound(kPlayerLevelMoveSoundList, CHAN_VOICE, theVolume);
}
}
}
else
{
if(GetHasUpgrade(this->pev->iuser4, MASK_BUFFED))
{
float theSilenceVolumeFactor = this->GetAlienAdjustedEventVolume();
// If player is part of primal scream, scream defiance!
if(RANDOM_FLOAT(0, 1) < .02f && theSilenceVolumeFactor > 0.0)
{
EMIT_SOUND(this->edict(), CHAN_VOICE, kPrimalScreamResponseSound, theSilenceVolumeFactor, ATTN_NORM);
}
}
else
{
bool theIsGestating = (this->GetUser3() == AVH_USER3_ALIEN_EMBRYO);
// if idle, check chance for playing idle sound
float theBaseChance = 0.0005f;
if(theIsGestating)
{
theBaseChance *= 10;
}
if(RANDOM_FLOAT(0, 1) < theBaseChance)
{
float theVolume = RANDOM_FLOAT(.2, .4)*theSilenceVolumeFactor;
if(theVolume > 0.01f)
{
if(theIsGestating)
{
EMIT_SOUND(this->edict(), CHAN_AUTO, kEggIdleSound, theVolume, ATTN_NORM);
}
else
{
this->PlayRandomRoleSound(kPlayerLevelIdleSoundList, CHAN_VOICE, theVolume);
}
}
}
}
}
}
}
if(this->mDesiredRoomType != this->mClientDesiredRoomType)
{
MESSAGE_BEGIN( MSG_ONE, SVC_ROOMTYPE, NULL, this->pev);
WRITE_SHORT( (short)this->mDesiredRoomType);
MESSAGE_END();
this->mClientDesiredRoomType = this->mDesiredRoomType;
}
}
void AvHPlayer::UpdateAlienUI()
{
AvHTeam* theTeamPointer = this->GetTeamPointer();
bool theIsMarine = false;
bool theIsAlien = false;
if(this->GetIsAlien())
{
theIsAlien = true;
}
else if(this->GetIsMarine())
{
theIsMarine = true;
}
// Update when going back to ready room, so check if not-marine instead of is-alien
if(!theIsMarine)
{
bool theCanRespawn = GetGameRules()->FPlayerCanRespawn(this);
AvHTeam* theTeamPointer = this->GetTeamPointer();
if(theTeamPointer)
{
AvHAlienUpgradeListType theUpgrades = theTeamPointer->GetAlienUpgrades();
int thePreSize = theUpgrades.size();
// Trim max upgrades in each category
AvHRemoveIrrelevantUpgrades(theUpgrades);
if(theUpgrades != this->mClientUpgrades)
{
NetMsg_AlienInfo_Upgrades( this->pev, theUpgrades );
this->mClientUpgrades = theUpgrades;
}
HiveInfoListType theTeamHiveInfo = theTeamPointer->GetHiveInfoList();
if(this->mClientHiveInfo != theTeamHiveInfo)
{
NetMsg_AlienInfo_Hives( this->pev, theTeamHiveInfo, this->mClientHiveInfo );
this->mClientHiveInfo = theTeamHiveInfo;
}
}
}
}
void AvHPlayer::UpdateMarineUI()
{
AvHTeam* theTeamPointer = this->GetTeamPointer();
bool theIsMarine = false;
bool theIsAlien = false;
if(this->GetIsAlien())
{
return;
}
int tmpUpgrades=0;
for(int i = 0; i < MAX_ITEM_TYPES; i++)
{
AvHBasePlayerWeapon* theActiveWeapon = dynamic_cast<AvHBasePlayerWeapon*>(this->m_rgpPlayerItems[i]);
while(theActiveWeapon)
{
ItemInfo ii;
theActiveWeapon->GetItemInfo(&ii);
switch ( ii.iId ) {
case AVH_WEAPON_WELDER:
tmpUpgrades |= 0x1;
break;
case AVH_WEAPON_MINE:
tmpUpgrades |= 0x2;
break;
case AVH_WEAPON_GRENADE:
tmpUpgrades |= 0x4;
break;
}
// Next weapon
theActiveWeapon = dynamic_cast<AvHBasePlayerWeapon*>(theActiveWeapon->m_pNext);
}
}
if ( tmpUpgrades != this->mMarineHUDUpgrades ) {
NetMsg_HUDSetUpgrades(this->pev, tmpUpgrades&0x7);
this->mMarineHUDUpgrades=tmpUpgrades;
}
}
// TODO: Send only changed blips, send only the changes for each blip.
void AvHPlayer::UpdateBlips()
{
if(this->mEnemyBlips != this->mClientEnemyBlips)
{
NetMsg_BlipList( this->pev, false, this->mEnemyBlips );
this->mClientEnemyBlips = this->mEnemyBlips;
}
if(this->mFriendlyBlips != this->mClientFriendlyBlips)
{
NetMsg_BlipList( this->pev, true, this->mFriendlyBlips );
this->mClientFriendlyBlips = this->mFriendlyBlips;
}
}
void AvHPlayer::UpdateClientData( void )
{
if(GET_RUN_CODE(128))
{
//UTIL_LogPrintf("UpdateClientData starting.\n");
CBasePlayer::UpdateClientData();
// Update one-shot stuff
this->UpdateFirst();
this->UpdateParticleTemplates();
this->UpdateInfoLocations();
this->UpdateOrders();
this->UpdateVUser4();
this->UpdateProgressBar();
this->UpdateSetSelect();
this->UpdateTechNodes();
this->UpdateBalanceVariables();
this->UpdateSoundNames();
this->UpdateTopDownMode();
this->UpdateEntityHierarchy();
this->UpdateExperienceLevelsSpent();
this->UpdateSpawnScreenFade();
this->UpdateEffectiveClassAndTeam();
//this->UpdateArmor();
//this->UpdateOverwatch();
this->UpdatePendingClientScripts();
this->UpdateGamma();
this->UpdateBlips();
this->UpdateAlienUI();
this->UpdateMarineUI();
this->UpdateFog();
//this->UpdateDebugCSP();
}
//UTIL_LogPrintf("UpdateClientData done.\n");
}
void AvHPlayer::UpdateEffectiveClassAndTeam()
{
// Don't send too many messages when these get updated really quickly. Too many messages are being sent on a game reset, and it's not needed. We only need the most recent message.
const float kClassAndTeamUpdateRate = .6f;
if((this->mTimeOfLastClassAndTeamUpdate == -1) || (gpGlobals->time > (this->mTimeOfLastClassAndTeamUpdate + kClassAndTeamUpdateRate)))
{
if(this->mEffectivePlayerClassChanged)
{
int theAuthMask = this->GetAuthenticationMask();
int theTotalScore = this->mScore + this->pev->frags /*- this->m_iDeaths*/;
if(GetGameRules()->GetIsCombatMode())
{
int theCurrentLevel = AvHPlayerUpgrade::GetPlayerLevel(this->GetExperience());
theTotalScore += max((theCurrentLevel - 1), 0);
}
ScoreInfo info;
info.player_index = ENTINDEX(this->edict());
info.score = theTotalScore;
info.frags = this->pev->frags;
info.deaths = this->m_iDeaths;
info.player_class = this->GetEffectivePlayerClass();
info.auth = this->GetAuthenticationMask();
info.team = GetGameRules()->GetTeamIndex(this->TeamID());
if ( GetGameRules()->GetIsCombatMode()) {
info.extra=this->GetExperienceLevel();
}
else {
if ( this->GetIsAlien() ) {
info.extra=(int)this->GetResources();
}
else {
info.extra=0;
if ( this->m_rgpPlayerItems ) {
CBasePlayerWeapon* thePrimaryWeapon = (CBasePlayerWeapon *)this->m_rgpPlayerItems[1];
if ( thePrimaryWeapon ) {
ItemInfo ii;
thePrimaryWeapon->GetItemInfo(&ii);
info.extra=ii.iId;
}
}
}
}
NetMsg_ScoreInfo( info );
this->mEffectivePlayerClassChanged = false;
}
if(this->mNeedsTeamUpdate)
{
for (int i = 1; i <= gpGlobals->maxClients; i++ )
{
CBasePlayer *plr = (CBasePlayer*)UTIL_PlayerByIndex( i );
if ( plr && GetGameRules()->IsValidTeam( plr->TeamID() ) )
{
NetMsg_TeamInfo( this->pev, plr->entindex(), plr->TeamID() );
}
}
this->mNeedsTeamUpdate = false;
}
if(this->mSendTeamUpdate)
{
// notify everyone's HUD of the team change
NetMsg_TeamInfo( this->entindex(), this->TeamID() );
this->mSendTeamUpdate = false;
}
this->mTimeOfLastClassAndTeamUpdate = gpGlobals->time;
}
}
void AvHPlayer::UpdateFirst()
{
if(this->mFirstUpdate)
{
// Tell this player to reset
int theState = (this->mNewMap ? kGameStatusResetNewMap : kGameStatusReset);
NetMsg_GameStatus_State( this->pev, theState, GetGameRules()->GetMapMode() );
if(this->mNewMap)
{
NetMsg_SetSoundNames( this->pev, true, string() );
this->mClientSoundNames.clear();
// Send down map extents so players can start computing it
// Cache this so it isn't computed every round, only the when a player connects or a new map starts?
const char* theCStrLevelName = STRING(gpGlobals->mapname);
const AvHMapExtents& theMapExtents = GetGameRules()->GetMapExtents();
ASSERT(theCStrLevelName);
ASSERT(!FStrEq(theCStrLevelName, ""));
float mins[3] = { theMapExtents.GetMinMapX(), theMapExtents.GetMinMapY(), theMapExtents.GetMinViewHeight() };
float maxs[3] = { theMapExtents.GetMaxMapX(), theMapExtents.GetMaxMapY(), theMapExtents.GetMaxViewHeight() };
NetMsg_SetupMap_Extents( this->pev, string( theCStrLevelName ), mins, maxs, theMapExtents.GetDrawMapBG() );
}
this->mFirstUpdate = false;
this->mNewMap = false;
}
}
void AvHPlayer::UpdateFog()
{
if(this->mClientCurrentFogEntity != this->mCurrentFogEntity)
{
bool theFogEnabled = this->mCurrentFogEntity > -1;
int theR, theG, theB;
float theStart, theEnd;
if(theFogEnabled)
{
AvHFog* theFogEntity = dynamic_cast<AvHFog*>(CBaseEntity::Instance(g_engfuncs.pfnPEntityOfEntIndex(this->mCurrentFogEntity)));
ASSERT(theFogEntity);
theFogEntity->GetFogColor(theR, theG, theB);
theStart = theFogEntity->GetFogStart();
theEnd = theFogEntity->GetFogEnd();
}
NetMsg_Fog( this->pev, theFogEnabled, theR, theG, theB, theStart, theEnd );
this->mClientCurrentFogEntity = this->mCurrentFogEntity;
}
}
void AvHPlayer::UpdateGamma()
{
float theMapGamma = GetGameRules()->GetMapGamma();
if(this->mClientGamma != theMapGamma)
{
if(!GetGameRules()->GetIsTesting())
{
NetMsg_SetGammaRamp( this->pev, theMapGamma );
this->mClientGamma = theMapGamma;
}
}
}
void AvHPlayer::UpdateOrders()
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
OrderListType theTeamOrders;
theTeam->GetOrders(theTeamOrders);
for(OrderListType::iterator theIter = theTeamOrders.begin(); theIter != theTeamOrders.end(); theIter++)
{
bool theClientHasOrder = false;
AvHOrder theClientOrder;
// Does client already have this order?
for(OrderListType::iterator theClientIter = this->mClientOrders.begin(); theClientIter != this->mClientOrders.end(); theClientIter++)
{
if(theIter->GetOrderID() == theClientIter->GetOrderID())
{
theClientHasOrder = true;
theClientOrder = *theClientIter;
break;
}
}
if(!theClientHasOrder || theClientOrder != *theIter)
{
NetMsg_SetOrder( this->pev, *theIter );
}
}
this->mClientOrders = theTeamOrders;
}
}
void AvHPlayer::UpdateParticleTemplates()
{
const float kParticleTemplateRate = 1.0f;
if(gParticleTemplateList.GetCreatedTemplates())
{
// Make sure client clears out all particle systems first
int theNumberTemplates = gParticleTemplateList.GetNumberTemplates();
if(theNumberTemplates > this->mNumParticleTemplatesSent)
{
if((this->mTimeOfLastParticleTemplateSending == -1) || (gpGlobals->time > this->mTimeOfLastParticleTemplateSending + kParticleTemplateRate))
{
AvHParticleTemplate* theTemplate = gParticleTemplateList.GetTemplateAtIndex(this->mNumParticleTemplatesSent);
ASSERT(theTemplate);
NetMsg_SetParticleTemplate( this->pev, *theTemplate );
this->mNumParticleTemplatesSent++;
this->mTimeOfLastParticleTemplateSending = gpGlobals->time;
}
}
}
}
void AvHPlayer::UpdateInfoLocations()
{
// Get map location list
const AvHBaseInfoLocationListType& theInfoLocations = GetGameRules()->GetInfoLocations();
// Compare with ours, send one down each tick (assumes that previous ones sent don't change)
int theNumClientInfoLocations = this->mClientInfoLocations.size();
if((signed)theInfoLocations.size() > theNumClientInfoLocations)
{
// Only send one at a time
AvHBaseInfoLocation theInfoLocation = theInfoLocations[theNumClientInfoLocations];
vec3_t theMaxExtents = theInfoLocation.GetMaxExtent();
vec3_t theMinExtents = theInfoLocation.GetMinExtent();
float mins[3] = { theMinExtents.x, theMinExtents.y, theMinExtents.z };
float maxs[3] = { theMaxExtents.x, theMaxExtents.y, theMinExtents.z };
NetMsg_SetupMap_Location( this->pev, theInfoLocation.GetLocationName(), mins, maxs );
this->mClientInfoLocations.push_back(theInfoLocation);
}
}
void AvHPlayer::UpdatePendingClientScripts()
{
if(this->mPendingClientScripts.size() > 0)
{
NetMsg_ClientScripts( this->pev, this->mPendingClientScripts );
this->mPendingClientScripts.clear();
}
}
void AvHPlayer::UpdateProgressBar()
{
// TODO: If this is the commander, send him all the progress bars of all his teammates so he can see them!
// Assumes that progress is normalized and stored in one of the fuser variables of the entity index sent down
if(this->mClientProgressBarEntityIndex != this->mProgressBarEntityIndex)
{
NetMsg_ProgressBar( this->pev, this->mProgressBarEntityIndex, this->mProgressBarParam );
this->mClientProgressBarEntityIndex = this->mProgressBarEntityIndex;
}
}
void AvHPlayer::UpdateVUser4()
{
// Update client with resources (as int)
int theResources = (short)(this->GetResources(true));
if(CVAR_GET_FLOAT(kvTesting))
{
theResources = g_engfuncs.pfnNumberOfEntities();
}
if(this->pev)
{
if(GetGameRules()->GetIsCombatMode())
{
this->pev->vuser4.z = this->GetExperience()*kNumericNetworkConstant;
}
else
{
this->pev->vuser4.z = theResources*kNumericNetworkConstant;
}
}
}
void AvHPlayer::UpdateSetSelect()
{
if(this->GetIsInTopDownMode())
{
if((this->mSelected != this->mClientSelected) || (this->mTrackingEntity != this->mClientTrackingEntity))
{
Selection selection;
selection.group_number = 0;
selection.selected_entities = this->mSelected;
selection.tracking_entity = max( this->mTrackingEntity, 0 );
NetMsg_SetSelect( this->pev, selection );
// Synch up
this->mClientSelected = this->mSelected;
this->mClientTrackingEntity = this->mTrackingEntity;
}
AvHTeam* theTeam = this->GetTeamPointer();
ASSERT(theTeam);
for(int j=0; j < kNumHotkeyGroups; j++)
{
EntityListType theGroup = theTeam->GetGroup(j);
EntityListType& theClientGroup = this->mClientGroups[j];
AvHUser3 theGroupType = theTeam->GetGroupType(j);
AvHAlertType theGroupAlert = ALERT_NONE;
AvHAlertType& theClientGroupAlert = this->mClientGroupAlerts[j];
// Is group under attack or no longer under attack?
for(EntityListType::iterator theIter = theGroup.begin(); theIter != theGroup.end(); theIter++)
{
if(GetGameRules()->GetIsEntityUnderAttack(*theIter))
{
theGroupAlert = ALERT_UNDER_ATTACK;
}
}
if((theClientGroup != theGroup) || (theClientGroupAlert != theGroupAlert))
{
Selection selection;
selection.group_number = j+1;
selection.selected_entities = theGroup;
selection.group_type = theGroupType;
selection.group_alert = theGroupAlert;
NetMsg_SetSelect( this->pev, selection );
theClientGroup = theGroup;
theClientGroupAlert = theGroupAlert;
}
}
// See if "selectall" hotgroup has changed and send it if needed
EntityListType theSelectAllGroup = theTeam->GetSelectAllGroup();
if(theSelectAllGroup != this->mClientSelectAllGroup)
{
Selection selection;
selection.group_number = kSelectAllHotGroup;
selection.selected_entities = theSelectAllGroup;
NetMsg_SetSelect( this->pev, selection );
this->mClientSelectAllGroup = theSelectAllGroup;
}
// Check idle soldiers, ammo requests and health requests
AvHMessageID theRequestList[kNumRequestTypes] = {COMMANDER_NEXTIDLE, COMMANDER_NEXTAMMO, COMMANDER_NEXTHEALTH};
for(int i = 0; i < kNumRequestTypes; i++)
{
AvHMessageID theCurrentRequestType = theRequestList[i];
int theNumClientRequests = this->mClientRequests[i];
int theNumTeamRequests = theTeam->GetAlerts(theCurrentRequestType).size();
if(theNumClientRequests != theNumTeamRequests)
{
NetMsg_SetRequest( this->pev, theCurrentRequestType, theNumTeamRequests );
this->mClientRequests[i] = theNumTeamRequests;
}
}
}
}
void AvHPlayer::UpdateSoundNames()
{
if(this->pev != NULL ) // Not fully connected yet
{
// Send list of special sounds
const StringList& theSoundNameList = AvHMP3Audio::GetSoundNameList();
int theNumberOfSounds = theSoundNameList.size();
int theNumberOfSoundsOnClient = this->mClientSoundNames.size();
ASSERT(theNumberOfSoundsOnClient <= theNumberOfSounds);
// Only send one new sound name every tick, to avoid sending too much data and overflowing too quickly
if(theNumberOfSounds > theNumberOfSoundsOnClient)
{
const char* theSoundNameToSend = theSoundNameList[theNumberOfSoundsOnClient].c_str();
if(GetGameRules()->GetIsTesting())
{
this->SendMessage(theSoundNameToSend);
}
ASSERT( strlen(theSoundNameToSend) < 50);
NetMsg_SetSoundNames( this->pev, false, theSoundNameToSend);
this->mClientSoundNames.push_back(theSoundNameToSend);
}
}
}
//TODO: (KGP) there are a lot of expensive per-frame operations here that can be eliminated through careful refactoring.
// 1) make AvHTechTree an abstract interface
// 2) create base case using current AvHTechTree code
// 3) create filter around AvHTechTree that uses override of IsResearchable by MessageID and returns AvHTechNode objects
// that reflect the filter.
// 4) create AvHTechChangeListener class and use it as basis for decision to send tech nodes
// 5) create NetMsg_SetTechNodeDelta function that bundles state changes for multiple nodes into a single call
// 6) always use a personal copy of AvHTechNodes interface for each player to eliminate the per-frame copy of the team nodes
// 7) use filter class for NS mode aliens and update state of the filter when alien lifeform changes instead of using per-frame update
// Combined, these changes should reduce CPU overhead for tech node update by at least 90%.
void AvHPlayer::UpdateTechNodes()
{
bool theIsCombatMode = GetGameRules()->GetIsCombatMode();
bool theIsNSMode = GetGameRules()->GetIsNSMode();
if((this->GetUser3() == AVH_USER3_COMMANDER_PLAYER) || theIsCombatMode || this->GetIsAlien())
{
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam)
{
// Propagate and use local tech nodes in combat mode, else use team nodes in NS mode
AvHTechTree& theTechNodes = theIsCombatMode ? this->mCombatNodes : theTeam->GetTechNodes();
// Now customize nodes for aliens in NS
if(theIsNSMode && this->GetIsAlien())
{
// Set current lifeform to be unavailable
AvHMessageID theLifeform = MESSAGE_NULL;
switch(this->GetUser3())
{
case AVH_USER3_ALIEN_PLAYER1:
theLifeform = ALIEN_LIFEFORM_ONE;
break;
case AVH_USER3_ALIEN_PLAYER2:
theLifeform = ALIEN_LIFEFORM_TWO;
break;
case AVH_USER3_ALIEN_PLAYER3:
theLifeform = ALIEN_LIFEFORM_THREE;
break;
case AVH_USER3_ALIEN_PLAYER4:
theLifeform = ALIEN_LIFEFORM_FOUR;
break;
case AVH_USER3_ALIEN_PLAYER5:
theLifeform = ALIEN_LIFEFORM_FIVE;
break;
}
// tankefugl 0001075 : Reset techs to researchable before flagging them otherwise
theTechNodes.SetIsResearchable(ALIEN_LIFEFORM_ONE, true);
theTechNodes.SetIsResearchable(ALIEN_LIFEFORM_TWO, true);
theTechNodes.SetIsResearchable(ALIEN_LIFEFORM_THREE, true);
theTechNodes.SetIsResearchable(ALIEN_LIFEFORM_FOUR, true);
theTechNodes.SetIsResearchable(ALIEN_LIFEFORM_FIVE, true);
theTechNodes.SetIsResearchable(theLifeform, false);
// tankefugl 0001075 : Reset techs to researchable before flagging them otherwise
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_ONE, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_TWO, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_THREE, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_SEVEN, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_EIGHT, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_NINE, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_TEN, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_ELEVEN, true);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_TWELVE, true);
// If not Gorge, set buildables to be unavailable
if(theLifeform != ALIEN_LIFEFORM_TWO)
{
theTechNodes.SetIsResearchable(ALIEN_BUILD_HIVE, false);
theTechNodes.SetIsResearchable(ALIEN_BUILD_RESOURCES, false);
theTechNodes.SetIsResearchable(ALIEN_BUILD_OFFENSE_CHAMBER, false);
theTechNodes.SetIsResearchable(ALIEN_BUILD_DEFENSE_CHAMBER, false);
theTechNodes.SetIsResearchable(ALIEN_BUILD_MOVEMENT_CHAMBER, false);
theTechNodes.SetIsResearchable(ALIEN_BUILD_SENSORY_CHAMBER, false);
}
else
{
// tankefugl 0001075 : Reset techs to researchable before flagging them otherwise
theTechNodes.SetIsResearchable(ALIEN_BUILD_HIVE, true);
theTechNodes.SetIsResearchable(ALIEN_BUILD_RESOURCES, true);
theTechNodes.SetIsResearchable(ALIEN_BUILD_OFFENSE_CHAMBER, true);
theTechNodes.SetIsResearchable(ALIEN_BUILD_DEFENSE_CHAMBER, true);
theTechNodes.SetIsResearchable(ALIEN_BUILD_MOVEMENT_CHAMBER, true);
theTechNodes.SetIsResearchable(ALIEN_BUILD_SENSORY_CHAMBER, true);
// If we have the max hives, disable hives
bool theHasFreeUpgradeCategory = false;
MessageIDListType theUnsupportedUpgrades;
theUnsupportedUpgrades.push_back(ALIEN_BUILD_DEFENSE_CHAMBER);
theUnsupportedUpgrades.push_back(ALIEN_BUILD_MOVEMENT_CHAMBER);
theUnsupportedUpgrades.push_back(ALIEN_BUILD_SENSORY_CHAMBER);
FOR_ALL_ENTITIES(kesTeamHive, AvHHive*)
if(theEntity && theEntity->GetIsActive() && (theEntity->pev->team == this->pev->team))
{
AvHMessageID theTechnology = theEntity->GetTechnology();
if(theTechnology == MESSAGE_NULL)
{
theHasFreeUpgradeCategory = true;
break;
}
else
{
// It's supported, so remove from unsupported list
theUnsupportedUpgrades.erase(std::find(theUnsupportedUpgrades.begin(), theUnsupportedUpgrades.end(), theTechnology));
}
}
END_FOR_ALL_ENTITIES(kesTeamHive);
// If we don't have a free upgrade category
if(!theHasFreeUpgradeCategory)
{
// Remove every unsupported structure
for(MessageIDListType::iterator theIter = theUnsupportedUpgrades.begin(); theIter != theUnsupportedUpgrades.end(); theIter++)
{
theTechNodes.SetIsResearchable(*theIter, false);
}
}
}
// If there are no free upgrades available, disable upgrades
AvHTeam* theTeamPointer = this->GetTeamPointer();
if(theTeamPointer)
{
AvHAlienUpgradeListType theUpgrades = theTeamPointer->GetAlienUpgrades();
if(!AvHGetHasFreeUpgradeCategory(ALIEN_UPGRADE_CATEGORY_DEFENSE, theUpgrades, this->pev->iuser4))
{
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_ONE, false);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_TWO, false);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_THREE, false);
}
if(!AvHGetHasFreeUpgradeCategory(ALIEN_UPGRADE_CATEGORY_MOVEMENT, theUpgrades, this->pev->iuser4))
{
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_SEVEN, false);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_EIGHT, false);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_NINE, false);
}
if(!AvHGetHasFreeUpgradeCategory(ALIEN_UPGRADE_CATEGORY_SENSORY, theUpgrades, this->pev->iuser4))
{
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_TEN, false);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_ELEVEN, false);
theTechNodes.SetIsResearchable(ALIEN_EVOLUTION_TWELVE, false);
}
}
}
theTechNodes.GetDelta( this->mClientTechNodes,this->mClientTechDelta );
if( !mClientTechDelta.empty() )
{
const AvHTechNode* Node = NULL;
MessageIDListType::iterator current, end = mClientTechDelta.end();
for( current = mClientTechDelta.begin(); current != end; ++current )
{
Node = theTechNodes.GetNode(*current);
if( Node != NULL )
{
NetMsg_SetTechNode( this->pev, Node );
this->mClientTechNodes.InsertNode( Node );
}
else
{
//TODO: send signal to remove the tech node from the client here...
this->mClientTechNodes.RemoveNode(*current);
}
}
mClientTechDelta.clear();
mClientTechNodes = theTechNodes;
}
// Propagate any tech slots that have changed
const AvHTechSlotListType& theTeamTechSlotList = theTeam->GetTechSlotManager().GetTechSlotList();
if(this->mClientTechSlotList != theTeamTechSlotList)
{
// Send any nodes that have changed
int theCurrentSlot = 0;
for(AvHTechSlotListType::const_iterator theIter = theTeamTechSlotList.begin(); theIter != theTeamTechSlotList.end(); theIter++, theCurrentSlot++)
{
bool theHasClientTechSlot = false;
AvHTechSlots theClientTechSlot;
if((signed)this->mClientTechSlotList.size() > theCurrentSlot)
{
theClientTechSlot = this->mClientTechSlotList[theCurrentSlot];
theHasClientTechSlot = true;
}
AvHTechSlots theServerTechSlot = theTeamTechSlotList[theCurrentSlot];
if(!theHasClientTechSlot || (theClientTechSlot != theServerTechSlot))
{
NetMsg_SetTechSlots( this->pev, theServerTechSlot );
}
}
this->mClientTechSlotList = theTeamTechSlotList;
}
}
}
}
void AvHPlayer::UpdateTopDownMode()
{
if((this->mClientInTopDownMode != this->mInTopDownMode) || (this->mSpecialPASOrigin != this->mClientSpecialPASOrigin))
{
vec3_t& angles = this->mInTopDownMode ? this->mSpecialPASOrigin : this->mAnglesBeforeTopDown;
float position[3] = { angles.x, angles.y, angles.z };
NetMsg_SetTopDown_Position( this->pev, this->mInTopDownMode, position );
this->mClientInTopDownMode = this->mInTopDownMode;
this->mClientSpecialPASOrigin = this->mSpecialPASOrigin;
}
// Send menu tech slots to commander
AvHTeam* theTeam = this->GetTeamPointer();
if(theTeam && this->mInTopDownMode)
{
int theMenuTechSlots = theTeam->GetMenuTechSlots();
if(theMenuTechSlots != this->mClientMenuTechSlots)
{
NetMsg_SetTopDown_TechSlots( this->pev, theMenuTechSlots );
this->mClientMenuTechSlots = theMenuTechSlots;
}
}
}
void AvHPlayer::UpdateExperienceLevelsSpent()
{
// If our spent level is different then our client's
if(this->mClientExperienceLevelsSpent != this->mExperienceLevelsSpent)
{
NetMsg_GameStatus_UnspentLevels( this->pev, kGameStatusUnspentLevels, GetGameRules()->GetMapMode(), this->mExperienceLevelsSpent );
this->mClientExperienceLevelsSpent = this->mExperienceLevelsSpent;
}
}
void AvHPlayer::UpdateEntityHierarchy()
{
AvHPlayer* player = this;
// If we're spectating, then use the minimap data for the player we're spectating.
AvHPlayer* spectatingPlayer = dynamic_cast<AvHPlayer*>(GetSpectatingEntity());
if (spectatingPlayer)
{
player = spectatingPlayer;
}
AvHTeam* theTeam = player->GetTeamPointer();
// Removed the check for gamestart. This ensures that the entity-hierarchy is propagated before
// game start.
//if((theTeam) && GetGameRules()->GetGameStarted())
if((theTeam)) // && GetGameRules()->GetGameStarted())
{
if (theTeam->GetTeamType() == AVH_CLASS_TYPE_MARINE ||
theTeam->GetTeamType() == AVH_CLASS_TYPE_ALIEN)
{
// Pass in previous version so it can optimize and only send diff
AvHEntityHierarchy& theEntityList = GetGameRules()->GetEntityHierarchy(player->GetTeam());
if(theEntityList.SendToNetworkStream(this->mClientEntityHierarchy, this->pev))
{
this->mClientEntityHierarchy = theEntityList;
}
}
}
}
void AvHPlayer::UpdateSpawnScreenFade()
{
if(this->mSendSpawnScreenFade)
{
Vector theFadeColor;
theFadeColor.x = 0;
theFadeColor.y = 0;
theFadeColor.z = 0;
UTIL_ScreenFade(this, theFadeColor, kSpawnInFadeTime, 0.0f, 255, FFADE_IN);
this->mSendSpawnScreenFade = false;
}
}
void AvHPlayer::UpdateDebugCSP()
{
bool theCSPChanged = memcmp(&this->mClientDebugCSPInfo, &this->mDebugCSPInfo, sizeof(weapon_data_t));
if(theCSPChanged || (this->mClientNextAttack != this->m_flNextAttack))
{
NetMsg_DebugCSP( this->pev, this->mDebugCSPInfo, this->m_flNextAttack );
memcpy(&this->mClientDebugCSPInfo, &this->mDebugCSPInfo, sizeof(weapon_data_t));
this->mClientNextAttack = this->m_flNextAttack;
}
}
void AvHPlayer::UpdateOverwatch()
{
// Update overwatch indicator
if(this->mClientInOverwatch != this->mInOverwatch)
{
// We are entering overwatch
if(this->mInOverwatch)
{
PLAYBACK_EVENT_FULL(0, this->edict(), gStartOverwatchEventID, 0, this->pev->origin, (float *)&g_vecZero, 0.0, 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
}
// We are leaving overwatch
else
{
PLAYBACK_EVENT_FULL(0, this->edict(), gEndOverwatchEventID, 0, this->pev->origin, (float *)&g_vecZero, 0.0, 0.0, /*theWeaponIndex*/ 0, 0, 0, 0 );
}
this->mClientInOverwatch = this->mInOverwatch;
}
}
bool AvHPlayer::GetCanUseWeapon() const
{
return GetIsAbleToAct() && pev->viewmodel;
}
// tankefugl: 0000953
// allows a player to join team only once each inCoolDownTime seconds
bool AvHPlayer::JoinTeamCooledDown(float inCoolDownTime) {
// UTIL_ClientPrintAll(HUD_PRINTTALK, UTIL_VarArgs("Enter: JoinTeamCooledDown(%f), gpGlobals->time = %f, this->mTimeLastJoinTeam = %f", inCoolDownTime, gpGlobals->time, this->mTimeLastJoinTeam));
if ((this->mTimeLastJoinTeam == -1) || (gpGlobals->time > this->mTimeLastJoinTeam + inCoolDownTime))
{
this->mTimeLastJoinTeam = gpGlobals->time;
return true;
}
else
return false;
}
// :tankefugl
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
// Nexus interface
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
//TODO: flesh this out with admin privileges, etc. once the UPP authorization interface has been expanded
bool AvHPlayer::GetIsAuthorized(AvHAuthAction inAction, int inParameter) const
{
switch( inAction )
{
case AUTH_ACTION_JOIN_TEAM:
{
AvHTeamNumber theTeam = (AvHTeamNumber)inParameter;
switch( theTeam )
{
case TEAM_IND: // ready room & spectator - game allows in all cases
case TEAM_SPECT:
return true;
default:
// check it's an active team
if( theTeam == GetGameRules()->GetTeamA()->GetTeamNumber() || theTeam == GetGameRules()->GetTeamB()->GetTeamNumber() )
{
// tankefugl: 0001042 -- allow switching of teams -- placeholder before Nexus
// if( GetGameRules()->GetCheatsEnabled() ) { return true; } // cheaters can switch
// if( !GetGameRules()->GetGameStarted() ) { return true; } // can switch teams before start
// if( this->GetHasBeenSpectator() ) { return false; } // spectators have seen everybody
// for(int counter = TEAM_ACTIVE_BEGIN; counter < TEAM_ACTIVE_END; counter++)
// {
// if( theTeam != counter && this->GetHasSeenTeam( (AvHTeamNumber)counter ) )
// { return false; } // we've seen another active team
// }
return true; // haven't seen another team, authorized to join
}
return false; // unknown/inactive team - never grant an unknown permission!
}
}
case AUTH_ACTION_ADJUST_BALANCE:
{
#ifndef BALANCE_ENABLED
return false;
#else
return this->GetIsMember(PLAYERAUTH_DEVELOPER);
#endif
}
default:
return false; // never grant an unknown permission!
}
}
#ifdef USE_OLDAUTH
bool AvHPlayer::GetIsMember(const AvHPlayerAuthentication inAuthGroup)
{
return (this->GetAuthenticationMask() & inAuthGroup);
}
#else
bool AvHPlayer::GetIsMember(const string& inAuthGroup) const
{
return false;
}
#endif
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
// BalanceChangeListener implementation and balance network code
//
// Balance is checked for changes at a set rate determined by the
// BALANCE_UPDATE_MAX_FREQUENCY const below. This prevents the
// balance system logic from bogging down the server if there are 32
// players and the entire system is reloaded. A maximum of one
// message will be sent with each check. Note that this system is
// much, much more efficient than the old check of the entire balance
// state every frame!
//
// Due to the setup of the balance system, the BalanceChanageListener
// functions will never be called for non-playtest compiles, so
// there is no need to gaurd with a playtest build #define. The
// call to UpdateBalanceVariables may benefit from an being #define'd
// out, but that function has very low overhead anyway.
//
// TODO: move this block (variables and logic) into a discrete class
// and associate that class with the player using an auto_ptr instead
// of embedding the information into the player class
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
const float BALANCE_UPDATE_MAX_FREQUENCY = 0.05; //maximum frequency at which checks occur
bool AvHPlayer::shouldNotify(const string& name, const BalanceValueType type) const
{
return true;
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
void AvHPlayer::balanceCleared(void) const
{
this->mBalanceRemovalList.clear();
this->mBalanceMapInts.clear();
this->mBalanceMapFloats.clear();
this->mBalanceMapStrings.clear();
NetMsg_BalanceVarClear( this->pev );
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
// INTEGER
void AvHPlayer::balanceValueInserted(const string& name, const int value) const
{
this->mBalanceRemovalList.erase(name); //in case we had previous signal for deletion
this->mBalanceMapInts.insert(BalanceIntCollection::value_type(name,value));
}
void AvHPlayer::balanceValueChanged(const string& name, const int old_value, const int new_value) const
{
this->mBalanceRemovalList.erase(name); //in case we had previous signal for deletion
this->mBalanceMapInts.insert(BalanceIntCollection::value_type(name,new_value));
}
void AvHPlayer::balanceValueRemoved(const string& name, const int old_value) const
{
this->mBalanceMapInts.erase(name); //in case we didn't send it yet
this->mBalanceRemovalList.insert(name);
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
// FLOAT
void AvHPlayer::balanceValueInserted(const string& name, const float value) const
{
this->mBalanceRemovalList.erase(name); //in case we had previous signal for deletion
this->mBalanceMapFloats.insert(BalanceFloatCollection::value_type(name,value));
}
void AvHPlayer::balanceValueChanged(const string& name, const float old_value, const float new_value) const
{
this->mBalanceRemovalList.erase(name); //in case we had previous signal for deletion
this->mBalanceMapFloats.insert(BalanceFloatCollection::value_type(name,new_value));
}
void AvHPlayer::balanceValueRemoved(const string& name, const float old_value) const
{
this->mBalanceMapFloats.erase(name); //in case we didn't send it yet
this->mBalanceRemovalList.insert(name);
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
// STRING
void AvHPlayer::balanceValueInserted(const string& name, const string& value) const
{
this->mBalanceRemovalList.erase(name); //in case we had previous signal for deletion
this->mBalanceMapStrings.insert(BalanceStringCollection::value_type(name,value));
}
void AvHPlayer::balanceValueChanged(const string& name, const string& old_value, const string& new_value) const
{
this->mBalanceRemovalList.erase(name); //in case we had previous signal for deletion
this->mBalanceMapStrings.insert(BalanceStringCollection::value_type(name,new_value));
}
void AvHPlayer::balanceValueRemoved(const string& name, const string& old_value) const
{
this->mBalanceMapStrings.erase(name); //in case we didn't send it yet
this->mBalanceRemovalList.insert(name);
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
void AvHPlayer::InitBalanceVariables(void)
{
//grab the entire current balance
BalanceValueContainer* container = BalanceValueContainerFactory::get(BalanceValueContainerFactory::getDefaultFilename());
this->mBalanceMapStrings = *container->getStringMap();
this->mBalanceMapInts = *container->getIntMap();
this->mBalanceMapFloats = *container->getFloatMap();
//clear the client in preparation to send everything again
//pev will be null if this is called during construction
if( this->pev ) { NetMsg_BalanceVarClear( this->pev ); }
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
void AvHPlayer::UpdateBalanceVariables(void)
{
if(mNextBalanceVarUpdate < gpGlobals->time)
{
//only send it if it can be used...
//CONSIDER: second security setting for read-only transfer
if(this->GetIsAuthorized(AUTH_ACTION_ADJUST_BALANCE,0))
{
//check number of changes we have to send
int total_changes = this->mBalanceRemovalList.size();
total_changes += this->mBalanceMapInts.size();
total_changes += this->mBalanceMapFloats.size();
total_changes += this->mBalanceMapStrings.size();
//if we have multiple changes and need to tell client they are a set
if( total_changes > 1 && !this->mBalanceMessagePending ) //flag multiple changes incoming
{
NetMsg_BalanceVarChangesPending( this->pev, true );
this->mBalanceMessagePending = true;
}
//else if we have no more changes -- check to see if we need to send end set signal
else if( total_changes == 0 )
{
if( this->mBalanceMessagePending )
{
NetMsg_BalanceVarChangesPending( this->pev, false );
this->mBalanceMessagePending = false;
}
}
// we have at least one change to make, possibly in a set
else if(!this->mBalanceRemovalList.empty())
{
set<string>::iterator item = this->mBalanceRemovalList.begin();
NetMsg_BalanceVarRemove( this->pev, *item );
this->mBalanceRemovalList.erase(item);
}
else if(!this->mBalanceMapInts.empty())
{
BalanceIntCollection::iterator item = this->mBalanceMapInts.begin();
NetMsg_BalanceVarInsertInt( this->pev, item->first, item->second );
this->mBalanceMapInts.erase(item);
}
else if(!this->mBalanceMapFloats.empty())
{
BalanceFloatCollection::iterator item = this->mBalanceMapFloats.begin();
NetMsg_BalanceVarInsertFloat( this->pev, item->first, item->second );
this->mBalanceMapFloats.erase(item);
}
else if(!this->mBalanceMapStrings.empty())
{
BalanceStringCollection::iterator item = this->mBalanceMapStrings.begin();
NetMsg_BalanceVarInsertString( this->pev, item->first, item->second );
this->mBalanceMapStrings.erase(item);
}
}
//update next check of balance message queue
mNextBalanceVarUpdate = gpGlobals->time + BALANCE_UPDATE_MAX_FREQUENCY;
}
}