diff --git a/releases/3.2.0/source/dlls/player.cpp b/releases/3.2.0/source/dlls/player.cpp index f3878a00..b7af908d 100644 --- a/releases/3.2.0/source/dlls/player.cpp +++ b/releases/3.2.0/source/dlls/player.cpp @@ -92,6 +92,7 @@ #include "game.h" #include "common/hltv.h" #include "mod/AvHNetworkMessages.h" +#include "util/MathUtil.h" // #define DUCKFIX @@ -1778,28 +1779,31 @@ void CBasePlayer::PlayerUse ( void ) UTIL_MakeVectors ( pev->v_angle );// so we know which way we are facing - while ((pObject = UTIL_FindEntityInSphere( pObject, pev->origin, PLAYER_SEARCH_RADIUS )) != NULL) + while ((pObject = UTIL_FindEntityInSphere( pObject, pev->origin, PLAYER_SEARCH_RADIUS*2 )) != NULL) { + float distance=VectorDistance(pObject->pev->origin, pev->origin); - if (pObject->ObjectCaps() & (FCAP_IMPULSE_USE | FCAP_CONTINUOUS_USE | FCAP_ONOFF_USE)) - { - // !!!PERFORMANCE- should this check be done on a per case basis AFTER we've determined that - // this object is actually usable? This dot is being done for every object within PLAYER_SEARCH_RADIUS - // when player hits the use key. How many objects can be in that area, anyway? (sjb) - vecLOS = (VecBModelOrigin( pObject->pev ) - (pev->origin + pev->view_ofs)); - - // This essentially moves the origin of the target to the corner nearest the player to test to see - // if it's "hull" is in the view cone - vecLOS = UTIL_ClampVectorToBox( vecLOS, pObject->pev->size * 0.5 ); - - flDot = DotProduct (vecLOS , gpGlobals->v_forward); - if (flDot > flMaxDot ) - {// only if the item is in front of the user - pClosest = pObject; - flMaxDot = flDot; -// ALERT( at_console, "%s : %f\n", STRING( pObject->pev->classname ), flDot ); + if ( distance < PLAYER_SEARCH_RADIUS || pObject->pev->iuser3 == AVH_USER3_HIVE ) { + if (pObject->ObjectCaps() & (FCAP_IMPULSE_USE | FCAP_CONTINUOUS_USE | FCAP_ONOFF_USE)) + { + // !!!PERFORMANCE- should this check be done on a per case basis AFTER we've determined that + // this object is actually usable? This dot is being done for every object within PLAYER_SEARCH_RADIUS + // when player hits the use key. How many objects can be in that area, anyway? (sjb) + vecLOS = (VecBModelOrigin( pObject->pev ) - (pev->origin + pev->view_ofs)); + + // This essentially moves the origin of the target to the corner nearest the player to test to see + // if it's "hull" is in the view cone + vecLOS = UTIL_ClampVectorToBox( vecLOS, pObject->pev->size * 0.5 ); + + flDot = DotProduct (vecLOS , gpGlobals->v_forward); + if (flDot > flMaxDot ) + {// only if the item is in front of the user + pClosest = pObject; + flMaxDot = flDot; + // ALERT( at_console, "%s : %f\n", STRING( pObject->pev->classname ), flDot ); + } + // ALERT( at_console, "%s : %f\n", STRING( pObject->pev->classname ), flDot ); } -// ALERT( at_console, "%s : %f\n", STRING( pObject->pev->classname ), flDot ); } } pObject = pClosest; diff --git a/releases/3.2.0/source/mod/AvHBaseBuildable.cpp b/releases/3.2.0/source/mod/AvHBaseBuildable.cpp index 57dabfa2..b3159a78 100644 --- a/releases/3.2.0/source/mod/AvHBaseBuildable.cpp +++ b/releases/3.2.0/source/mod/AvHBaseBuildable.cpp @@ -243,7 +243,7 @@ void AvHBaseBuildable::ConstructUse( CBaseEntity *pActivator, CBaseEntity *pCall // Ensure that buildings are never absolutely painful to create int theBuildTime = max(GetGameRules()->GetBuildTimeForMessageID(this->mMessageID), 1); - if(GetGameRules()->GetIsTesting() || GetGameRules()->GetCheatsEnabled()) + if((GetGameRules()->GetIsTesting() || GetGameRules()->GetCheatsEnabled()) && !GetGameRules()->GetIsCheatEnabled(kcSlowResearch)) { theBuildTime = 2; } diff --git a/releases/3.2.0/source/mod/AvHHive.cpp b/releases/3.2.0/source/mod/AvHHive.cpp index de2e1fd7..4d44dd9a 100644 --- a/releases/3.2.0/source/mod/AvHHive.cpp +++ b/releases/3.2.0/source/mod/AvHHive.cpp @@ -77,17 +77,22 @@ // - Post-crash checkin. Restored @Backup from around 4/16. Contains changes for last four weeks of development. // //=============================================================================== + #include "mod/AvHHive.h" #include "mod/AvHGamerules.h" #include "mod/AvHServerUtil.h" #include "mod/AvHSharedUtil.h" +#include "mod/AvHAlienAbilityConstants.h" #include "mod/AvHAlienEquipmentConstants.h" #include "mod/AvHHulls.h" +#include "mod/AvHMovementUtil.h" #include "mod/AvHSoundListManager.h" #include "mod/AvHServerVariables.h" #include "mod/AvHParticleConstants.h" #include "mod/AvHSpecials.h" #include "mod/AvHPlayerUpgrade.h" +#include "util/MathUtil.h" +#include extern AvHSoundListManager gSoundListManager; BOOL IsSpawnPointValid( CBaseEntity *pPlayer, CBaseEntity *pSpot ); @@ -108,6 +113,9 @@ AvHHive::AvHHive() : AvHBaseBuildable(TECH_HIVE, ALIEN_BUILD_HIVE, kesTeamHive, this->mTimeLastWoundSound = -1; this->mTechnology = MESSAGE_NULL; this->mEnergy = 0.0f; + this->mLastTimeScannedHives=-1.0f; + this->mTeleportHiveIndex=-1;; + } bool AvHHive::CanBecomeActive() const @@ -424,8 +432,6 @@ void AvHHive::ResetEntity(void) AvHReinforceable::ResetEntity(); AvHBaseBuildable::ResetEntity(); - SetUse(&AvHHive::ConstructUse); - this->ResetCloaking(); this->SetInactive(); @@ -541,6 +547,7 @@ bool AvHHive::StartSpawningForTeam(AvHTeamNumber inTeam, bool inForce) this->pev->nextthink = gpGlobals->time + kHiveAliveThinkInterval; SetThink(&AvHHive::HiveAliveThink); + SetUse(&AvHHive::TeleportUse); theSuccess = true; } @@ -574,7 +581,6 @@ void AvHHive::Spawn() this->ResetEntity(); - SetUse(&AvHHive::ConstructUse); } void AvHHive::SetActive() @@ -639,7 +645,7 @@ void AvHHive::SetInactive() // No longer built at all this->pev->fuser1 = 0.0f; SetThink(NULL); - + SetUse(NULL); // Stop looping UTIL_EmitAmbientSound(ENT(this->pev), this->pev->origin, kHiveAmbientSound, 1.0f, .5, SND_STOP, 100); @@ -897,3 +903,86 @@ void AvHHive::HiveTouch(CBaseEntity* inOther) AvHBaseBuildable::BuildableTouch(inOther); } + +void AvHHive::TeleportUse(CBaseEntity* inActivator, CBaseEntity* inCaller, USE_TYPE inUseType, float inValue) +{ + if ( this->GetIsSpawning() ) + return; + + const float kHiveScanInterval = 1.0f; + + AvHPlayer* thePlayer = dynamic_cast(inActivator); + + vector theHives; + + if(thePlayer && (thePlayer->pev->team == this->pev->team) && (thePlayer->GetUser3() != AVH_USER3_ALIEN_EMBRYO)) + { + if((this->mLastTimeScannedHives == -1) || (gpGlobals->time > (this->mLastTimeScannedHives + kHiveScanInterval))) + { + this->mTeleportHiveIndex = -1; + float theFarthestDistance = 0.0f; //sqrt((kMaxMapDimension*2)*(kMaxMapDimension*2)); + // 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) && theEntity != this ) + { + bool theHiveIsUnderAttack = GetGameRules()->GetIsEntityUnderAttack(theEntity->entindex()); + // allow teleport to any built hive, or unbuilt hives under attack. + if(!theEntity->GetIsSpawning() || ( theEntity->GetIsSpawning() && theHiveIsUnderAttack ) ) + { + theHives.push_back(theEntity->entindex()); + } + } + END_FOR_ALL_ENTITIES(kesTeamHive) + + this->mLastTimeScannedHives = gpGlobals->time; + } + + if ( theHives.size() > 0 ) { + int myIndex=this->entindex(); + for ( int i=0; i < theHives.size(); i++ ) { + int hiveIndex=theHives[i]; + if ( hiveIndex > myIndex ) { + this->mTeleportHiveIndex=hiveIndex; + break; + } + } + if ( this->mTeleportHiveIndex == -1 ) { + this->mTeleportHiveIndex=theHives[0]; + } + } + + // If we have a valid hive index, jump the player to it + if(this->mTeleportHiveIndex != -1) + { + // Play sound at this entity + EMIT_SOUND(this->edict(), CHAN_AUTO, kAlienSightOnSound, 1.0f, ATTN_NORM); + + // Move him to it! + AvHHive* theHive = NULL; + AvHSUGetEntityFromIndex(this->mTeleportHiveIndex, theHive); + if(theHive) + { + CBaseEntity* theSpawnEntity = GetGameRules()->GetRandomHiveSpawnPoint(thePlayer, theHive->pev->origin, theHive->GetMaxSpawnDistance()); + if(theSpawnEntity) + { + Vector theMinSize; + Vector theMaxSize; + thePlayer->GetSize(theMinSize, theMaxSize); + + int theOffset = AvHMUGetOriginOffsetForUser3(AvHUser3(thePlayer->pev->iuser3)); + Vector theOriginToSpawn = theSpawnEntity->pev->origin; + theOriginToSpawn.z += theOffset; + + if(AvHSUGetIsEnoughRoomForHull(theOriginToSpawn, AvHMUGetHull(false, thePlayer->pev->iuser3), thePlayer->edict())) + { + thePlayer->SetPosition(theOriginToSpawn); + thePlayer->pev->velocity = Vector(0, 0, 0); + + // Play teleport sound before and after + EMIT_SOUND(inActivator->edict(), CHAN_AUTO, kAlienSightOffSound, 1.0f, ATTN_NORM); + } + } + } + } + } +} diff --git a/releases/3.2.0/source/mod/AvHHive.h b/releases/3.2.0/source/mod/AvHHive.h index 11e8222c..d9c294f8 100644 --- a/releases/3.2.0/source/mod/AvHHive.h +++ b/releases/3.2.0/source/mod/AvHHive.h @@ -156,6 +156,8 @@ public: virtual void Spawn(); + void EXPORT TeleportUse(CBaseEntity* inActivator, CBaseEntity* inCaller, USE_TYPE inUseType, float inValue); + //virtual void UpdateReinforcements(); protected: @@ -179,6 +181,9 @@ private: float mTimeLastWoundSound; float mTimeOfNextUmbra; float mEnergy; + float mLastTimeScannedHives; + int mTeleportHiveIndex; + }; #endif \ No newline at end of file