2005-04-10 19:09:49 +00:00
//======== (C) Copyright 2002 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: AvHSharedUtil.cpp $
// $Date: 2002/11/12 22:39:25 $
//
//-------------------------------------------------------------------------------
// $Log: AvHSharedUtil.cpp,v $
// Revision 1.33 2002/11/12 22:39:25 Flayra
// - Logging changes for Psychostats compatibility
//
// Revision 1.32 2002/10/24 21:42:32 Flayra
// - Don't give waypoint to selected troops when scanning
//
// Revision 1.31 2002/10/19 21:19:49 Flayra
// - Debugging info for linux
//
// Revision 1.30 2002/10/16 01:07:26 Flayra
// - Added official sizes that HL supports (ugh)
// - Added utility function for drawing range of ghost building, but it's unused
//
// Revision 1.29 2002/10/03 19:07:40 Flayra
// - Hives can never be blocked by func_resources
//
// Revision 1.28 2002/09/25 20:50:58 Flayra
// - Allow small items to be built on entities (health can be dropped right on players)
//
// Revision 1.27 2002/09/23 22:31:40 Flayra
// - Updated team 0 colors so dictation can be read in readyroom
// - Draw range for prototype lab, observatory and sensory chamber
// - Alien building rings
// - Don't allow non-resource buildings built too close to func_resources
// - Added heavy armor and jetpacks
//
// Revision 1.26 2002/09/09 20:06:43 Flayra
// - New observatory artwork
//
// Revision 1.25 2002/08/31 18:01:03 Flayra
// - Work at VALVe
//
// Revision 1.24 2002/08/16 02:47:06 Flayra
// - Fixed bug where not all the entities were iterated through
// - Support for ring-drawing (but not for all entities)
//
// Revision 1.23 2002/08/09 00:51:11 Flayra
// - Cleaned up useless special casing of tracetangible, fixed some problems where buildings could be built on players
//
// Revision 1.22 2002/08/02 21:50:05 Flayra
// - Made more general, by detecting all entities, not just players
//
// Revision 1.21 2002/07/24 18:45:43 Flayra
// - Linux and scripting changes
//
// Revision 1.20 2002/07/23 17:26:49 Flayra
// - Siege don't require a nearby turret factory, only add built buildings for range detection, build problems on client (point contents failing on rough surfaces)
//
// Revision 1.19 2002/07/08 17:17:44 Flayra
// - Reworked colors, moved functions into server util
//
// Revision 1.18 2002/07/01 21:46:39 Flayra
// - Added support for generic building ranges, fixed morphing problems
//
// Revision 1.17 2002/06/25 18:18:00 Flayra
// - Renamed buildings, better is-area-free detection
//
// Revision 1.16 2002/06/03 16:58:13 Flayra
// - Renamed weapons factory and armory, hive is now a buildable
//
// Revision 1.15 2002/05/28 18:13:43 Flayra
// - Changed "entity blocked by whatever" message to be logged for PR/screenshot purposes
//
// Revision 1.14 2002/05/23 02:33:20 Flayra
// - Post-crash checkin. Restored @Backup from around 4/16. Contains changes for last four weeks of development.
//
//===============================================================================
# ifdef AVH_SERVER
# include "dlls/extdll.h"
# include "dlls/util.h"
# include "types.h"
# endif
# ifdef AVH_CLIENT
# include "cl_dll/hud.h"
# include "cl_dll/cl_util.h"
# endif
# include "mod/AvHSharedUtil.h"
# include "mod/AvHSelectionHelper.h"
# include "mod/AvHConstants.h"
# ifdef AVH_SERVER
# include "mod/AvHPlayer.h"
# include "mod/AvHServerUtil.h"
# include "mod/AvHEntities.h"
# include "mod/AvHWeldable.h"
# include "mod/AvHGamerules.h"
# include "dlls/cfuncwall.h"
//#include "common/com_model.h"
//int NS_PointContents(const hull_t *hull, int num, float p[3]);
# endif
# include "pm_shared/pm_defs.h"
# include "pm_shared/pm_shared.h"
# ifdef AVH_CLIENT
# include "pm_shared/pm_debug.h"
//extern DebugPointListType gTriDebugLocations;
//extern DebugPointListType gSquareDebugLocations;
# endif
# include "util/MathUtil.h"
# include "util/STLUtil.h"
# include "common/vector_util.h"
# ifdef AVH_CLIENT
# include "cl_dll/eventscripts.h"
# include "common/r_efx.h"
# include "common/event_api.h"
# include "common/event_args.h"
# include "cl_dll/in_defs.h"
# endif
# include "common/com_model.h"
# include "mod/AvHSpecials.h"
# include "mod/AvHMarineEquipmentConstants.h"
# include "dlls/turretconst.h"
# include "mod/AvHMarineWeaponConstants.h"
# include "mod/AvHHulls.h"
# include "mod/AvHAlienEquipmentConstants.h"
# include "mod/CollisionUtil.h"
/*
bool NS_BoxesOverlap ( float origin1 [ 3 ] , float size1 [ 3 ] , float origin2 [ 3 ] , float size2 [ 3 ] ) ;
int NS_PointContents ( const hull_t * hull , int num , float p [ 3 ] ) ;
int NS_BoxContents ( const hull_t * hull , int num , float mins [ 3 ] , float maxs [ 3 ] ) ;
int NS_GetValveHull ( int inHull ) ;
void NS_TraceLine ( const hull_t * hull , float srcPoint [ 3 ] , float dstPoint [ 3 ] , trace_t * trace ) ;
*/
# include <time.h>
extern playermove_t * pmove ;
vec3_t AvHSHUGetRealLocation ( const vec3_t & inLocation , const vec3_t & inMinBox , const vec3_t & inMaxBox ) ;
vec3_t gPMDebugPoint ;
int kTeamColors [ iNumberOfTeamColors ] [ 3 ] =
{
{ 255 , 255 , 255 } , // White (default)
// { 125, 165, 210 }, // Blue (marine 1)
{ 0 , 153 , 255 } , // Blue (marine 1)
// { 255, 170, 0 }, // HL orange (alien 1)
{ 255 , 160 , 0 } , // HL orange (alien 1)
{ 145 , 215 , 140 } , // Green (marine 2)
{ 200 , 90 , 70 } , // Red (alien 2)
{ 255 , 255 , 255 } // White (spectator)
} ;
float kFTeamColors [ iNumberOfTeamColors ] [ 3 ] =
{
{ kTeamColors [ 0 ] [ 0 ] / 255.0f , kTeamColors [ 0 ] [ 1 ] / 255.0f , kTeamColors [ 0 ] [ 2 ] / 255.0f } ,
{ kTeamColors [ 1 ] [ 0 ] / 255.0f , kTeamColors [ 1 ] [ 1 ] / 255.0f , kTeamColors [ 1 ] [ 2 ] / 255.0f } ,
{ kTeamColors [ 2 ] [ 0 ] / 255.0f , kTeamColors [ 2 ] [ 1 ] / 255.0f , kTeamColors [ 2 ] [ 2 ] / 255.0f } ,
{ kTeamColors [ 3 ] [ 0 ] / 255.0f , kTeamColors [ 3 ] [ 1 ] / 255.0f , kTeamColors [ 3 ] [ 2 ] / 255.0f } ,
{ kTeamColors [ 4 ] [ 0 ] / 255.0f , kTeamColors [ 4 ] [ 1 ] / 255.0f , kTeamColors [ 4 ] [ 2 ] / 255.0f } ,
{ kTeamColors [ 5 ] [ 0 ] / 255.0f , kTeamColors [ 5 ] [ 1 ] / 255.0f , kTeamColors [ 5 ] [ 2 ] / 255.0f }
} ;
// Official allowed sizes
// {0, 0, 0 } { 0, 0, 0 } 0x0x0
// { -16, -16, -18 } { 16, 16, 18 } 32x32x36
// { -16, -16, -36 } { 16, 16, 36 } 32x32x72
// { -32, -32, -32 } { 32, 32, 32 } 64x64x64
//#define kBuilding1MinSize Vector(-14.33, -14.84, 0.02)
////#define kBuilding1MaxSize Vector(21.61, 14.86, 66.9686)
//#define kBuilding1MaxSize Vector(14.33, 14.86, 66.9686)
//
//#define kBuilding2MinSize Vector(-25.0, -25.0, 0.02)
//#define kBuilding2MaxSize Vector(25.0, 25.0, 66.9686)
# define kResourceMinSize Vector(-16.0, -16.0, 0.0)
# define kResourceMaxSize Vector(16.0, 16.0, 66.9486)
// Tried 100, 110, still jitters. Shrink tower down a bit?
# define kAlienResourceMinSize Vector(-16.0, -16.0, 0.0)
# define kAlienResourceMaxSize Vector(16.0, 16.0, 80.7443)
//physent_t* AvHSUGetEntity(int inPhysIndex)
//{
// physent_t* theTarget = NULL;
// int i = 0;
//
// if(inPhysIndex > 0)
// {
// #ifdef AVH_CLIENT
// gEngfuncs.pEventAPI->EV_SetUpPlayerPrediction( false, true );
//
// // Store off the old count
// //gEngfuncs.pEventAPI->EV_PushPMStates();
//
// // Now add in all of the players.
// gEngfuncs.pEventAPI->EV_SetSolidPlayers (-1);
//
// for (i = 0; i < MAX_PHYSENTS; i++)
// //for (i = 0; i < pmove->numphysent; i++)
// {
// physent_t* thePotentialCandidate = gEngfuncs.pEventAPI->EV_GetPhysent(i);
// if ( thePotentialCandidate && (thePotentialCandidate->info == inPhysIndex))
// {
// theTarget = &pmove->physents[i];
// break;
// }
// }
//
// //gEngfuncs.pEventAPI->EV_PopPMStates();
// #endif
//
// // Check phys entities on server
// #ifdef AVH_SERVER
// for (i = 0; i < MAX_PHYSENTS; i++)
// //for (i = 0; i < pmove->numphysent; i++)
// {
// if ( pmove->physents[i].info == inPhysIndex )
// {
// theTarget = &pmove->physents[i];
// break;
// }
// }
//
// //CBaseEntity* theEntityHit = CBaseEntity::Instance(tr.pHit);
// #endif
// }
//
// return theTarget;
//}
const char * AvHSHUGetClassNameFromUser3 ( AvHUser3 inUser3 )
{
const char * theClassName = " " ;
switch ( inUser3 )
{
case AVH_USER3_MARINE_PLAYER :
theClassName = " soldier " ;
break ;
case AVH_USER3_COMMANDER_PLAYER :
theClassName = " commander " ;
break ;
case AVH_USER3_ALIEN_PLAYER1 :
theClassName = " skulk " ;
break ;
case AVH_USER3_ALIEN_PLAYER2 :
theClassName = " gorge " ;
break ;
case AVH_USER3_ALIEN_PLAYER3 :
theClassName = " lerk " ;
break ;
case AVH_USER3_ALIEN_PLAYER4 :
theClassName = " fade " ;
break ;
case AVH_USER3_ALIEN_PLAYER5 :
theClassName = " onos " ;
break ;
case AVH_USER3_ALIEN_EMBRYO :
theClassName = " gestate " ;
break ;
}
return theClassName ;
}
// Pass in -1 to get all entities with a non-zero iuser3
void AvHSHUGetEntities ( int inUser3 , EntityListType & outEntities )
{
# ifdef AVH_SERVER
FOR_ALL_BASEENTITIES ( ) ;
if ( ( theBaseEntity - > pev - > iuser3 = = inUser3 ) | | ( ( inUser3 = = - 1 ) & & ( theBaseEntity - > pev - > iuser3 > 0 ) ) )
{
outEntities . push_back ( theBaseEntity - > entindex ( ) ) ;
}
END_FOR_ALL_BASEENTITIES ( ) ;
# endif
# ifdef AVH_CLIENT
gEngfuncs . pEventAPI - > EV_SetUpPlayerPrediction ( false , true ) ;
// Store off the old count
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
// Now add in all of the players.
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
physent_t * theEntity = NULL ;
int theNumEnts = pmove - > numphysent ;
for ( int i = 0 ; i < theNumEnts ; i + + )
{
theEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( i ) ;
if ( ( theEntity - > iuser3 = = inUser3 ) | | ( ( inUser3 = = - 1 ) & & ( theEntity - > iuser3 > 0 ) ) )
{
outEntities . push_back ( i ) ;
}
}
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
# endif
}
bool AvHSHUGetEntityLocation ( int inEntity , vec3_t & outLocation )
{
bool theSuccess = false ;
# ifdef AVH_SERVER
edict_t * pEdict = g_engfuncs . pfnPEntityOfEntIndex ( inEntity ) ;
if ( pEdict & & ! pEdict - > free )
{
CBaseEntity * theEntity = CBaseEntity : : Instance ( pEdict ) ;
if ( theEntity )
{
outLocation = AvHSHUGetRealLocation ( theEntity - > pev - > origin , theEntity - > pev - > mins , theEntity - > pev - > maxs ) ;
theSuccess = true ;
}
}
# endif
# ifdef AVH_CLIENT
gEngfuncs . pEventAPI - > EV_SetUpPlayerPrediction ( false , true ) ;
// Store off the old count
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
// Now add in all of the players.
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
physent_t * theEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( inEntity ) ;
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
if ( theEntity )
{
outLocation = AvHSHUGetRealLocation ( theEntity - > origin , theEntity - > mins , theEntity - > maxs ) ;
theSuccess = true ;
}
# endif
return theSuccess ;
}
bool AvHSHUGetEntityIUser3 ( int inEntity , AvHUser3 & outIUser3 )
{
bool theSuccess = false ;
# ifdef AVH_SERVER
edict_t * pEdict = g_engfuncs . pfnPEntityOfEntIndex ( inEntity ) ;
if ( pEdict & & ! pEdict - > free )
{
CBaseEntity * theEntity = CBaseEntity : : Instance ( pEdict ) ;
if ( theEntity )
{
outIUser3 = ( AvHUser3 ) theEntity - > pev - > iuser3 ;
theSuccess = true ;
}
}
# endif
# ifdef AVH_CLIENT
// Store off the old count
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
// Now add in all of the players.
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
physent_t * theEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( inEntity ) ;
if ( theEntity )
{
outIUser3 = ( AvHUser3 ) theEntity - > iuser3 ;
theSuccess = true ;
}
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
# endif
return theSuccess ;
}
bool AvHSHUGetEntityIUser4 ( int inEntity , int & outIUser4 )
{
bool theSuccess = false ;
# ifdef AVH_SERVER
edict_t * pEdict = g_engfuncs . pfnPEntityOfEntIndex ( inEntity ) ;
if ( pEdict & & ! pEdict - > free )
{
CBaseEntity * theEntity = CBaseEntity : : Instance ( pEdict ) ;
if ( theEntity )
{
outIUser4 = theEntity - > pev - > iuser4 ;
theSuccess = true ;
}
}
# endif
# ifdef AVH_CLIENT
physent_t * theEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( inEntity ) ;
if ( theEntity )
{
outIUser4 = theEntity - > iuser4 ;
theSuccess = true ;
}
# endif
return theSuccess ;
}
vec3_t AvHSHUGetRealLocation ( const vec3_t & inLocation , const vec3_t & inMinBox , const vec3_t & inMaxBox )
{
vec3_t outLocation ;
VectorCopy ( inLocation , outLocation ) ;
if ( ( outLocation . x = = outLocation . y ) & & ( outLocation . y = = outLocation . z ) & & ( outLocation . z = = 0.0f ) )
{
outLocation . x = ( inMinBox . x + inMaxBox . x ) / 2.0f ;
outLocation . y = ( inMinBox . y + inMaxBox . y ) / 2.0f ;
outLocation . z = ( inMinBox . z + inMaxBox . z ) / 2.0f ;
}
return outLocation ;
}
// Returns range of user3, whatever that means. Returns -1 for no meaningful range.
int AvHSHUGetDrawRangeForUser3 ( AvHUser3 inUser3 )
{
int theRange = - 1 ;
switch ( inUser3 )
{
case AVH_USER3_COMMANDER_STATION :
theRange = BALANCE_VAR ( kCommandStationBuildDistance ) ;
break ;
case AVH_USER3_FUNC_RESOURCE :
theRange = BALANCE_VAR ( kResourceTowerBuildDistanceTolerance ) ;
break ;
case AVH_USER3_TURRET_FACTORY :
case AVH_USER3_ADVANCED_TURRET_FACTORY :
theRange = BALANCE_VAR ( kTurretFactoryBuildDistance ) ;
break ;
case AVH_USER3_ARMORY :
case AVH_USER3_ADVANCED_ARMORY :
theRange = BALANCE_VAR ( kArmoryBuildDistance ) ;
break ;
case AVH_USER3_TURRET :
theRange = BALANCE_VAR ( kTurretRange ) ;
break ;
case AVH_USER3_SIEGETURRET :
// TODO: Figure out a way to return minimum range also?
theRange = BALANCE_VAR ( kSiegeTurretRange ) ;
break ;
case AVH_USER3_PROTOTYPE_LAB :
theRange = BALANCE_VAR ( kArmorDropRange ) ;
break ;
case AVH_USER3_OBSERVATORY :
theRange = BALANCE_VAR ( kObservatoryXYDetectionRadius ) ;
break ;
case AVH_USER3_SENSORY_CHAMBER :
theRange = BALANCE_VAR ( kSensoryChamberRange ) ;
break ;
}
return theRange ;
}
int AvHSHUGetDrawRangeForMessageID ( AvHMessageID inMessageID )
{
AvHUser3 theUser3 = AVH_USER3_NONE ;
switch ( inMessageID )
{
case BUILD_COMMANDSTATION :
theUser3 = AVH_USER3_COMMANDER_STATION ;
break ;
case BUILD_TURRET :
theUser3 = AVH_USER3_TURRET ;
break ;
case BUILD_SIEGE :
theUser3 = AVH_USER3_SIEGETURRET ;
break ;
case BUILD_OBSERVATORY :
theUser3 = AVH_USER3_OBSERVATORY ;
break ;
case BUILD_TURRET_FACTORY :
theUser3 = AVH_USER3_TURRET_FACTORY ;
break ;
case BUILD_PROTOTYPE_LAB :
theUser3 = AVH_USER3_PROTOTYPE_LAB ;
break ;
case BUILD_ARMORY :
theUser3 = AVH_USER3_ARMORY ;
break ;
}
return AvHSHUGetDrawRangeForUser3 ( theUser3 ) ;
}
// Specify whether we draw rings for this entity or not, and draw them at different sizes for aesthetics. The scalar might go away when the
// artwork all has the correct bounding boxes.
bool AvHSHUGetDrawRingsForUser3 ( AvHUser3 inUser3 , float & outScalar )
{
bool theDrawRings = false ;
float theScalar = 1.0f ;
switch ( inUser3 )
{
case AVH_USER3_MARINE_PLAYER :
theScalar = .9f ;
theDrawRings = true ;
break ;
case AVH_USER3_RESTOWER :
theScalar = 1.6f ;
theDrawRings = true ;
break ;
case AVH_USER3_INFANTRYPORTAL :
2005-04-12 00:03:30 +00:00
theScalar = 1.0f ;
2005-04-10 19:09:49 +00:00
theDrawRings = true ;
break ;
case AVH_USER3_ARMORY :
case AVH_USER3_ADVANCED_ARMORY :
theScalar = 1.35f ;
theDrawRings = true ;
break ;
case AVH_USER3_COMMANDER_STATION :
theScalar = 1.4f ;
theDrawRings = true ;
break ;
case AVH_USER3_PHASEGATE :
theScalar = 2.0f ;
theDrawRings = true ;
break ;
case AVH_USER3_OBSERVATORY :
theScalar = .9f ;
theDrawRings = true ;
break ;
case AVH_USER3_TURRET_FACTORY :
case AVH_USER3_ADVANCED_TURRET_FACTORY :
theScalar = 1.6f ;
theDrawRings = true ;
break ;
case AVH_USER3_TURRET :
theScalar = .7f ;
theDrawRings = true ;
break ;
case AVH_USER3_NUKEPLANT :
theScalar = 1.5f ;
theDrawRings = true ;
break ;
case AVH_USER3_ARMSLAB :
case AVH_USER3_PROTOTYPE_LAB :
case AVH_USER3_CHEMLAB :
case AVH_USER3_MEDLAB :
case AVH_USER3_SIEGETURRET :
theScalar = 1.3f ;
theDrawRings = true ;
break ;
// Alien buildings
case AVH_USER3_DEFENSE_CHAMBER :
case AVH_USER3_OFFENSE_CHAMBER :
case AVH_USER3_MOVEMENT_CHAMBER :
case AVH_USER3_SENSORY_CHAMBER :
case AVH_USER3_ALIENRESTOWER :
theScalar = 1.3f ;
theDrawRings = true ;
break ;
case AVH_USER3_ALIEN_PLAYER1 :
theScalar = .6f ;
theDrawRings = true ;
break ;
case AVH_USER3_ALIEN_PLAYER2 :
theScalar = 1.0f ;
theDrawRings = true ;
break ;
case AVH_USER3_ALIEN_PLAYER3 :
theScalar = 1.0 ;
theDrawRings = true ;
break ;
case AVH_USER3_ALIEN_PLAYER4 :
theScalar = 1.3f ;
theDrawRings = true ;
break ;
case AVH_USER3_ALIEN_PLAYER5 :
theScalar = 1.2f ;
theDrawRings = true ;
break ;
}
if ( theDrawRings )
{
outScalar = theScalar ;
}
return theDrawRings ;
}
bool AvHSHUGetBuildRegions ( AvHMessageID inMessageID , EntityListType & outEntities , IntList & outRanges , float & outZAdjustment , bool & outSnapToLocation )
{
bool theFoundTech = false ;
typedef vector < AvHUser3 > User3ListType ;
User3ListType theUser3s ;
outZAdjustment = 0.0f ;
outSnapToLocation = false ;
switch ( inMessageID )
{
case BUILD_RESOURCES :
case ALIEN_BUILD_RESOURCES :
theUser3s . push_back ( AVH_USER3_FUNC_RESOURCE ) ;
outZAdjustment = ( kFuncResourceMaxSize . z - kFuncResourceMinSize . z ) ;
outSnapToLocation = true ;
break ;
case BUILD_INFANTRYPORTAL :
theUser3s . push_back ( AVH_USER3_COMMANDER_STATION ) ;
break ;
case BUILD_TURRET :
theUser3s . push_back ( AVH_USER3_TURRET_FACTORY ) ;
theUser3s . push_back ( AVH_USER3_ADVANCED_TURRET_FACTORY ) ;
break ;
case BUILD_SIEGE :
theUser3s . push_back ( AVH_USER3_ADVANCED_TURRET_FACTORY ) ;
break ;
case BUILD_MINES :
case BUILD_WELDER :
case BUILD_SHOTGUN :
theUser3s . push_back ( AVH_USER3_ARMORY ) ;
theUser3s . push_back ( AVH_USER3_ADVANCED_ARMORY ) ;
break ;
case BUILD_HMG :
case BUILD_GRENADE_GUN :
theUser3s . push_back ( AVH_USER3_ADVANCED_ARMORY ) ;
break ;
case BUILD_HEAVY :
case BUILD_JETPACK :
theUser3s . push_back ( AVH_USER3_PROTOTYPE_LAB ) ;
break ;
}
if ( theUser3s . size ( ) > 0 )
{
for ( User3ListType : : iterator theUser3Iter = theUser3s . begin ( ) ; theUser3Iter ! = theUser3s . end ( ) ; theUser3Iter + + )
{
EntityListType theEntities ;
AvHSHUGetEntities ( * theUser3Iter , theEntities ) ;
for ( EntityListType : : iterator theEntityIter = theEntities . begin ( ) ; theEntityIter ! = theEntities . end ( ) ; theEntityIter + + )
{
// Only add buildings that are fully built
int theIuser4 = 0 ;
if ( AvHSHUGetEntityIUser4 ( * theEntityIter , theIuser4 ) )
{
if ( ! GetHasUpgrade ( theIuser4 , MASK_BUILDABLE ) )
{
outEntities . push_back ( * theEntityIter ) ;
outRanges . push_back ( AvHSHUGetDrawRangeForUser3 ( * theUser3Iter ) ) ;
}
}
}
}
theFoundTech = true ;
}
return theFoundTech ;
}
// tankefugl: 0000291 -- allows listed structures to be dropped on resource towers
bool AvHSHUGetIsDroppableOnRTs ( AvHMessageID inMessageID )
{
switch ( inMessageID )
{
case BUILD_HEALTH :
case BUILD_AMMO :
case BUILD_MINES :
case BUILD_WELDER :
case BUILD_SHOTGUN :
case BUILD_HMG :
case BUILD_GRENADE_GUN :
case BUILD_CAT :
case BUILD_HEAVY :
case BUILD_JETPACK :
case BUILD_RESOURCES :
case ALIEN_BUILD_RESOURCES :
case ALIEN_BUILD_HIVE :
return true ;
default :
return false ;
}
}
// :tankefugl
bool AvHSHUGetIsMarineStructure ( AvHMessageID inMessageID )
{
switch ( inMessageID )
{
case BUILD_INFANTRYPORTAL :
case BUILD_RESOURCES :
case BUILD_TURRET_FACTORY :
case BUILD_ARMSLAB :
case BUILD_PROTOTYPE_LAB :
case BUILD_ARMORY :
case BUILD_NUKE_PLANT :
case BUILD_OBSERVATORY :
case BUILD_PHASEGATE :
case BUILD_TURRET :
case BUILD_SIEGE :
case BUILD_COMMANDSTATION :
return true ;
default :
return false ;
}
}
bool AvHSHUGetIsMarineStructure ( AvHUser3 inUser3 )
{
switch ( inUser3 )
{
case AVH_USER3_COMMANDER_STATION :
case AVH_USER3_TURRET_FACTORY :
case AVH_USER3_ARMORY :
case AVH_USER3_ADVANCED_ARMORY :
case AVH_USER3_ARMSLAB :
case AVH_USER3_PROTOTYPE_LAB :
case AVH_USER3_OBSERVATORY :
case AVH_USER3_CHEMLAB :
case AVH_USER3_MEDLAB :
case AVH_USER3_NUKEPLANT :
case AVH_USER3_TURRET :
case AVH_USER3_SIEGETURRET :
case AVH_USER3_RESTOWER :
case AVH_USER3_INFANTRYPORTAL :
case AVH_USER3_PHASEGATE :
case AVH_USER3_ADVANCED_TURRET_FACTORY :
return true ;
default :
return false ;
}
}
void AvHSHUGetMinBuildRadiusViolations ( AvHMessageID inMessageID , vec3_t & inLocation , EntityListType & outViolations )
{
// Enforce a minimum build radius for marine structures.
if ( AvHSHUGetIsMarineStructure ( inMessageID ) )
{
EntityListType theEntities ;
AvHSHUGetEntities ( - 1 , theEntities ) ;
vec3_t theMinSize , theMaxSize ;
AvHSHUGetSizeForTech ( inMessageID , theMinSize , theMaxSize ) ;
float theMaxRadius1 = max ( - min ( theMinSize . x , theMinSize . y ) , max ( theMaxSize . x , theMaxSize . y ) ) ;
for ( EntityListType : : iterator theIter = theEntities . begin ( ) ; theIter ! = theEntities . end ( ) ; theIter + + )
{
AvHUser3 theUser3 ;
AvHSHUGetEntityIUser3 ( * theIter , theUser3 ) ;
bool theEntityCouldBlock = AvHSHUGetIsMarineStructure ( theUser3 ) ;
if ( inMessageID ! = BUILD_RESOURCES )
{
theEntityCouldBlock | = ( theUser3 = = AVH_USER3_FUNC_RESOURCE ) ;
}
bool theEntityIsSolid = false ;
# ifdef AVH_SERVER
edict_t * theEntity = INDEXENT ( * theIter ) ;
if ( theEntity )
theEntityIsSolid = ( theEntity - > v . solid = = SOLID_BBOX ) ;
# endif
# ifdef AVH_CLIENT
physent_t * theEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( * theIter ) ;
if ( theEntity )
theEntityIsSolid = ( theEntity - > solid = = SOLID_BBOX ) ;
# endif
// joev: 0000291
// It's possible to place "on" marines if you're offset a little from center. This code and
// associated changes below and in AvHHudRender.cpp is to enforce a build distance around players
// in the same way as buildings to prevent this exploit.
if ( theUser3 = = AVH_USER3_MARINE_PLAYER )
{
theEntityIsSolid = true ;
theEntityCouldBlock = true ;
}
if ( theEntityCouldBlock & & theEntityIsSolid )
{
AvHSHUGetSizeForUser3 ( theUser3 , theMinSize , theMaxSize ) ;
float theMaxRadius2 = max ( max ( theMinSize . x , theMaxSize . x ) , max ( theMinSize . y , theMaxSize . y ) ) ;
vec3_t theLocation ;
if ( AvHSHUGetEntityLocation ( * theIter , theLocation ) )
{
vec3_t theXYInLocation = inLocation ;
vec3_t theXYTheLocation = theLocation ;
theXYInLocation . z = 0 ;
theXYTheLocation . z = 0 ;
float theDistance = VectorDistance ( ( float * ) & theXYInLocation , ( float * ) & theXYTheLocation ) ;
// joev: 0000291
// It's possible to place "on" marines if you're offset a little from center. This code and
// associated changes above and in AvHHudRender.cpp is to enforce a build distance around players
// in the same way as buildings to prevent this exploit.
float theMinMarineBuildDistance ;
if ( theUser3 = = AVH_USER3_MARINE_PLAYER ) {
theMinMarineBuildDistance = BALANCE_VAR ( kMinMarinePlayerBuildDistance ) ;
}
else
{
theMinMarineBuildDistance = BALANCE_VAR ( kMinMarineBuildDistance ) ;
}
// :joev
if ( theDistance < theMinMarineBuildDistance + theMaxRadius1 + theMaxRadius2 )
{
outViolations . push_back ( * theIter ) ;
}
}
}
}
}
}
bool AvHSHUGetAreSpecialBuildingRequirementsMet ( AvHMessageID inMessageID , vec3_t & inLocation )
{
bool theRequirementsMet = false ;
EntityListType theEntities ;
IntList theDistanceRequirements ;
float theZAdjustment = 0.0f ;
bool theSnapToLocation = false ;
if ( inMessageID = = ALIEN_BUILD_HIVE )
{
// Look for inactive hives within radius
EntityListType theEntities ;
// Look for a unoccupied hive spot within range
AvHSHUGetEntities ( AVH_USER3_HIVE , theEntities ) ;
for ( EntityListType : : iterator theIter = theEntities . begin ( ) ; theIter ! = theEntities . end ( ) ; theIter + + )
{
vec3_t theLocation ;
if ( AvHSHUGetEntityLocation ( * theIter , theLocation ) )
{
// Set z's equal to check 2D distance only
inLocation . z = theLocation . z ;
float theDistance = VectorDistance ( ( float * ) & inLocation , ( float * ) & theLocation ) ;
if ( theDistance < = kHiveXYDistanceTolerance )
{
// Make sure this hive isn't already active
# ifdef AVH_SERVER
CBaseEntity * theEntity = AvHSUGetEntityFromIndex ( * theIter ) ;
ASSERT ( theEntity ) ;
if ( theEntity - > pev - > team = = 0 )
{
# endif
theRequirementsMet = true ;
inLocation = theLocation ;
# ifdef AVH_SERVER
}
# endif
break ;
}
}
}
}
else if ( AvHSHUGetBuildRegions ( inMessageID , theEntities , theDistanceRequirements , theZAdjustment , theSnapToLocation ) )
{
ASSERT ( theEntities . size ( ) = = theDistanceRequirements . size ( ) ) ;
vec3_t theBestLocation ;
int i = 0 ;
float theClosestDistance = kMaxMapDimension ;
for ( EntityListType : : iterator theIter = theEntities . begin ( ) ; theIter ! = theEntities . end ( ) ; theIter + + , i + + )
{
vec3_t theLocation ;
if ( AvHSHUGetEntityLocation ( * theIter , theLocation ) )
{
// Only check xy distance
vec3_t theXYInLocation = inLocation ;
vec3_t theXYTheLocation = theLocation ;
theXYInLocation . z = 0 ;
theXYTheLocation . z = 0 ;
float theDistance = VectorDistance ( ( float * ) & theXYInLocation , ( float * ) & theXYTheLocation ) ;
int theDistanceRequirement = theDistanceRequirements [ i ] ;
if ( ( theDistance < = theDistanceRequirement ) | | ( theDistanceRequirement = = - 1 ) )
{
// Pick the closest one, in case there are multiples in range
if ( theDistance < theClosestDistance )
{
theClosestDistance = theDistance ;
VectorCopy ( theLocation , theBestLocation ) ;
theRequirementsMet = true ;
}
}
}
}
if ( theRequirementsMet & & theSnapToLocation )
{
inLocation = theBestLocation ;
inLocation . z + = theZAdjustment ;
}
}
else
{
theRequirementsMet = true ;
}
EntityListType theBuildRadiusViolations ;
AvHSHUGetMinBuildRadiusViolations ( inMessageID , inLocation , theBuildRadiusViolations ) ;
if ( theBuildRadiusViolations . size ( ) > 0 )
{
theRequirementsMet = false ;
}
// Anti-llama/newbie tactic: don't allow non-resource buildings to be placed such that they block access to nozzles
// Make sure generic building isn't being placed on top of resource nozzles
// tankefugl: 0000291
// allow equipment, rts and hives to be dropped around nodes
if ( AvHSHUGetIsDroppableOnRTs ( inMessageID ) = = false )
{
// :tankefugl
// If building is too close to an empty nozzle, don't allow it
float theResourceBuildingRadius , theTotalMinRadius ;
vec3_t theMinSize , theMaxSize , theMinRadius , theFlattenedInLocation , theLocation ;
theFlattenedInLocation [ 0 ] = inLocation [ 0 ] ;
theFlattenedInLocation [ 1 ] = inLocation [ 1 ] ;
theFlattenedInLocation [ 2 ] = 0 ;
theResourceBuildingRadius = 60 ;
EntityListType theEntities ;
AvHSHUGetEntities ( AVH_USER3_FUNC_RESOURCE , theEntities ) ;
if ( AvHSHUGetSizeForTech ( inMessageID , theMinSize , theMaxSize ) )
{
EntityListType : : iterator end = theEntities . end ( ) ;
for ( EntityListType : : iterator current = theEntities . begin ( ) ; current < end ; + + current )
{
if ( AvHSHUGetEntityLocation ( * current , theLocation ) )
{
//flatten to 2 dimensions
theLocation [ 2 ] = 0 ;
//space = radius of both buildings combined
theTotalMinRadius = theResourceBuildingRadius ;
theTotalMinRadius + = max ( - min ( theMinSize . x , theMinSize . y ) , max ( theMaxSize . x , theMaxSize . y ) ) ;
if ( VectorDistance ( ( float * ) & theFlattenedInLocation , ( float * ) & theLocation ) < theTotalMinRadius )
{
theRequirementsMet = false ;
break ;
}
}
}
}
}
return theRequirementsMet ;
}
bool AvHSHUGetBuildTechClassName ( AvHMessageID inMessageID , char * & outClassName )
{
bool theSuccess = true ;
switch ( inMessageID )
{
// Buildings
case BUILD_RESOURCES :
outClassName = kwsResourceTower ;
break ;
//case BUILD_REINFORCEMENTS:
// outClassName = kwsInfantryPortal;
// break;
case BUILD_INFANTRYPORTAL :
outClassName = kwsInfantryPortal ;
break ;
case BUILD_COMMANDSTATION :
outClassName = kwsTeamCommand ;
break ;
case BUILD_TURRET_FACTORY :
outClassName = kwsTurretFactory ;
break ;
case BUILD_ARMSLAB :
outClassName = kwsArmsLab ;
break ;
case BUILD_PROTOTYPE_LAB :
outClassName = kwsPrototypeLab ;
break ;
case BUILD_ARMORY :
outClassName = kwsArmory ;
break ;
case ARMORY_UPGRADE :
outClassName = kwsAdvancedArmory ;
break ;
case BUILD_NUKE_PLANT :
outClassName = kwsNukePlant ;
break ;
case BUILD_OBSERVATORY :
outClassName = kwsObservatory ;
break ;
case BUILD_SCAN :
outClassName = kwsScan ;
break ;
case BUILD_PHASEGATE :
outClassName = kwsPhaseGate ;
break ;
case BUILD_TURRET :
outClassName = kwsDeployedTurret ;
break ;
case BUILD_SIEGE :
outClassName = kwsSiegeTurret ;
break ;
// Equipment
case BUILD_HEALTH :
outClassName = kwsHealth ;
break ;
case BUILD_CAT :
outClassName = kwsCatalyst ;
break ;
case BUILD_JETPACK :
outClassName = kwsJetpack ;
break ;
case BUILD_HEAVY :
outClassName = kwsHeavyArmor ;
break ;
case BUILD_AMMO :
outClassName = kwsGenericAmmo ;
break ;
case BUILD_WELDER :
outClassName = kwsWelder ;
break ;
case BUILD_MINES :
outClassName = kwsMine ;
break ;
case BUILD_SHOTGUN :
outClassName = kwsShotGun ;
break ;
case BUILD_HMG :
outClassName = kwsHeavyMachineGun ;
break ;
case BUILD_NUKE :
outClassName = kwsNuke ;
break ;
case BUILD_GRENADE_GUN :
outClassName = kwsGrenadeGun ;
break ;
//case BUILD_MEDKIT:
// break;
case ALIEN_BUILD_RESOURCES :
outClassName = kwsAlienResourceTower ;
break ;
case ALIEN_BUILD_OFFENSE_CHAMBER :
outClassName = kwsOffenseChamber ;
break ;
case ALIEN_BUILD_DEFENSE_CHAMBER :
outClassName = kwsDefenseChamber ;
break ;
case ALIEN_BUILD_SENSORY_CHAMBER :
outClassName = kwsSensoryChamber ;
break ;
case ALIEN_BUILD_MOVEMENT_CHAMBER :
outClassName = kwsMovementChamber ;
break ;
case ALIEN_BUILD_HIVE :
outClassName = kesTeamHive ;
break ;
default :
theSuccess = false ;
break ;
}
return theSuccess ;
}
bool AvHSHUGetResearchTechName ( AvHMessageID inResearchID , char * & outResearchTechName )
{
bool theSuccess = true ;
switch ( inResearchID )
{
case RESEARCH_ELECTRICAL :
outResearchTechName = " research_electrical " ;
break ;
case RESEARCH_ARMOR_ONE :
outResearchTechName = " research_armorl1 " ;
break ;
case RESEARCH_ARMOR_TWO :
outResearchTechName = " research_armorl2 " ;
break ;
case RESEARCH_ARMOR_THREE :
outResearchTechName = " research_armorl3 " ;
break ;
case RESEARCH_WEAPONS_ONE :
outResearchTechName = " research_weaponsl1 " ;
break ;
case RESEARCH_WEAPONS_TWO :
outResearchTechName = " research_weaponsl2 " ;
break ;
case RESEARCH_WEAPONS_THREE :
outResearchTechName = " research_weaponsl3 " ;
break ;
case ARMORY_UPGRADE :
outResearchTechName = " research_advarmory " ;
break ;
case TURRET_FACTORY_UPGRADE :
outResearchTechName = " research_advturretfactory " ;
break ;
case RESEARCH_JETPACKS :
outResearchTechName = " research_jetpacks " ;
break ;
case RESEARCH_HEAVYARMOR :
outResearchTechName = " research_heavyarmor " ;
break ;
case RESEARCH_DISTRESSBEACON :
outResearchTechName = " research_distressbeacon " ;
break ;
case RESEARCH_HEALTH :
outResearchTechName = " research_health " ;
break ;
case RESEARCH_CATALYSTS :
outResearchTechName = " research_catalysts " ;
break ;
case MESSAGE_CANCEL :
outResearchTechName = " research_cancel " ;
break ;
case RESEARCH_MOTIONTRACK :
outResearchTechName = " research_motiontracking " ;
break ;
case RESEARCH_PHASETECH :
outResearchTechName = " research_phasetech " ;
break ;
case RESEARCH_GRENADES :
outResearchTechName = " research_grenades " ;
break ;
default :
theSuccess = false ;
break ;
}
return theSuccess ;
}
bool AvHSHUGetCenterPositionForGroup ( int inGroupNumber , float * inPlayerOrigin , float * outCenterPosition )
{
bool theSuccess = false ;
vec3_t thePosition ;
float theX , theY ;
# ifdef AVH_CLIENT
theSuccess = gHUD . GetCenterPositionForGroup ( inGroupNumber , thePosition ) ;
theX = thePosition [ 0 ] ;
theY = thePosition [ 1 ] ;
# endif
# ifdef AVH_SERVER
// Loop through players, find the closest player to inPlayerOrigin, to see which player is being predicted. Is there a better way?
AvHPlayer * theClosestPlayer = NULL ;
float theClosestDistance = 10000 ;
FOR_ALL_ENTITIES ( kAvHPlayerClassName , AvHPlayer * )
float theDistance = VectorDistance ( theEntity - > pev - > origin , inPlayerOrigin ) ;
if ( theDistance < theClosestDistance )
{
theClosestPlayer = theEntity ;
theClosestDistance = theDistance ;
}
END_FOR_ALL_ENTITIES ( kAvHPlayerClassName )
if ( theClosestPlayer )
{
theSuccess = theClosestPlayer - > GetCenterPositionForGroup ( inGroupNumber , theX , theY ) ;
}
# endif
if ( theSuccess )
{
outCenterPosition [ 0 ] = theX ;
outCenterPosition [ 1 ] = theY ;
}
return theSuccess ;
}
bool AvHSHUGetIsBuilding ( AvHMessageID inMessageID )
{
bool theIsBuilding = false ;
switch ( inMessageID )
{
// Buildings
case BUILD_RESOURCES :
case BUILD_INFANTRYPORTAL :
case BUILD_COMMANDSTATION :
case BUILD_TURRET_FACTORY :
case BUILD_ARMSLAB :
case BUILD_PROTOTYPE_LAB :
case BUILD_ARMORY :
case BUILD_NUKE_PLANT :
case BUILD_OBSERVATORY :
case BUILD_PHASEGATE :
case BUILD_TURRET :
case BUILD_SIEGE :
// Alien buildings
case ALIEN_BUILD_OFFENSE_CHAMBER :
case ALIEN_BUILD_DEFENSE_CHAMBER :
case ALIEN_BUILD_SENSORY_CHAMBER :
case ALIEN_BUILD_MOVEMENT_CHAMBER :
case ALIEN_BUILD_HIVE :
theIsBuilding = true ;
break ;
}
return theIsBuilding ;
}
bool AvHSHUGetIsBuildTech ( AvHMessageID inMessageID )
{
bool theIsBuildTech = false ;
switch ( inMessageID )
{
// Buildings
case BUILD_RESOURCES :
case BUILD_INFANTRYPORTAL :
case BUILD_COMMANDSTATION :
case BUILD_TURRET_FACTORY :
case BUILD_ARMSLAB :
case BUILD_PROTOTYPE_LAB :
case BUILD_ARMORY :
//case UPGRADE_ADVANCED_WEAPON_FACTORY:
case BUILD_NUKE_PLANT :
case BUILD_OBSERVATORY :
case BUILD_SCAN :
case BUILD_PHASEGATE :
case BUILD_TURRET :
case BUILD_SIEGE :
case BUILD_HEAVY :
case BUILD_JETPACK :
// Equipment
case BUILD_AMMO :
case BUILD_HEALTH :
case BUILD_CAT :
case BUILD_WELDER :
case BUILD_MINES :
case BUILD_SHOTGUN :
case BUILD_HMG :
case BUILD_NUKE :
case BUILD_GRENADE_GUN :
//case BUILD_MEDKIT:
//case BUILD_STIMPACK:
// Alien buildings
case ALIEN_BUILD_OFFENSE_CHAMBER :
case ALIEN_BUILD_DEFENSE_CHAMBER :
case ALIEN_BUILD_SENSORY_CHAMBER :
case ALIEN_BUILD_MOVEMENT_CHAMBER :
theIsBuildTech = true ;
break ;
}
return theIsBuildTech ;
}
bool AvHSHUGetIsWeaponFocusable ( AvHWeaponID inWeaponID )
{
bool theIsFocusable = false ;
switch ( inWeaponID )
{
case AVH_WEAPON_BITE :
case AVH_WEAPON_SPIT :
case AVH_WEAPON_BITE2 :
case AVH_WEAPON_SWIPE :
case AVH_WEAPON_CLAWS :
theIsFocusable = true ;
break ;
}
return theIsFocusable ;
}
bool AvHSHUGetDoesTechCostEnergy ( AvHMessageID inMessageID )
{
bool theTechCostsEnergy = false ;
switch ( inMessageID )
{
case BUILD_SCAN :
theTechCostsEnergy = true ;
break ;
}
return theTechCostsEnergy ;
}
bool AvHSHUGetIsCombatModeTech ( AvHMessageID inMessageID )
{
bool theIsCombatModeTech = false ;
switch ( inMessageID )
{
case BUILD_SHOTGUN :
case BUILD_GRENADE_GUN :
case BUILD_HMG :
case BUILD_WELDER :
case BUILD_MINES :
case BUILD_JETPACK :
case BUILD_HEAVY :
case BUILD_SCAN :
theIsCombatModeTech = true ;
break ;
}
return theIsCombatModeTech ;
}
bool AvHSHUGetIsResearchTech ( AvHMessageID inMessageID )
{
bool theIsResearchTech = false ;
switch ( inMessageID )
{
case RESEARCH_ELECTRICAL :
case RESEARCH_ARMOR_ONE :
case RESEARCH_ARMOR_TWO :
case RESEARCH_ARMOR_THREE :
case RESEARCH_WEAPONS_ONE :
case RESEARCH_WEAPONS_TWO :
case RESEARCH_WEAPONS_THREE :
case TURRET_FACTORY_UPGRADE :
case RESEARCH_JETPACKS :
case RESEARCH_HEAVYARMOR :
case RESEARCH_DISTRESSBEACON :
case RESEARCH_HEALTH :
case RESEARCH_CATALYSTS :
case MESSAGE_CANCEL :
case RESEARCH_MOTIONTRACK :
case RESEARCH_PHASETECH :
case RESEARCH_GRENADES :
case RESOURCE_UPGRADE :
case ARMORY_UPGRADE :
theIsResearchTech = true ;
break ;
}
return theIsResearchTech ;
}
//Here's TFC's code that checks whether a player's allowed to build a sentry
//or not.
//I can't remember if there was any good reason why we used
//UTIL_FindEntitiesInSphere()
//instead of UTIL_EntitiesInBox().
//
////=========================================================================
//// Returns 0 if the area around obj is safe to build in
//int CBaseEntity::CheckArea( CBaseEntity *pIgnore )
//{
// TraceResult tr;
// Vector vecOrg = pev->origin;
//
// // Check the origin
// int iContents = UTIL_PointContents(vecOrg);
// if ( iContents != CONTENT_EMPTY && iContents != CONTENT_WATER )
// return CAREA_BLOCKED;
//
// Vector vecIgnoreOrg = pIgnore->pev->origin;
// // Get the player's origin irrelevant of crouching
// if ( pIgnore->pev->flags & FL_DUCKING )
// {
// vecIgnoreOrg = vecIgnoreOrg + (VEC_DUCK_HULL_MIN -
// VEC_HULL_MIN);
// }
// // Trace a hull
// UTIL_TraceHull( vecIgnoreOrg, pev->origin, ignore_monsters,
// large_hull, edict(), &tr );
// CBaseEntity *pEntity = CBaseEntity::Instance(tr.pHit);
// if (tr.flFraction != 1 || tr.fAllSolid == 1)
// return CAREA_BLOCKED;
//
// // Check for solid entities in the area
// CBaseEntity *pEnt = NULL;
// while ( (pEnt = UTIL_FindEntityInSphere( pEnt, pev->origin, 48 )) !=
// NULL )
// {
// // If it's not the engineer, and its a solid entity, fail
// if (pEnt != pIgnore && pEnt != this && pEnt->pev->solid >
// SOLID_TRIGGER)
// return CAREA_BLOCKED;
// }
//
// // Cycle through all the Nobuild zones in the map and make sure this
// isn't in one of them
// CBaseEntity *pNoBuild = UTIL_FindEntityByClassname( NULL,
// "func_nobuild" );
// while ( pNoBuild )
// {
// // Check to see if we're building in this zone
// if ( vecOrg.x >= pNoBuild->pev->mins.x && vecOrg.y >=
// pNoBuild->pev->mins.y && vecOrg.z >= pNoBuild->pev->mins.z &&
// vecOrg.x <= pNoBuild->pev->maxs.x &&
// vecOrg.y <= pNoBuild->pev->maxs.y && vecOrg.z <= pNoBuild->pev->maxs.z )
// return CAREA_NOBUILD;
//
// pNoBuild = UTIL_FindEntityByClassname( pNoBuild,
// "func_nobuild" );
// }
//
// // Check below
// UTIL_TraceLine( vecOrg, vecOrg + Vector(0,0,-64),
// dont_ignore_monsters, edict(), &tr );
// if ( tr.flFraction == 1.0 )
// return CAREA_BLOCKED;
//
// return CAREA_CLEAR;
//}
//bool AvHSHUGetIsEnoughRoom(const vec3_t& inCenter, const vec3_t& inMin, const vec3_t& inMax)
//{
// bool theEnoughRoom = false;
//
// // Do a traceline from center to center + min
// // Hit nothing?
// // Do a traceline from center to center + max
// // Hit nothing?
// // Success
// theEnoughRoom = true;
//
// return theEnoughRoom;
//}
bool AvHSHUGetIsGroupMessage ( AvHMessageID inMessageID )
{
bool theIsGroupMessage = false ;
switch ( inMessageID )
{
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 :
theIsGroupMessage = true ;
}
return theIsGroupMessage ;
}
bool AvHSHUGetSizeForTech ( AvHMessageID inMessageID , Vector & outMinSize , Vector & outMaxSize , bool inGetSizeToPlace )
{
bool theSuccess = false ;
// Onos-sized
//const int theOnosHeightNeededToSpawn = HULL3_MAXZ - HULL3_MINZ + kRespawnFudgeFactorHeight;
//const int theOnosWidthNeededToSpawn = HULL3_MAXY - HULL3_MINY + kRespawnFudgeFactorHeight;
// Marine-sized
const int theMarineHeightNeededToSpawn = HULL0_MAXZ - HULL0_MINZ + kRespawnFudgeFactorHeight ;
switch ( inMessageID )
{
case BUILD_INFANTRYPORTAL :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 14.49 ) ;
// Make sure there is room above it for players to spawn
if ( inGetSizeToPlace )
{
outMaxSize . z + = theMarineHeightNeededToSpawn ;
}
theSuccess = true ;
break ;
case BUILD_PHASEGATE :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 14.49 ) ;
// Make sure there is remove above it for players to spawn
if ( inGetSizeToPlace )
{
//outMinSize.x = outMinSize.y = -theOnosWidthNeededToSpawn;
//outMaxSize.x = outMaxSize.y = theOnosWidthNeededToSpawn;
//outMaxSize.z += theOnosHeightNeededToSpawn;
outMaxSize . z + = theMarineHeightNeededToSpawn ;
}
theSuccess = true ;
break ;
case BUILD_RESOURCES :
outMinSize = kResourceMinSize ;
outMaxSize = kResourceMaxSize ;
theSuccess = true ;
break ;
case ALIEN_BUILD_RESOURCES :
outMinSize = kAlienResourceMinSize ;
outMaxSize = kAlienResourceMaxSize ;
theSuccess = true ;
break ;
default :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 66.9486 ) ;
theSuccess = true ;
break ;
case BUILD_TURRET :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 42.0 ) ;
theSuccess = true ;
break ;
case ALIEN_BUILD_OFFENSE_CHAMBER :
case ALIEN_BUILD_DEFENSE_CHAMBER :
case ALIEN_BUILD_SENSORY_CHAMBER :
case ALIEN_BUILD_MOVEMENT_CHAMBER :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 44.0 ) ;
theSuccess = true ;
break ;
case BUILD_COMMANDSTATION :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 70.34 ) ;
theSuccess = true ;
break ;
case BUILD_TURRET_FACTORY :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
//outMaxSize = Vector(16.0, 16.0, 55.68);
outMaxSize = Vector ( 16.0 , 16.0 , 62.1931 ) ;
theSuccess = true ;
break ;
case BUILD_ARMORY :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 62.1931 ) ;
theSuccess = true ;
break ;
case BUILD_PROTOTYPE_LAB :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 67.7443 ) ;
theSuccess = true ;
break ;
case BUILD_OBSERVATORY :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 80.7443 ) ;
theSuccess = true ;
break ;
case BUILD_SIEGE :
outMinSize = Vector ( - 16 , - 16 , 0 ) ;
outMaxSize = Vector ( 16.0 , 16.0 , 62.1931 /*50.6678*/ ) ;
theSuccess = true ;
break ;
case BUILD_HEALTH :
outMinSize = kHealthMinSize ;
outMaxSize = kHealthMaxSize ;
theSuccess = true ;
break ;
case BUILD_CAT :
outMinSize = kCatalystMinSize ;
outMaxSize = kCatalystMaxSize ;
theSuccess = true ;
break ;
case BUILD_HEAVY :
outMinSize = kHeavyMinSize ;
outMaxSize = kHeavyMaxSize ;
theSuccess = true ;
break ;
case BUILD_JETPACK :
outMinSize = kJetpackMinSize ;
outMaxSize = kJetpackMaxSize ;
theSuccess = true ;
break ;
case BUILD_AMMO :
outMinSize = kAmmoMinSize ;
outMaxSize = kAmmoMaxSize ;
theSuccess = true ;
break ;
case BUILD_MINES :
outMinSize = kWeaponMinSize ;
outMaxSize = Vector ( 16 , 16 , 40 ) ;
theSuccess = true ;
break ;
case BUILD_SHOTGUN :
case BUILD_HMG :
case BUILD_WELDER :
case BUILD_NUKE :
case BUILD_GRENADE_GUN :
outMinSize = kWeaponMinSize ;
outMaxSize = kWeaponMaxSize ;
theSuccess = true ;
break ;
case ALIEN_BUILD_HIVE :
outMinSize = kHiveMinSize ;
outMaxSize = kHiveMaxSize ;
theSuccess = true ;
break ;
}
return theSuccess ;
}
bool AvHSHUGetSizeForPlayerUser3 ( AvHUser3 inUser3 , Vector & outMinSize , Vector & outMaxSize , bool inDucking )
{
bool theSuccess = false ;
// Now set size
switch ( inUser3 )
{
case AVH_USER3_NONE :
case AVH_USER3_MARINE_PLAYER :
case AVH_USER3_COMMANDER_PLAYER :
case AVH_USER3_ALIEN_PLAYER4 :
// Get simple case working first
if ( inDucking )
{
outMinSize = HULL1_MIN ;
outMaxSize = HULL1_MAX ;
}
else
{
outMinSize = HULL0_MIN ;
outMaxSize = HULL0_MAX ;
}
theSuccess = true ;
break ;
case AVH_USER3_ALIEN_PLAYER1 :
case AVH_USER3_ALIEN_PLAYER2 :
case AVH_USER3_ALIEN_PLAYER3 :
case AVH_USER3_ALIEN_EMBRYO :
outMinSize = HULL1_MIN ;
outMaxSize = HULL1_MAX ;
theSuccess = true ;
break ;
case AVH_USER3_ALIEN_PLAYER5 :
if ( inDucking )
{
outMinSize = HULL0_MIN ;
outMaxSize = HULL0_MAX ;
}
else
{
outMinSize = HULL3_MIN ;
outMaxSize = HULL3_MAX ;
}
theSuccess = true ;
break ;
}
return theSuccess ;
}
bool AvHSHUGetSizeForUser3 ( AvHUser3 inUser3 , Vector & outMinSize , Vector & outMaxSize )
{
bool theSuccess = false ;
// If it's a player, get max size he can be (assuming he's not ducking)
theSuccess = AvHSHUGetSizeForPlayerUser3 ( inUser3 , outMinSize , outMaxSize , false ) ;
if ( ! theSuccess )
{
AvHMessageID theMessageID = MESSAGE_NULL ;
// Convert it to a AvHMessageID if possible
if ( AvHSHUUser3ToMessageID ( inUser3 , theMessageID ) )
{
theSuccess = AvHSHUGetSizeForTech ( theMessageID , outMinSize , outMaxSize ) ;
}
}
return theSuccess ;
}
bool AvHSHUUser3ToMessageID ( AvHUser3 inUser3 , AvHMessageID & outMessageID )
{
AvHMessageID theMessageID = MESSAGE_NULL ;
bool theSuccess = false ;
switch ( inUser3 )
{
case AVH_USER3_FUNC_RESOURCE :
case AVH_USER3_RESTOWER :
theMessageID = BUILD_RESOURCES ;
break ;
case AVH_USER3_COMMANDER_STATION :
theMessageID = BUILD_COMMANDSTATION ;
break ;
case AVH_USER3_TURRET_FACTORY :
case AVH_USER3_ADVANCED_TURRET_FACTORY :
theMessageID = BUILD_TURRET_FACTORY ;
break ;
case AVH_USER3_ARMORY :
case AVH_USER3_ADVANCED_ARMORY :
theMessageID = BUILD_ARMORY ;
break ;
case AVH_USER3_ARMSLAB :
theMessageID = BUILD_ARMSLAB ;
break ;
case AVH_USER3_PROTOTYPE_LAB :
theMessageID = BUILD_PROTOTYPE_LAB ;
break ;
case AVH_USER3_OBSERVATORY :
theMessageID = BUILD_OBSERVATORY ;
break ;
case AVH_USER3_TURRET :
theMessageID = BUILD_TURRET ;
break ;
case AVH_USER3_SIEGETURRET :
theMessageID = BUILD_SIEGE ;
break ;
case AVH_USER3_INFANTRYPORTAL :
theMessageID = BUILD_INFANTRYPORTAL ;
break ;
case AVH_USER3_PHASEGATE :
theMessageID = BUILD_PHASEGATE ;
break ;
case AVH_USER3_HEAVY :
theMessageID = BUILD_HEAVY ;
break ;
case AVH_USER3_JETPACK :
theMessageID = BUILD_JETPACK ;
break ;
case AVH_USER3_DEFENSE_CHAMBER :
theMessageID = ALIEN_BUILD_DEFENSE_CHAMBER ;
break ;
case AVH_USER3_MOVEMENT_CHAMBER :
theMessageID = ALIEN_BUILD_MOVEMENT_CHAMBER ;
break ;
case AVH_USER3_OFFENSE_CHAMBER :
theMessageID = ALIEN_BUILD_OFFENSE_CHAMBER ;
break ;
case AVH_USER3_SENSORY_CHAMBER :
theMessageID = ALIEN_BUILD_SENSORY_CHAMBER ;
break ;
case AVH_USER3_ALIENRESTOWER :
theMessageID = ALIEN_BUILD_RESOURCES ;
break ;
case AVH_USER3_HIVE :
theMessageID = ALIEN_BUILD_HIVE ;
break ;
case AVH_USER3_ALIEN_PLAYER1 :
theMessageID = ALIEN_LIFEFORM_ONE ;
break ;
case AVH_USER3_ALIEN_PLAYER2 :
theMessageID = ALIEN_LIFEFORM_TWO ;
break ;
case AVH_USER3_ALIEN_PLAYER3 :
theMessageID = ALIEN_LIFEFORM_THREE ;
break ;
case AVH_USER3_ALIEN_PLAYER4 :
theMessageID = ALIEN_LIFEFORM_FOUR ;
break ;
case AVH_USER3_ALIEN_PLAYER5 :
theMessageID = ALIEN_LIFEFORM_FIVE ;
break ;
}
if ( theMessageID ! = MESSAGE_NULL )
{
outMessageID = theMessageID ;
theSuccess = true ;
}
return theSuccess ;
}
bool AvHSHUMessageIDToUser3 ( AvHMessageID inMessageID , AvHUser3 & outUser3 )
{
bool theSuccess = false ;
AvHUser3 theUser3 = AVH_USER3_NONE ;
switch ( inMessageID )
{
case BUILD_RESOURCES :
theUser3 = AVH_USER3_RESTOWER ;
break ;
case BUILD_COMMANDSTATION :
theUser3 = AVH_USER3_COMMANDER_STATION ;
break ;
case BUILD_TURRET_FACTORY :
theUser3 = AVH_USER3_TURRET_FACTORY ;
break ;
case TURRET_FACTORY_UPGRADE :
theUser3 = AVH_USER3_ADVANCED_TURRET_FACTORY ;
break ;
case BUILD_ARMORY :
theUser3 = AVH_USER3_ARMORY ;
break ;
case ARMORY_UPGRADE :
theUser3 = AVH_USER3_ADVANCED_ARMORY ;
break ;
case BUILD_ARMSLAB :
theUser3 = AVH_USER3_ARMSLAB ;
break ;
case BUILD_PROTOTYPE_LAB :
theUser3 = AVH_USER3_PROTOTYPE_LAB ;
break ;
case BUILD_OBSERVATORY :
theUser3 = AVH_USER3_OBSERVATORY ;
break ;
case BUILD_TURRET :
theUser3 = AVH_USER3_TURRET ;
break ;
case BUILD_SIEGE :
theUser3 = AVH_USER3_SIEGETURRET ;
break ;
case BUILD_INFANTRYPORTAL :
theUser3 = AVH_USER3_INFANTRYPORTAL ;
break ;
case BUILD_PHASEGATE :
theUser3 = AVH_USER3_PHASEGATE ;
break ;
case BUILD_HEAVY :
theUser3 = AVH_USER3_HEAVY ;
break ;
case BUILD_JETPACK :
theUser3 = AVH_USER3_JETPACK ;
break ;
// Menus
case MENU_BUILD :
theUser3 = AVH_USER3_MENU_BUILD ;
break ;
case MENU_BUILD_ADVANCED :
theUser3 = AVH_USER3_MENU_BUILD_ADVANCED ;
break ;
case MENU_ASSIST :
theUser3 = AVH_USER3_MENU_ASSIST ;
break ;
case MENU_EQUIP :
theUser3 = AVH_USER3_MENU_EQUIP ;
break ;
// Weapons
case BUILD_MINES :
theUser3 = AVH_USER3_MINE ;
break ;
// Lifeforms
case ALIEN_LIFEFORM_ONE :
theUser3 = AVH_USER3_ALIEN_PLAYER1 ;
break ;
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 ;
}
if ( theUser3 ! = AVH_USER3_NONE )
{
outUser3 = theUser3 ;
theSuccess = true ;
}
return theSuccess ;
}
float AvHSHUGetTime ( )
{
float theTime = 0 ;
# ifdef AVH_SERVER
theTime = gpGlobals - > time ;
# else
theTime = gEngfuncs . GetClientTime ( ) ;
# endif
return theTime ;
}
// Note, all these models must be precached already
char * AvHSHUGetBuildTechModelName ( AvHMessageID inMessageID )
{
char * theModelName = NULL ;
switch ( inMessageID )
{
case BUILD_RESOURCES :
theModelName = kResourceTowerModel ;
break ;
//case BUILD_REINFORCEMENTS:
// theModelName = kInfantryPortalModel;
// break;
case BUILD_INFANTRYPORTAL :
theModelName = kInfantryPortalModel ;
break ;
case BUILD_COMMANDSTATION :
theModelName = kCommandStationModel ;
break ;
case BUILD_TURRET_FACTORY :
theModelName = kTurretFactoryModel ;
break ;
case BUILD_ARMSLAB :
theModelName = kArmsLabModel ;
break ;
case BUILD_PROTOTYPE_LAB :
theModelName = kPrototypeLabModel ;
break ;
case BUILD_ARMORY :
theModelName = kArmoryModel ;
break ;
case ARMORY_UPGRADE :
theModelName = kAdvancedWeaponFactoryModel ;
break ;
case BUILD_OBSERVATORY :
theModelName = kObservatoryModel ;
break ;
case BUILD_SCAN :
theModelName = kScanModel ;
break ;
case BUILD_PHASEGATE :
theModelName = kPhaseGateModel ;
break ;
case BUILD_TURRET :
theModelName = kDeployedTurretModel ;
break ;
case BUILD_NUKE :
theModelName = kNukeModel ;
break ;
case BUILD_SIEGE :
theModelName = kSiegeTurretModel ;
//theModelName = kDeployedTurretModel;
break ;
case BUILD_CAT :
theModelName = kCatalystModel ;
break ;
case BUILD_HEALTH :
theModelName = kHealthModel ;
break ;
case BUILD_HEAVY :
theModelName = kHeavyModel ;
break ;
case BUILD_JETPACK :
theModelName = kJetpackModel ;
break ;
case BUILD_AMMO :
theModelName = kAmmoModel ;
break ;
case BUILD_WELDER :
theModelName = kWelderWModel ;
break ;
case BUILD_MINES :
theModelName = kTripmineW2Model ;
break ;
case BUILD_SHOTGUN :
theModelName = kSGWModel ;
break ;
case BUILD_HMG :
theModelName = kHMGWModel ;
break ;
case BUILD_GRENADE_GUN :
theModelName = kGGWModel ;
break ;
// Alien buildings
case ALIEN_BUILD_HIVE :
theModelName = kHiveModel ;
break ;
case ALIEN_BUILD_RESOURCES :
theModelName = kAlienResourceTowerModel ;
break ;
case ALIEN_BUILD_OFFENSE_CHAMBER :
theModelName = kOffenseChamberModel ;
break ;
case ALIEN_BUILD_DEFENSE_CHAMBER :
theModelName = kDefenseChamberModel ;
break ;
case ALIEN_BUILD_SENSORY_CHAMBER :
theModelName = kSensoryChamberModel ;
break ;
case ALIEN_BUILD_MOVEMENT_CHAMBER :
theModelName = kMovementChamberModel ;
break ;
}
return theModelName ;
}
bool AvHSHUGetBuildTechRange ( AvHMessageID inMessageID , float & outRange )
{
bool theSuccess = false ;
// switch(inMessageID)
// {
// case BUILD_CAMERA:
// outRange = kResourceTowerSightRange;
// theSuccess = true;
// break;
// case BUILD_PHASEGATE:
// break;
// case BUILD_TURRET:
// outRange = TURRET_RANGE;
// theSuccess = true;
// break;
// case BUILD_SIEGE:
// outRange = kSiegeTurretMaxRange;
// theSuccess = true;
// break;
// }
return theSuccess ;
}
bool AvHSHUTraceLineIsAreaFree ( Vector & inStart , Vector & inEnd , edict_t * inIgnoreEntity , bool inIgnorePlayers )
{
bool theAreaIsFree = true ;
# ifdef AVH_SERVER
TraceResult theTR ;
bool theIsDone = false ;
// Do tracelines between the corners, to make sure there's no geometry inside the box
int theNumIters = 0 ;
IGNORE_MONSTERS theIgnoreMonsters = dont_ignore_monsters ;
if ( inIgnorePlayers )
{
theIgnoreMonsters = ignore_monsters ;
}
while ( ! theIsDone )
{
UTIL_TraceLine ( inStart , inEnd , theIgnoreMonsters , inIgnoreEntity , & theTR ) ;
if ( theTR . flFraction ! = 1.0f )
{
CBaseEntity * theEntity = CBaseEntity : : Instance ( ENT ( theTR . pHit ) ) ;
if ( theEntity & & ( theEntity - > pev - > solid ! = SOLID_BBOX ) & & ( theEntity - > pev - > solid ! = SOLID_SLIDEBOX ) & & ( theEntity - > pev - > solid ! = SOLID_BSP ) & & ( inIgnorePlayers ! = ! ! theEntity - > IsPlayer ( ) ) )
{
VectorCopy ( theTR . vecEndPos , inStart ) ;
}
else
{
theIsDone = true ;
theAreaIsFree = false ;
}
}
else
{
theIsDone = true ;
}
if ( theNumIters + + > 50 )
{
theIsDone = true ;
theAreaIsFree = false ;
}
}
# endif
// int theIndex;
// vec3_t theEndLocation;
// AvHTeamNumber theTeam;
// bool thePlayerHit;
// int theUserThree;
// int theUserFour;
//
// if(AvHSHUTraceTangible(inStart, inEnd, theIndex, theEndLocation, theTeam, thePlayerHit, theUserThree, theUserFour))
// {
// if(theIndex >= 0)
// {
// theAreaIsFree = false;
// }
// }
return theAreaIsFree ;
}
float AvHTraceLineAgainstWorld ( Vector & vecStart , Vector & vecEnd )
{
# ifdef AVH_SERVER
TraceResult tr ;
UTIL_TraceLine ( vecStart , vecEnd , ignore_monsters , dont_ignore_glass , NULL , & tr ) ;
return tr . flFraction ;
# endif
# ifdef AVH_CLIENT
// This may not be the most efficient way, but it seems to get the job done.
float theFraction = 1 ;
for ( int i = 0 ; i < kMaxEntities & & theFraction > 0 ; + + i )
{
cl_entity_t * theEntity = gEngfuncs . GetEntityByIndex ( i ) ;
if ( theEntity ! = NULL & & theEntity - > model ! = NULL & & theEntity - > model - > type = = mod_brush )
{
vec3_t localStart ;
vec3_t localEnd ;
// Translate the start and end into the model's frame of reference.
VectorSubtract ( vecStart , theEntity - > origin , localStart ) ;
VectorSubtract ( vecEnd , theEntity - > origin , localEnd ) ;
// Rotate the start and end into the model's frame of reference.
if ( theEntity - > angles [ 0 ] | |
theEntity - > angles [ 1 ] | |
theEntity - > angles [ 2 ] )
{
vec3_t forward ;
vec3_t right ;
vec3_t up ;
AngleVectors ( theEntity - > angles , forward , right , up ) ;
vec3_t temp ;
VectorCopy ( localStart , temp ) ;
localStart [ 0 ] = DotProduct ( temp , forward ) ;
localStart [ 1 ] = - DotProduct ( temp , right ) ;
localStart [ 2 ] = DotProduct ( temp , up ) ;
VectorCopy ( localEnd , temp ) ;
localEnd [ 0 ] = DotProduct ( temp , forward ) ;
localEnd [ 1 ] = - DotProduct ( temp , right ) ;
localEnd [ 2 ] = DotProduct ( temp , up ) ;
}
trace_t tr ;
NS_TraceLine ( & theEntity - > model - > hulls [ 0 ] , localStart , localEnd , & tr ) ;
if ( tr . fraction < theFraction )
{
theFraction = tr . fraction ;
}
}
}
return theFraction ;
# endif
}
bool AvHSHUGetCanBeBuiltOnPlayers ( AvHMessageID inMessageID )
{
bool theCanBeBuiltOnPlayers = false ;
switch ( inMessageID )
{
case ALIEN_BUILD_HIVE :
case BUILD_SCAN :
case BUILD_AMMO :
case BUILD_HEALTH :
case BUILD_CAT :
case BUILD_MINES :
case BUILD_WELDER :
case BUILD_SHOTGUN :
case BUILD_HMG :
case BUILD_GRENADE_GUN :
case BUILD_HEAVY :
case BUILD_JETPACK :
theCanBeBuiltOnPlayers = true ;
break ;
}
return theCanBeBuiltOnPlayers ;
}
bool AvHSHUGetIsSiteValidForBuild ( AvHMessageID inMessageID , Vector * inLocation , int inIgnoredEntityIndex )
{
//TODO: - check entity returned by client side commander trace for drop validity before we ever get this far
bool theSuccess = false ;
// Check that there's enough room for building
vec3_t theMinSize , theMaxSize ;
if ( AvHSHUGetSizeForTech ( inMessageID , theMinSize , theMaxSize , true ) )
{
// If it's a build with special placement requirements, check that here (this could modify the location also)
if ( AvHSHUGetAreSpecialBuildingRequirementsMet ( inMessageID , * inLocation ) )
{
// Assume mappers place hives correctly (it's such a showstopper if aliens can't create a hive because it's too close to a wall)
// KGP 4/28/04 - add res towers to hives for the same reason -- if a mapper didn't give room, it's a showstopper.
bool theIgnorePlayers = AvHSHUGetCanBeBuiltOnPlayers ( inMessageID ) ;
float theSlopeTangent = theIgnorePlayers ? kMaxEquipmentDropSlope : kMaxBuildingDropSlope ;
bool theSkipDropCheck = false ;
switch ( inMessageID )
{
case BUILD_RESOURCES :
theSkipDropCheck = true ;
break ;
case ALIEN_BUILD_RESOURCES :
theSkipDropCheck = true ;
break ;
case ALIEN_BUILD_HIVE :
theSkipDropCheck = true ;
break ;
}
if ( theSkipDropCheck | | AvHSHUGetCanDropItem ( * inLocation , theMinSize , theMaxSize , theSlopeTangent , inIgnoredEntityIndex , theIgnorePlayers ) )
{
//TODO: - better check for entity below drop, since it's often off center.
// Check to make sure building isn't being created on top of another building
vec3_t thePositionBelowBuild = * inLocation ;
thePositionBelowBuild . z = - kMaxMapDimension ;
int theIndex ;
vec3_t theEndLocation ;
AvHTeamNumber theTeam ;
bool thePlayerHit ;
int theUserThree ;
int theUserFour ;
// Don't allow building on any entities, except allow resource towers on top of func_resources
//TODO : use full list instead of physents, which isn't accounting for items the commander can't see
bool theHitSomething = AvHSHUTraceTangible ( * inLocation , thePositionBelowBuild , theIndex , theEndLocation , theTeam , thePlayerHit , theUserThree , theUserFour ) ;
//res building case
if ( inMessageID = = BUILD_RESOURCES | | inMessageID = = ALIEN_BUILD_RESOURCES )
{
# ifdef AVH_SERVER //on server, return false if occupied
FOR_ALL_ENTITIES ( kesFuncResource , AvHFuncResource * )
if ( ! theEntity - > GetIsOccupied ( ) ) // open for use
{
if ( VectorDistance ( theEntity - > pev - > origin , * inLocation ) < 20 ) //small enough that we don't check the wrong nozzle
{
theSuccess = ( ! theHitSomething | | theIndex < = 0 | | theUserThree = = AVH_USER3_FUNC_RESOURCE ) ;
break ;
}
}
END_FOR_ALL_ENTITIES ( kesFuncResource )
# else //on client the occupied function isn't available
theSuccess = ( ! theHitSomething | | theIndex < = 0 | | theUserThree = = AVH_USER3_FUNC_RESOURCE ) ;
# endif
}
else if ( inMessageID = = ALIEN_BUILD_HIVE )
{
theSuccess = true ;
//theSuccess = ( !theHitSomething || ( *inLocation[2] - theEndLocation[2] > 40.0f ) );
//#ifdef AVH_SERVER
// ALERT(at_console, "theHitSomething=%d theSuccess=%d\n", theHitSomething, theSuccess);
// ALERT(at_console, "%f\n", *inLocation[2] - theEndLocation[2]);
// ALERT(at_console, "{%f, %f, %f} : { %f, %f, %f }\n",
// inLocation[0], inLocation[1], inLocation[2],
// theEndLocation[0], theEndLocation[1], theEndLocation[2]);
//#endif
}
else if ( ! theHitSomething | | ( theIgnorePlayers & & thePlayerHit ) | | theIndex < = 0 )
{
// THEN it's a legal build spot and the building shows up green.
theSuccess = true ;
}
}
}
}
return theSuccess ;
}
int AvHSHUGetPointContents ( vec3_t inPoint )
{
int thePointContents = 0 ;
# ifdef AVH_SERVER
thePointContents = UTIL_PointContents ( inPoint ) ;
# endif
# ifdef AVH_CLIENT
gEngfuncs . pEventAPI - > EV_SetUpPlayerPrediction ( false , true ) ;
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
gEngfuncs . pEventAPI - > EV_SetTraceHull ( 2 ) ;
thePointContents = gEngfuncs . PM_PointContents ( inPoint , NULL ) ;
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
# endif
return thePointContents ;
}
/*
* Check to see if the build or item is overlapping the world .
*/
bool AvHSHUGetIsVolumeContentNonsolid ( vec3_t inCenter , Vector & inMinSize , Vector & inMaxSize , edict_t * inIgnoreEntity , bool inIgnorePlayers )
{
float theMins [ 3 ] , theMaxs [ 3 ] ;
VectorAdd ( inCenter , inMinSize , theMins ) ;
VectorAdd ( inCenter , inMaxSize , theMaxs ) ;
bool theIsNonsolid = true ; //innocent until proven guilty
extern playermove_t * pmove ;
if ( pmove ! = NULL & & pmove - > physents [ 0 ] . model ! = NULL )
{
const hull_t * theHull = & pmove - > physents [ 0 ] . model - > hulls [ NS_GetValveHull ( 2 ) ] ;
int theContents = NS_BoxContents ( theHull , theHull - > firstclipnode , theMins , theMaxs ) ;
theIsNonsolid = ( theContents ! = CONTENT_SOLID ) ;
}
else //fallback if we're just starting a round
{
int theContents = AvHSHUGetPointContents ( inCenter ) ;
if ( theContents ! = CONTENT_SOLID )
{
//trace between each corner pair once looking for obstructions - 28 total comparisons
const static int MIN = 0 ;
const static int MAX = 1 ;
const static int NUM_CORNERS = 8 ;
int theIndex ;
Vector theCorners [ NUM_CORNERS ] ;
for ( int XPos = MIN ; XPos < = MAX ; + + XPos )
{
for ( int YPos = MIN ; YPos < = MAX ; + + YPos )
{
for ( int ZPos = MIN ; ZPos < = MAX ; + + ZPos )
{
theIndex = XPos + YPos * 2 + ZPos * 4 ;
theCorners [ theIndex ] . x = inCenter . x + ( ( XPos = = MIN ) ? inMinSize . x : inMaxSize . x ) ;
theCorners [ theIndex ] . y = inCenter . y + ( ( YPos = = MIN ) ? inMinSize . y : inMaxSize . y ) ;
theCorners [ theIndex ] . z = inCenter . z + ( ( ZPos = = MIN ) ? inMinSize . z : inMaxSize . z ) ;
}
}
}
for ( int startCorner = 0 ; startCorner < NUM_CORNERS ; + + startCorner )
{
for ( int endCorner = startCorner + 1 ; endCorner < NUM_CORNERS ; + + endCorner )
{
if ( ! AvHSHUTraceLineIsAreaFree ( theCorners [ startCorner ] , theCorners [ endCorner ] , inIgnoreEntity , inIgnorePlayers ) )
{
theIsNonsolid = false ;
break ;
}
}
if ( ! theIsNonsolid )
{
break ;
}
}
}
else
{
theIsNonsolid = false ;
}
}
return theIsNonsolid ;
}
bool AvHSHUGetIsAreaFree ( vec3_t inCenter , Vector & inMinSize , Vector & inMaxSize , edict_t * inIgnoreEntity , bool inIgnorePlayers )
{
bool theAreaIsFree = AvHSHUGetIsVolumeContentNonsolid ( inCenter , inMinSize , inMaxSize , inIgnoreEntity , inIgnorePlayers ) ;
if ( theAreaIsFree )
{
theAreaIsFree = ! AvHSHUGetEntitiesBlocking ( inCenter , inMinSize , inMaxSize , inIgnoreEntity , inIgnorePlayers ) ;
}
return theAreaIsFree ;
}
bool AvHSHUGetEntitiesBlocking ( Vector & inOrigin , Vector & inMinSize , Vector & inMaxSize , edict_t * inIgnoreEntity , bool inIgnorePlayers )
{
bool theEntBlocking = false ;
typedef vector < pair < int , Vector > > PhysEntListType ;
PhysEntListType theEntList ;
const int kSearchRadius = 800 ;
// Populate phys ent list
# ifdef AVH_SERVER
CBaseEntity * theEntity = NULL ;
while ( ( theEntity = UTIL_FindEntityInSphere ( theEntity , inOrigin , kSearchRadius ) ) ! = NULL )
{
// If entity is visible non-world object, add it
if ( ( theEntity - > pev - > team ! = TEAM_IND ) & & ( theEntity - > pev - > solid ! = SOLID_NOT & & theEntity - > pev - > solid ! = SOLID_TRIGGER ) )
{
AvHPlayer * thePlayer = dynamic_cast < AvHPlayer * > ( theEntity ) ;
if ( theEntity - > edict ( ) ! = inIgnoreEntity & & ! ( thePlayer ! = NULL & & inIgnorePlayers ) )
{
theEntList . push_back ( make_pair ( theEntity - > pev - > iuser3 , theEntity - > pev - > origin ) ) ;
}
}
}
# endif
# ifdef AVH_CLIENT
// Haven't implemented this, so depend on it not being passed in
ASSERT ( inIgnoreEntity = = NULL ) ;
for ( int i = 0 ; i < pmove - > numphysent ; i + + )
{
physent_t * thePhysEnt = pmove - > physents + i ;
if ( ( thePhysEnt - > team ! = TEAM_IND ) & & ( thePhysEnt - > solid ! = SOLID_NOT & & thePhysEnt - > solid ! = SOLID_TRIGGER ) )
{
theEntList . push_back ( make_pair ( thePhysEnt - > iuser3 , thePhysEnt - > origin ) ) ;
}
}
# endif
Vector theAbsMax ;
VectorAdd ( inOrigin , inMaxSize , theAbsMax ) ;
Vector theAbsMin ;
VectorAdd ( inOrigin , inMinSize , theAbsMin ) ;
Vector theOrigin1 ;
VectorAdd ( theAbsMax , theAbsMin , theOrigin1 ) ;
theOrigin1 = theOrigin1 * .5f ;
Vector theSize1 ;
VectorSubtract ( theAbsMax , theAbsMin , theSize1 ) ;
theSize1 = theSize1 * .5f ;
// Do collision on list, making sure none are too close to inOrigin
for ( PhysEntListType : : iterator theIter = theEntList . begin ( ) ; theIter ! = theEntList . end ( ) ; theIter + + )
{
// Get size for tech
Vector theMinSize , theMaxSize ;
if ( AvHSHUGetSizeForUser3 ( AvHUser3 ( theIter - > first ) , theMinSize , theMaxSize ) )
{
Vector theEntOrigin = theIter - > second ;
Vector theAbsMax ;
VectorAdd ( theEntOrigin , theMaxSize , theAbsMax ) ;
Vector theAbsMin ;
VectorAdd ( theEntOrigin , inMinSize , theAbsMin ) ;
Vector theOrigin2 ;
VectorAdd ( theAbsMax , theAbsMin , theOrigin2 ) ;
theOrigin2 = theOrigin2 * .5f ;
Vector theSize2 ;
VectorSubtract ( theAbsMax , theAbsMin , theSize2 ) ;
theSize2 = theSize2 * .5f ;
// Do simple box collision here
if ( NS_BoxesOverlap ( ( float * ) theOrigin1 , ( float * ) theSize1 , ( float * ) theOrigin2 , ( float * ) theSize2 ) )
{
theEntBlocking = true ;
break ;
}
}
}
return theEntBlocking ;
}
void AvHSHUMakeViewFriendlyKillerName ( string & ioKillerName )
{
string theOutputName = ioKillerName ;
int theStrLen = ( int ) ioKillerName . length ( ) ;
if ( ! strncmp ( ioKillerName . c_str ( ) , " weapon_ " , 7 ) )
{
theOutputName = ioKillerName . substr ( 7 ) ;
}
else if ( ! strncmp ( ioKillerName . c_str ( ) , " monster_ " , 8 ) )
{
theOutputName = ioKillerName . substr ( 8 ) ;
}
else if ( ! strncmp ( ioKillerName . c_str ( ) , " func_ " , 5 ) )
{
theOutputName = ioKillerName . substr ( 5 ) ;
}
ioKillerName = theOutputName ;
}
bool AvHSHUGetCanDropItem ( vec3_t & ioCenter , Vector & inMinSize , Vector & inMaxSize , float inMaxSlopeTangent , int inIgnoreIndex , bool inIgnorePlayers )
{
float theMaxXWidth = max ( - inMinSize [ 0 ] , inMaxSize [ 0 ] ) ;
float theMaxYWidth = max ( - inMinSize [ 1 ] , inMaxSize [ 1 ] ) ;
float theRadius = sqrt ( theMaxXWidth * theMaxXWidth + theMaxYWidth * theMaxYWidth ) ;
float theHeight = inMaxSize [ 2 ] - inMinSize [ 2 ] ;
//adjust origin to be base
float theOrigin [ 3 ] = { ioCenter [ 0 ] , ioCenter [ 1 ] , ioCenter [ 2 ] + inMinSize [ 2 ] } ;
CollisionChecker Checker ( pmove , kHLPointHullIndex , CollisionChecker : : HULL_TYPE_ALL , inIgnorePlayers , CollisionChecker : : IGNORE_NONE , inIgnoreIndex ) ;
bool theCanDropItem = ( Checker . GetContentsInCylinder ( theOrigin , theRadius , theHeight ) ! = CONTENTS_SOLID ) ;
if ( ! theCanDropItem ) //can't place it -- can we drop it?
{
float theMaxDropHeight = theRadius * inMaxSlopeTangent ;
float theDropOrigin [ 3 ] = { theOrigin [ 0 ] , theOrigin [ 1 ] , theOrigin [ 2 ] + theMaxDropHeight } ;
theCanDropItem = ( Checker . GetContentsInCylinder ( theDropOrigin , theRadius , theHeight ) ! = CONTENTS_SOLID ) ;
if ( theCanDropItem ) //can drop it -- get as low to ground as possible for drop
{
float theBestDropHeight = theMaxDropHeight ;
float theShiftFraction = 1.0f ;
for ( float theShiftAdjust = 0.5f ; theShiftAdjust > 0.05f ; theShiftAdjust / = 2 )
{
theShiftFraction + = theShiftAdjust * ( theCanDropItem ? - 1 : 1 ) ; //try lower if last was a success
theDropOrigin [ 2 ] = theMaxDropHeight ;
theDropOrigin [ 2 ] * = theShiftFraction ;
theDropOrigin [ 2 ] + = theOrigin [ 2 ] ;
theCanDropItem = ( Checker . GetContentsInCylinder ( theDropOrigin , theRadius , theHeight ) ! = CONTENTS_SOLID ) ;
if ( theCanDropItem )
{
theBestDropHeight = theShiftFraction ;
theBestDropHeight * = theMaxDropHeight ;
}
}
//adjust center to best position
ioCenter [ 2 ] + = theBestDropHeight ;
theCanDropItem = true ;
}
}
return theCanDropItem ;
}
bool AvHSHUTraceAndGetIsSiteValidForBuild ( AvHMessageID inMessageID , const Vector & inPointOfView , const Vector & inNormRay , Vector * outLocation )
{
bool theSuccess = false ;
// TODO: Check if area is within the mapextents
int theUser3 ;
bool theTraceSuccess = AvHSHUTraceTangible ( inPointOfView , inNormRay , & theUser3 , outLocation ) ;
// tankefugl: 0000291
// ignore trace for scans (removed due to cost being paid when drop failed)
//if (inMessageID == BUILD_SCAN)
//{
// theSuccess = true;
//}
//else
// :tankefugl
if ( theTraceSuccess )
{
// tankefugl: 0000291
if ( ( inMessageID = = BUILD_SCAN ) | | ( AvHSHUGetIsSiteValidForBuild ( inMessageID , outLocation ) ) )
// :tankefugl
{
theSuccess = true ;
}
}
return theSuccess ;
}
# ifdef AVH_CLIENT
bool AvHSHUGetEntityAtRay ( const Vector & inPointOfView , const Vector & inNormRay , int & outEntIndex )
{
bool theSuccess = false ;
// Offset starting position a little so we don't select ourselves
Vector theStartPosition ;
VectorMA ( inPointOfView , kSelectionStartRange , inNormRay , theStartPosition ) ;
Vector theEndPos ;
VectorMA ( inPointOfView , kSelectionEndRange , inNormRay , theEndPos ) ;
gEngfuncs . pEventAPI - > EV_SetUpPlayerPrediction ( false , true ) ;
// Store off the old count
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
// Now add in all of the players.
int theNumPlayers = gEngfuncs . GetMaxClients ( ) ;
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
bool theDone = false ;
int theEntityIndex = - 1 ;
vec3_t theNewStartPos ;
AvHSHUGetFirstNonSolidPoint ( theStartPosition , theEndPos , theNewStartPos ) ;
VectorCopy ( theNewStartPos , theStartPosition ) ;
do
{
pmtrace_t tr ;
gEngfuncs . pEventAPI - > EV_SetTraceHull ( 2 ) ;
gEngfuncs . pEventAPI - > EV_PlayerTrace ( theStartPosition , theEndPos , PM_NORMAL , theEntityIndex , & tr ) ;
physent_t * pEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( tr . ent ) ;
theEntityIndex = gEngfuncs . pEventAPI - > EV_IndexFromTrace ( & tr ) ;
if ( pEntity )
{
int theUser3 = pEntity - > iuser3 ;
if ( theEntityIndex > 0 )
{
outEntIndex = theEntityIndex ;
theSuccess = true ;
theDone = true ;
}
}
if ( ( tr . fraction > = ( 1.0f - kFloatTolerance ) ) | | ( tr . fraction < kFloatTolerance ) )
{
theDone = true ;
}
else
{
VectorCopy ( tr . endpos , theStartPosition ) ;
}
} while ( ! theDone ) ;
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
return theSuccess ;
}
# endif
# ifdef AVH_SERVER
bool AvHSHUGetEntityAtRay ( const Vector & inPointOfView , const Vector & inNormRay , int & outEntIndex )
{
TraceResult tr ;
Vector theStartPos ;
Vector theEndPos ;
bool theSuccess = false ;
bool theDone = false ;
VectorMA ( inPointOfView , kSelectionStartRange , inNormRay , theStartPos ) ;
VectorMA ( inPointOfView , kSelectionEndRange , inNormRay , theEndPos ) ;
CBaseEntity * theEntityHit = NULL ;
edict_t * theEdictToIgnore = NULL ;
vec3_t theNewStartPos ;
AvHSHUGetFirstNonSolidPoint ( theStartPos , theEndPos , theNewStartPos ) ;
VectorCopy ( theNewStartPos , theStartPos ) ;
do
{
UTIL_TraceLine ( theStartPos , theEndPos , dont_ignore_monsters , ignore_glass , theEdictToIgnore , & tr ) ;
theEntityHit = CBaseEntity : : Instance ( tr . pHit ) ;
if ( theEntityHit )
{
if ( theEntityHit - > entindex ( ) > 0 )
{
outEntIndex = theEntityHit - > entindex ( ) ;
theSuccess = true ;
theDone = true ;
}
}
if ( ( tr . flFraction > ( 1.0f - kFloatTolerance ) ) | | ( tr . flFraction < kFloatTolerance ) )
{
theDone = true ;
}
else
{
if ( theEntityHit )
{
theEdictToIgnore = ENT ( theEntityHit - > pev ) ;
}
VectorCopy ( tr . vecEndPos , theStartPos ) ;
}
} while ( ! theDone ) ;
return theSuccess ;
}
# endif
const AvHMapExtents & AvHSHUGetMapExtents ( )
{
# ifdef AVH_CLIENT
const AvHMapExtents & theMapExtents = gHUD . GetMapExtents ( ) ;
# endif
# ifdef AVH_SERVER
const AvHMapExtents & theMapExtents = GetGameRules ( ) - > GetMapExtents ( ) ;
# endif
return theMapExtents ;
}
# ifdef AVH_CLIENT
bool AvHSUClientTraceTangible ( const vec3_t & inStartPos , const vec3_t & inEndPos , int & outIndex , vec3_t & outLocation , AvHTeamNumber & outTeamNumber , bool & outPlayerWasHit , int & outUserThree , int & outUserFour )
{
physent_t * theEntity = NULL ;
vec3_t theStartPos ;
vec3_t theEndPos ;
int theFoundEntity = - 1 ;
bool theSuccess = false ;
bool theDone = false ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inEndPos , theEndPos ) ;
vec3_t theNormal ;
VectorSubtract ( inEndPos , inStartPos , theNormal ) ;
VectorNormalize ( theNormal ) ;
outIndex = - 1 ;
gEngfuncs . pEventAPI - > EV_SetUpPlayerPrediction ( false , true ) ;
// Store off the old count
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
// Now add in all of the players.
int theNumPlayers = gEngfuncs . GetMaxClients ( ) ;
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
vec3_t theNewStartPos ;
AvHSHUGetFirstNonSolidPoint ( theStartPos , theEndPos , theNewStartPos ) ;
VectorCopy ( theNewStartPos , theStartPos ) ;
do
{
pmtrace_t tr ;
gEngfuncs . pEventAPI - > EV_SetTraceHull ( 2 ) ;
gEngfuncs . pEventAPI - > EV_PlayerTrace ( theStartPos , theEndPos , PM_NORMAL /*PM_GLASS_IGNORE*/ , theFoundEntity , & tr ) ;
physent_t * pEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( tr . ent ) ;
theFoundEntity = gEngfuncs . pEventAPI - > EV_IndexFromTrace ( & tr ) ;
if ( pEntity )
{
if ( pEntity - > iuser3 ! = AVH_USER3_NONE )
{
VectorCopy ( tr . endpos , outLocation ) ;
outIndex = theFoundEntity ;
theEntity = pEntity ;
outUserThree = pEntity - > iuser3 ;
outUserFour = pEntity - > iuser4 ;
outTeamNumber = ( AvHTeamNumber ) ( pEntity - > team ) ;
if ( pEntity - > player )
{
outPlayerWasHit = true ;
}
theSuccess = true ;
theDone = true ;
}
}
if ( tr . fraction > = ( 1.0f - kFloatTolerance ) )
{
theDone = true ;
}
else
{
vec3_t theDiff = theNormal * kHitOffsetAmount ;
float theLength = sqrt ( theDiff . x * theDiff . x + theDiff . y * theDiff . y + theDiff . z * theDiff . z ) ;
if ( theLength > kFloatTolerance )
{
VectorAdd ( tr . endpos , theDiff , theStartPos ) ;
}
else
{
theDone = true ;
}
// Offset a bit so we don't hit again
//VectorMA(tr.endpos, kHitOffsetAmount, theNormal, theStartPos);
}
} while ( ! theDone ) ;
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
// If we didn't hit any special targets, see if it's a valid area to build or be ordered to
if ( ! theSuccess )
{
WaypointReturnCode theReturnCode = WAYPOINT_SUCCESS ;
theSuccess = AvHSHUClientTraceWaypoint ( inStartPos , inEndPos , & outLocation , & theReturnCode ) ;
bool theWaypointTooSteep = ( theReturnCode = = WAYPOINT_TOOSTEEP ) ;
if ( theSuccess | | theWaypointTooSteep )
{
outUserThree = AVH_USER3_WAYPOINT ;
outPlayerWasHit = false ;
outIndex = - 1 ;
outTeamNumber = TEAM_IND ;
theSuccess = true ;
}
// Treat too steep as success, but mark it as nobuild
if ( theWaypointTooSteep )
{
outUserThree = AVH_USER3_NOBUILD ;
}
}
return theSuccess ;
}
# endif
# ifdef AVH_SERVER
bool AvHSUServerTraceTangible ( const vec3_t & inStartPos , const vec3_t & inEndPos , int & outIndex , vec3_t & outLocation , AvHTeamNumber & outTeamNumber , bool & outPlayerWasHit , int & outUserThree , int & outUserFour )
{
bool theSuccess = false ;
bool theDone = false ;
edict_t * theEdictToIgnore = NULL ;
vec3_t theStartPos ;
vec3_t theEndPos ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inEndPos , theEndPos ) ;
vec3_t theNormal ;
VectorSubtract ( inEndPos , inStartPos , theNormal ) ;
VectorNormalize ( theNormal ) ;
// vec3_t theNewStartPos;
// AvHSHUGetFirstNonSolidPoint(theStartPos, theEndPos, theNewStartPos);
// VectorCopy(theNewStartPos, theStartPos);
do
{
TraceResult tr ;
UTIL_TraceLine ( theStartPos , theEndPos , dont_ignore_monsters , theEdictToIgnore , & tr ) ;
CBaseEntity * theEntityHit = NULL ;
// Return the entity in special way
if ( tr . flFraction < 1 & & tr . pHit )
{
theEntityHit = CBaseEntity : : Instance ( tr . pHit ) ;
AvHPlayer * thePlayer = dynamic_cast < AvHPlayer * > ( theEntityHit ) ;
AvHWeldable * theWeldable = dynamic_cast < AvHWeldable * > ( theEntityHit ) ;
AvHBasePlayerWeapon * theWeapon = dynamic_cast < AvHBasePlayerWeapon * > ( theEntityHit ) ;
if ( AvHSUGetIsDebugging ( ) )
{
// Create an entity where the trace hit
Vector theAngles ( 0 , 0 , 0 ) ;
CBaseEntity * pEnt = CBaseEntity : : Create ( kwsDebugEntity , tr . vecEndPos , theAngles ) ;
ASSERT ( pEnt ) ;
pEnt - > pev - > movetype = MOVETYPE_FLY ;
pEnt - > pev - > solid = SOLID_NOT ;
}
if ( theEntityHit & & ( theEntityHit - > pev - > iuser3 ! = AVH_USER3_NONE ) )
{
// Don't hit seethroughs
if ( theEntityHit - > pev - > iuser3 ! = AVH_USER3_ALPHA )
{
outIndex = ENTINDEX ( tr . pHit ) ;
VectorCopy ( tr . vecEndPos , outLocation ) ;
outTeamNumber = ( AvHTeamNumber ) theEntityHit - > pev - > team ;
outUserThree = theEntityHit - > pev - > iuser3 ;
outUserFour = theEntityHit - > pev - > iuser4 ;
if ( thePlayer )
{
outPlayerWasHit = true ;
}
theSuccess = true ;
theDone = true ;
}
}
theEdictToIgnore = theEntityHit - > edict ( ) ;
}
if ( ( tr . flFraction > = ( 1.0f - kFloatTolerance ) ) | | ( tr . flFraction < kFloatTolerance ) )
{
theDone = true ;
}
else
{
vec3_t theDiff = theNormal * kHitOffsetAmount ;
float theLength = sqrt ( theDiff . x * theDiff . x + theDiff . y * theDiff . y + theDiff . z * theDiff . z ) ;
if ( theLength > kFloatTolerance )
{
VectorAdd ( theStartPos , theDiff , theStartPos ) ;
}
else
{
theDone = true ;
}
// Offset a bit so we don't hit again
//VectorMA(tr.vecEndPos, kHitOffsetAmount, theNormal, theStartPos);
}
} while ( ! theDone ) ;
if ( ! theSuccess )
{
WaypointReturnCode theReturnCode = WAYPOINT_SUCCESS ;
theSuccess = AvHSHUServerTraceWaypoint ( inStartPos , inEndPos , & outLocation , & theReturnCode ) ;
bool theWaypointTooSteep = ( theReturnCode = = WAYPOINT_TOOSTEEP ) ;
if ( theSuccess | | theWaypointTooSteep )
{
outUserThree = AVH_USER3_WAYPOINT ;
outPlayerWasHit = false ;
outIndex = - 1 ;
outTeamNumber = TEAM_IND ;
theSuccess = true ;
}
// Treat too steep as success, but mark it as nobuild
if ( theWaypointTooSteep )
{
outUserThree = AVH_USER3_NOBUILD ;
}
}
return theSuccess ;
}
# endif
# ifdef AVH_CLIENT
bool AvHSHUClientTraceWaypoint ( const vec3_t & inStartPos , const vec3_t & inEndPos , vec3_t * outLocation , WaypointReturnCode * outReturnCode )
{
vec3_t theStartPos ;
vec3_t theEndPos ;
int theFoundEntity = - 1 ;
int theEntityToIgnore = - 1 ;
bool theDone = false ;
bool theLegalToBuild = false ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inEndPos , theEndPos ) ;
vec3_t theNormal ;
VectorSubtract ( inEndPos , inStartPos , theNormal ) ;
VectorNormalize ( theNormal ) ;
int theNumPlayers = gEngfuncs . GetMaxClients ( ) ;
pmtrace_t tr ;
//DebugPoint theDebugPoint(theStartPos[0], theStartPos[1], theStartPos[2]);
//gSquareDebugLocations.push_back(theDebugPoint);
gEngfuncs . pEventAPI - > EV_SetUpPlayerPrediction ( false , true ) ;
gEngfuncs . pEventAPI - > EV_PushPMStates ( ) ;
gEngfuncs . pEventAPI - > EV_SetSolidPlayers ( - 1 ) ;
gEngfuncs . pEventAPI - > EV_SetTraceHull ( 2 ) ;
vec3_t theNewStartPos ;
AvHSHUGetFirstNonSolidPoint ( theStartPos , theEndPos , theNewStartPos ) ;
VectorCopy ( theNewStartPos , theStartPos ) ;
do
{
gEngfuncs . pEventAPI - > EV_PlayerTrace ( theStartPos , theEndPos , PM_NORMAL /*PM_GLASS_IGNORE*/ , theEntityToIgnore , & tr ) ;
physent_t * pEntity = gEngfuncs . pEventAPI - > EV_GetPhysent ( tr . ent ) ;
theFoundEntity = gEngfuncs . pEventAPI - > EV_IndexFromTrace ( & tr ) ;
// Trace until we hit a worldbrush or non-seethrough
if ( ! pEntity | | ( pEntity - > iuser3 ! = AVH_USER3_ALPHA ) )
{
// If entity is a "no waypoint" entity we can't build here and we're done
if ( pEntity & & ( pEntity - > iuser3 = = AVH_USER3_NOBUILD ) )
{
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_NOBUILD ;
}
theDone = true ;
}
else
{
// else if texture is NOBUILD, we're done
const char * theTextureHitCStr = gEngfuncs . pEventAPI - > EV_TraceTexture ( theFoundEntity , tr . endpos , theEndPos ) ;
if ( theTextureHitCStr & & ( LowercaseString ( theTextureHitCStr ) = = LowercaseString ( kNoBuildTexture ) ) )
{
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_NOBUILD ;
}
theDone = true ;
}
else if ( theTextureHitCStr & & ( LowercaseString ( theTextureHitCStr ) = = LowercaseString ( kSeeThroughTexture ) ) )
{
// Not valid, but allow it to pass through
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_SEETHROUGH ;
}
}
else
{
// Trace texture sometimes seems to miss entities that the start point lies within. Don't count
// the trace texture if it found the texture of the next entity below it
// else if surface is more flat than vertical
if ( tr . plane . normal [ 2 ] > = 0.7f )
{
// and if surface isn't under water, lava, in the sky, etc.
int thePointContents = gEngfuncs . PM_PointContents ( tr . endpos , NULL ) ;
if ( thePointContents = = CONTENTS_EMPTY )
{
// and if there's enough room to build
// we can build here, and we're done
theLegalToBuild = true ;
theDone = true ;
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_SUCCESS ;
}
}
else
{
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_CONTENTSFULL ;
theDone = true ;
}
}
}
else
{
if ( outReturnCode )
{
if ( theTextureHitCStr )
{
* outReturnCode = WAYPOINT_TOOSTEEP ;
}
else
{
* outReturnCode = WAYPOINT_ENTITYHIT ;
}
theDone = true ;
}
}
}
}
}
if ( theFoundEntity ! = 0 )
{
theEntityToIgnore = theFoundEntity ;
}
if ( ( ( tr . fraction > = ( 1.0f - kFloatTolerance ) ) | | ( tr . fraction = = 0.0f ) ) /*&& !tr.startsolid && !tr.allsolid*/ )
{
theDone = true ;
}
else
{
vec3_t theDiff = theNormal * kHitOffsetAmount ;
float theLength = sqrt ( theDiff . x * theDiff . x + theDiff . y * theDiff . y + theDiff . z * theDiff . z ) ;
if ( theLength > kFloatTolerance )
{
VectorAdd ( tr . endpos , theDiff , theStartPos ) ;
}
else
{
theDone = true ;
}
// Offset a bit so we don't hit again
//VectorMA(tr.endpos, kHitOffsetAmount, theNormal, theStartPos);
}
} while ( ! theDone ) ;
gEngfuncs . pEventAPI - > EV_PopPMStates ( ) ;
// Always set end location to show where invalid position is
* outLocation = tr . endpos ;
return theLegalToBuild ;
}
# endif
# ifdef AVH_CLIENT
void AvHSHUClientGetFirstNonSolidPoint ( const vec3_t & inStartPos , const vec3_t & inEndPos , vec3_t & outNonSolidPoint )
{
vec3_t theStartPos ;
vec3_t theEndPos ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inStartPos , outNonSolidPoint ) ;
VectorCopy ( inEndPos , theEndPos ) ;
pmtrace_t tr ;
gEngfuncs . pEventAPI - > EV_PlayerTrace ( theStartPos , theEndPos , PM_NORMAL , - 1 , & tr ) ;
// Put this workaround in because a bug in EV_PlayerTrace means that when it starts solid, tr.fraction isn't returned (but it is in UTIL_TraceLine)
if ( ( tr . startsolid ) & & ( tr . fraction = = 0.0f ) )
{
int theFoundEntity = - 1 ;
bool theDone = false ;
vec3_t theStartToEnd ;
VectorSubtract ( inEndPos , inStartPos , theStartToEnd ) ;
gEngfuncs . pEventAPI - > EV_SetTraceHull ( 2 ) ;
float theIncrement = 10.0f / theStartToEnd . Length ( ) ;
float theT = 0.0f ;
int theNumIterations = 0 ;
do
{
theStartPos = inStartPos + theT * theStartToEnd ;
gEngfuncs . pEventAPI - > EV_PlayerTrace ( theStartPos , theEndPos , PM_WORLD_ONLY , - 1 , & tr ) ;
theNumIterations + + ;
// If start point is solid, bisect area and move start point 1/2 between current start and current end
if ( tr . startsolid )
{
theT + = theIncrement ;
}
// else if start point isn't solid, bisect area and move start point back towards original start point
else
{
theDone = true ;
}
} while ( ! theDone & & ( theNumIterations < 200 ) ) ;
// Always set end location to show where invalid position is
if ( ! theDone )
{
outNonSolidPoint = inStartPos ;
}
else
{
outNonSolidPoint = theStartPos ;
}
}
}
# endif
# ifdef AVH_SERVER
void AvHSHUServerGetFirstNonSolidPoint ( const vec3_t & inStartPos , const vec3_t & inEndPos , vec3_t & outNonSolidPoint )
{
vec3_t theStartPos ;
vec3_t theEndPos ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inStartPos , outNonSolidPoint ) ;
VectorCopy ( inEndPos , theEndPos ) ;
TraceResult tr ;
UTIL_TraceLine ( theStartPos , theEndPos , ignore_monsters , NULL , & tr ) ;
bool theDone = false ;
// Put this workaround in because a bug in EV_PlayerTrace means that when it starts solid, tr.fraction isn't returned (but it is in UTIL_TraceLine)
if ( ( tr . fStartSolid ) | | ( tr . flFraction = = 0.0f ) )
{
int theFoundEntity = - 1 ;
vec3_t theStartToEnd ;
VectorSubtract ( inEndPos , inStartPos , theStartToEnd ) ;
float theIncrement = 10.0f / theStartToEnd . Length ( ) ;
float theT = 0.0f ;
int theNumIterations = 0 ;
int thePointContents = 0 ;
do
{
theStartPos = inStartPos + theT * theStartToEnd ;
thePointContents = UTIL_PointContents ( theStartPos ) ;
theNumIterations + + ;
// If start point is solid, bisect area and move start point 1/2 between current start and current end
if ( ( thePointContents = = CONTENTS_SOLID ) | | ( thePointContents = = CONTENTS_SKY ) )
{
theT + = theIncrement ;
}
// else if start point isn't solid, bisect area and move start point back towards original start point
else
{
theDone = true ;
}
} while ( ! theDone & & ( theNumIterations < 200 ) ) ;
// Always set end location to show where invalid position is
outNonSolidPoint = theStartPos ;
}
else
{
VectorCopy ( tr . vecEndPos , outNonSolidPoint ) ;
theDone = true ;
}
if ( ! theDone )
{
// When we don't hit anything, return start point
VectorCopy ( inStartPos , outNonSolidPoint ) ;
}
VectorCopy ( outNonSolidPoint , gPMDebugPoint ) ;
}
# endif
void AvHSHUGetFirstNonSolidPoint ( float * inStartPos , float * inEndPos , float * outNonSolidPoint )
{
vec3_t theStartPos ;
vec3_t theEndPos ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inEndPos , theEndPos ) ;
vec3_t theNonSolidPoint ;
AvHSHUGetFirstNonSolidPoint ( theStartPos , theEndPos , theNonSolidPoint ) ;
VectorCopy ( theNonSolidPoint , outNonSolidPoint ) ;
}
void AvHSHUGetFirstNonSolidPoint ( const vec3_t & inStartPos , const vec3_t & inEndPos , vec3_t & outNonSolidPoint )
{
# ifdef AVH_CLIENT
AvHSHUClientGetFirstNonSolidPoint ( inStartPos , inEndPos , outNonSolidPoint ) ;
//extern DebugPointListType gSquareDebugLocations;
//DebugPoint theDebugPoint(outNonSolidPoint[0], outNonSolidPoint[1], outNonSolidPoint[2]);
//gSquareDebugLocations.push_back(theDebugPoint);
# endif
# ifdef AVH_SERVER
AvHSHUServerGetFirstNonSolidPoint ( inStartPos , inEndPos , outNonSolidPoint ) ;
# endif
}
# ifdef AVH_SERVER
bool AvHSHUServerTraceWaypoint ( const vec3_t & inStartPos , const vec3_t & inEndPos , vec3_t * outLocation , WaypointReturnCode * outReturnCode )
{
bool theLegalToBuild = false ;
bool theDone = false ;
edict_t * theEdictToIgnore = NULL ;
TraceResult tr ;
vec3_t theStartPos ;
vec3_t theEndPos ;
VectorCopy ( inStartPos , theStartPos ) ;
VectorCopy ( inEndPos , theEndPos ) ;
vec3_t theNormal ;
VectorSubtract ( inEndPos , inStartPos , theNormal ) ;
VectorNormalize ( theNormal ) ;
vec3_t theNewStartPos ;
AvHSHUGetFirstNonSolidPoint ( theStartPos , theEndPos , theNewStartPos ) ;
VectorCopy ( theNewStartPos , theStartPos ) ;
do
{
UTIL_TraceLine ( theStartPos , theEndPos , dont_ignore_monsters , theEdictToIgnore , & tr ) ;
// Trace until we hit a worldbrush or non-seethrough
CBaseEntity * theEntityHit = CBaseEntity : : Instance ( tr . pHit ) ;
CWorld * theWorld = dynamic_cast < CWorld * > ( theEntityHit ) ;
bool theHitSeethrough = ( theEntityHit - > pev - > iuser3 = = AVH_USER3_ALPHA ) ;
if ( ! theHitSeethrough )
{
// If entity is a "no waypoint" entity we can't build here and we're done
if ( theEntityHit & & ( theEntityHit - > pev - > iuser3 = = AVH_USER3_NOBUILD ) )
{
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_NOBUILD ;
}
theDone = true ;
}
else
{
edict_t * theEdict = ENT ( theEntityHit - > pev ) ;
const char * theTextureHitCStr = TRACE_TEXTURE ( theEdict , tr . vecEndPos , theEndPos ) ;
if ( theTextureHitCStr & & ( LowercaseString ( theTextureHitCStr ) = = LowercaseString ( kNoBuildTexture ) ) )
{
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_NOBUILD ;
}
theDone = true ;
}
else if ( theTextureHitCStr & & ( LowercaseString ( theTextureHitCStr ) = = LowercaseString ( kSeeThroughTexture ) ) )
{
// Do nothing, but allow it to pass through
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_SEETHROUGH ;
}
}
else
{
// else if surface is more flat than vertical
if ( tr . vecPlaneNormal . z > = .7f )
{
// and if surface isn't under water, lava, sky
int thePointContents = UTIL_PointContents ( tr . vecEndPos ) ;
if ( thePointContents = = CONTENTS_EMPTY )
{
// and if there's enough room to build
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_SUCCESS ;
}
// we can build here, and we're done
theLegalToBuild = true ;
theDone = true ;
}
else
{
if ( outReturnCode )
{
* outReturnCode = WAYPOINT_CONTENTSFULL ;
}
}
}
else
{
if ( theTextureHitCStr )
{
* outReturnCode = WAYPOINT_TOOSTEEP ;
}
else
{
* outReturnCode = WAYPOINT_ENTITYHIT ;
}
theDone = true ;
}
}
}
if ( ! theWorld )
{
theEdictToIgnore = theEntityHit - > edict ( ) ;
}
if ( ( ( tr . flFraction > = ( 1.0f - kFloatTolerance ) ) | | ( tr . flFraction < kFloatTolerance ) ) /*&& !tr.fStartSolid && !tr.fAllSolid*/ )
{
theDone = true ;
}
else
{
vec3_t theDiff = theNormal * kHitOffsetAmount ;
float theLength = sqrt ( theDiff . x * theDiff . x + theDiff . y * theDiff . y + theDiff . z * theDiff . z ) ;
if ( theLength > kFloatTolerance )
{
//VectorAdd(tr.vecEndPos, theDiff, theStartPos);
VectorAdd ( theStartPos , theDiff , theStartPos ) ;
}
else
{
theDone = true ;
}
// Offset a bit so we don't hit again
//VectorMA(tr.vecEndPos, kHitOffsetAmount, theNormal, theStartPos);
}
}
else
{
theEdictToIgnore = theEntityHit - > edict ( ) ;
if ( tr . vecEndPos = = theStartPos )
{
vec3_t theDiff = theNormal * kHitOffsetAmount ;
float theLength = sqrt ( theDiff . x * theDiff . x + theDiff . y * theDiff . y + theDiff . z * theDiff . z ) ;
if ( theLength > kFloatTolerance )
{
//VectorAdd(tr.vecEndPos, theDiff, theStartPos);
VectorAdd ( theStartPos , theDiff , theStartPos ) ;
}
}
else
{
VectorCopy ( tr . vecEndPos , theStartPos ) ;
}
// vec3_t theEntityPosition = AvHSHUGetRealLocation(theEntityHit->pev->origin, theEntityHit->pev->mins, theEntityHit->pev->maxs);
// //VectorCopy(theEntityPosition, theStartPos);
// theStartPos[2] = theEntityPosition[2];
}
} while ( ! theDone ) ;
// Always set end location to show where invalid position is
* outLocation = tr . vecEndPos ;
return theLegalToBuild ;
}
# endif
bool AvHSHUTraceTangible ( const vec3_t & inStartPos , const vec3_t & inEndPos , int & outIndex , vec3_t & outLocation , AvHTeamNumber & outTeamNumber , bool & outPlayerWasHit , int & outUserThree , int & outUserFour )
{
bool theSuccess = false ;
outPlayerWasHit = false ;
outUserThree = 0 ;
outUserFour = 0 ;
// On client, set up players for prediction and do a PM_TraceLine
# ifdef AVH_CLIENT
theSuccess = AvHSUClientTraceTangible ( inStartPos , inEndPos , outIndex , outLocation , outTeamNumber , outPlayerWasHit , outUserThree , outUserFour ) ;
# endif
// On the server, do a UTIL_TraceLine
# ifdef AVH_SERVER
theSuccess = AvHSUServerTraceTangible ( inStartPos , inEndPos , outIndex , outLocation , outTeamNumber , outPlayerWasHit , outUserThree , outUserFour ) ;
# endif
return theSuccess ;
}
bool AvHSHUTraceTangible ( const Vector & inPointOfView , const Vector & inNormRay , int * outUserThree , vec3_t * outLocation , AvHTeamNumber * outTeamNumber , bool * outPlayerWasHit )
{
bool theSuccess = false ;
vec3_t theTraceStart ;
vec3_t theTraceEnd ;
vec3_t theFoundLocation ;
int theFoundIndex = - 1 ;
int theUserThree = 0 ;
int theUserFour = 0 ;
AvHTeamNumber theTeamOfThingHit ;
bool thePlayerHit = false ;
// Offset a little so we don't hit the commander
VectorMA ( inPointOfView , kSelectionStartRange , inNormRay , theTraceStart ) ;
VectorMA ( inPointOfView , kSelectionEndRange , inNormRay , theTraceEnd ) ;
if ( ! outUserThree )
outUserThree = & theUserThree ;
if ( ! outLocation )
outLocation = & theFoundLocation ;
if ( ! outTeamNumber )
outTeamNumber = & theTeamOfThingHit ;
if ( ! outPlayerWasHit )
outPlayerWasHit = & thePlayerHit ;
theSuccess = AvHSHUTraceTangible ( theTraceStart , theTraceEnd , theFoundIndex , * outLocation , * outTeamNumber , * outPlayerWasHit , * outUserThree , theUserFour ) ;
return theSuccess ;
}
bool AvHSHUTraceVerticalTangible ( float inX , float inY , float inZ , int & outUserThree , float & outHeight )
{
bool theSuccess = false ;
// Create a start pos above
vec3_t theStartPos ( inX , inY , inZ - 100.0f ) ;
//vec3_t theEndPos(inX, inY, inZ - 800.0f/*-4096.0f*/);
vec3_t theEndPos ( inX , inY , ( float ) kWorldMinimumZ ) ;
// #ifdef AVH_CLIENT
// extern DebugPointListType gTriDebugLocations;
// bool theDrawDebug = true;//(rand() % 100 == 0);
//
// if(theDrawDebug)
// {
// DebugPoint thePoint;
// thePoint.x = theStartPos.x;
// thePoint.y = theStartPos.y;
// thePoint.z = theStartPos.z;
// gTriDebugLocations.push_back(thePoint);
//
// thePoint.z = theEndPos.z;
// gTriDebugLocations.push_back(thePoint);
// }
//
// #endif
int theIndex ;
vec3_t theEndLocation ;
AvHTeamNumber theTeam ;
bool thePlayerHit ;
int theUserFour ;
theSuccess = AvHSHUTraceTangible ( theStartPos , theEndPos , theIndex , theEndLocation , theTeam , thePlayerHit , outUserThree , theUserFour ) ;
if ( theSuccess )
{
outHeight = theEndLocation . z ;
ASSERT ( outHeight < = inZ ) ;
}
return theSuccess ;
}
const char * AvHSHUGetCommonSoundName ( bool inIsAlien , WeaponHUDSound inHUDSound )
{
const char * theSoundName = NULL ;
switch ( inHUDSound )
{
case WEAPON_SOUND_HUD_ON :
theSoundName = inIsAlien ? " common/wpn_hudon-a.wav " : " common/wpn_hudon.wav " ;
break ;
case WEAPON_SOUND_HUD_OFF :
theSoundName = inIsAlien ? " common/wpn_hudoff-a.wav " : " common/wpn_hudoff.wav " ;
break ;
case WEAPON_SOUND_MOVE_SELECT :
theSoundName = inIsAlien ? " common/wpn_moveselect-a.wav " : " common/wpn_moveselect.wav " ;
break ;
case WEAPON_SOUND_SELECT :
theSoundName = inIsAlien ? " common/wpn_select-a.wav " : " common/wpn_select.wav " ;
break ;
case WEAPON_SOUND_DENYSELECT :
theSoundName = inIsAlien ? " common/wpn_denyselect-a.wav " : " common/wpn_denyselect.wav " ;
break ;
}
return theSoundName ;
}
// Values for fuser1: 0-1 means it's building. 2-3 means it's researching (see kResearchFuser1Base), 3-4 is "energy" (returns both building and researching)
const float kResearchFuser1Base = 2.0f ;
const float kEnergyFuser1Base = 3.0f ;
void AvHSHUGetBuildResearchState ( int inUser3 , int inUser4 , float inFuser1 , bool & outIsBuilding , bool & outIsResearching , float & outNormalizedPercentage )
{
outIsBuilding = outIsResearching = false ;
float theStatus = ( inFuser1 / kNormalizationNetworkFactor ) ;
if ( ( GetHasUpgrade ( inUser4 , MASK_BUILDABLE ) | | ( inUser3 = = AVH_USER3_WELD ) ) & & ! GetHasUpgrade ( inUser4 , MASK_RECYCLING ) )
{
if ( ( theStatus > = 0.0f ) & & ( theStatus < = 1.0f ) )
{
outNormalizedPercentage = theStatus ;
outIsBuilding = true ;
}
if ( inUser3 = = AVH_USER3_WELD )
{
if ( outNormalizedPercentage = = - 1 )
{
outIsBuilding = outIsResearching = false ;
}
}
}
else
{
if ( ( theStatus > kResearchFuser1Base ) & & ( theStatus < = ( 1.0f + kResearchFuser1Base ) ) )
{
outNormalizedPercentage = theStatus - kResearchFuser1Base ;
outIsResearching = true ;
}
else if ( ( theStatus > kEnergyFuser1Base ) & & ( theStatus < = ( 1.0f + kEnergyFuser1Base ) ) )
{
// Set "energy"
outNormalizedPercentage = theStatus - kEnergyFuser1Base ;
outIsBuilding = outIsResearching = true ;
}
else
{
outNormalizedPercentage = 1.0f ;
}
}
}
void AvHSHUSetEnergyState ( int inUser3 , float & outFuser1 , float inNormalizedPercentage )
{
outFuser1 = ( kEnergyFuser1Base + inNormalizedPercentage ) * kNormalizationNetworkFactor ;
}
void AvHSHUSetBuildResearchState ( int inUser3 , int inUser4 , float & outFuser1 , bool inTrueBuildOrFalseResearch , float inNormalizedPercentage )
{
//ASSERT(GetHasUpgrade(inUser4, MASK_BUILDABLE) /*|| GetHasUpgrade(inUser4, MASK_SELECTABLE)*/ || (inUser3 == AVH_USER3_WELD));
if ( inNormalizedPercentage ! = - 1 )
{
inNormalizedPercentage = min ( inNormalizedPercentage , 1.0f ) ;
inNormalizedPercentage = max ( inNormalizedPercentage , 0.0f ) ;
}
// Build
if ( inTrueBuildOrFalseResearch )
{
//ASSERT(GetHasUpgrade(inUser4, MASK_BUILDABLE) || (inUser3 == AVH_USER3_WELD));
outFuser1 = inNormalizedPercentage * kNormalizationNetworkFactor ;
}
// Research
else
{
outFuser1 = ( kResearchFuser1Base + inNormalizedPercentage ) * kNormalizationNetworkFactor ;
}
}
string AvHSHUBuildExecutableScriptName ( const string & inScriptName , const string & inCurrentMapName )
{
string theExecutableScriptName = string ( getModDirectory ( ) ) + string ( " / " ) + kScriptsDirectory + string ( " / " ) + inCurrentMapName + string ( " / " ) + inScriptName ;
return theExecutableScriptName ;
}
string AvHSHUGetTimeDateString ( )
{
string theTimeDateString ( " <date unknown> " ) ;
time_t theTimeThingie = time ( ( time_t * ) NULL ) ;
struct tm * theTime = localtime ( & theTimeThingie ) ;
if ( theTime )
{
// 04/22/03, 16:59:12
char theBuffer [ 512 ] ;
sprintf ( theBuffer , " %.2d/%.2d/%.2d, %.2d:%.2d:%.2d " , ( theTime - > tm_mon + 1 ) , theTime - > tm_mday , ( theTime - > tm_year - 100 ) , theTime - > tm_hour , theTime - > tm_min , theTime - > tm_sec ) ;
theTimeDateString = theBuffer ;
}
return theTimeDateString ;
}
bool AvHSHUGetNameOfLocation ( const AvHBaseInfoLocationListType & inLocations , vec3_t inLocation , string & outLocation )
{
bool theSuccess = false ;
// Look at our current position, and see if we lie within of the map locations
for ( AvHBaseInfoLocationListType : : const_iterator theIter = inLocations . begin ( ) ; theIter ! = inLocations . end ( ) ; theIter + + )
{
if ( theIter - > GetIsPointInRegion ( inLocation ) )
{
outLocation = theIter - > GetLocationName ( ) ;
theSuccess = true ;
break ;
}
}
return theSuccess ;
}
// Also check AvHSUGetAlwaysPlayAlert to make sure some alerts are always triggered
bool AvHSHUGetForceHUDSound ( AvHHUDSound inHUDSound )
{
bool theForceSound = false ;
switch ( inHUDSound )
{
case HUD_SOUND_ALIEN_HIVE_COMPLETE :
case HUD_SOUND_ALIEN_HIVE_DYING :
case HUD_SOUND_ALIEN_HIVE_ATTACK :
case HUD_SOUND_MARINE_CCUNDERATTACK :
case HUD_SOUND_SQUAD1 :
case HUD_SOUND_SQUAD2 :
case HUD_SOUND_SQUAD3 :
case HUD_SOUND_SQUAD4 :
case HUD_SOUND_SQUAD5 :
theForceSound = true ;
break ;
}
return theForceSound ;
}