mirror of
https://github.com/ENSL/NS.git
synced 2025-01-26 19:00:58 +00:00
df34e24de5
git-svn-id: https://unknownworlds.svn.cloudforge.com/ns1@506 67975925-1194-0748-b3d5-c16f83f1a3a1
3629 lines
120 KiB
C++
3629 lines
120 KiB
C++
//======== (C) Copyright 2001 Charles G. Cleveland All rights reserved. =========
|
|
//
|
|
// The copyright to the contents herein is the property of Charles G. Cleveland.
|
|
// The contents may be used and/or copied only with the written permission of
|
|
// Charles G. Cleveland, or in accordance with the terms and conditions stipulated in
|
|
// the agreement/contract under which the contents have been supplied.
|
|
//
|
|
// Purpose: AvH event declarations
|
|
//
|
|
// $Workfile: AvHEvents.cpp$
|
|
// $Date: 2002/10/24 21:23:37 $
|
|
//
|
|
//-------------------------------------------------------------------------------
|
|
// $Log: AvHEvents.cpp,v $
|
|
// Revision 1.47 2002/10/24 21:23:37 Flayra
|
|
// - Reworked jetpacks
|
|
// - Removed bullet casing from grenade launcher
|
|
//
|
|
// Revision 1.46 2002/10/16 20:52:34 Flayra
|
|
// - Removed weapon upgrade sounds
|
|
// - Fixed acid projectile hitting player
|
|
// - Play HMG firing at full volume
|
|
// - Added paralysis projectile
|
|
// - Fixed spike orientation
|
|
//
|
|
// Revision 1.45 2002/10/16 00:54:44 Flayra
|
|
// - Removed unneeded events
|
|
// - Commented out broken parasite projectile
|
|
// - Added general particle event
|
|
// - Added distress beacon event
|
|
//
|
|
// Revision 1.44 2002/09/25 20:43:53 Flayra
|
|
// - Effects update
|
|
//
|
|
// Revision 1.43 2002/09/23 22:13:38 Flayra
|
|
// - Fixed bug where damage upgrades were applying when they shouldn't (for aliens)
|
|
// - Updated spike effect
|
|
// - Removed jetpack effect until it can be done properly
|
|
// - Lots of sound and effects changes
|
|
//
|
|
// Revision 1.42 2002/09/09 19:50:29 Flayra
|
|
// - Reworking jetpack effects, still needs more work
|
|
//
|
|
// Revision 1.41 2002/08/31 18:01:01 Flayra
|
|
// - Work at VALVe
|
|
//
|
|
// Revision 1.40 2002/08/16 02:34:42 Flayra
|
|
// - Removed ASSERTs when creating temporary entities, it can fail in big firefights
|
|
//
|
|
// Revision 1.39 2002/08/09 00:56:38 Flayra
|
|
// - Added particle system back when phasing in items from commander mode
|
|
//
|
|
// Revision 1.38 2002/08/02 22:00:53 Flayra
|
|
// - Removed old HL events, add correct bullet types for new bullet effects
|
|
//
|
|
// Revision 1.37 2002/07/26 23:04:14 Flayra
|
|
// - Generate numerical feedback for damage events
|
|
//
|
|
// Revision 1.36 2002/07/23 17:02:23 Flayra
|
|
// - Turret velocity is constructed on client in same way as server, instead of being sent across as calculated (network precision issues)
|
|
//
|
|
// Revision 1.35 2002/07/10 14:40:23 Flayra
|
|
// - Added special spike hit effect
|
|
//
|
|
// Revision 1.34 2002/07/08 16:57:19 Flayra
|
|
// - Added "invalid action" event, reworking for random spread for bullet fire
|
|
//
|
|
// Revision 1.33 2002/07/01 22:41:40 Flayra
|
|
// - Removed outdated overwatch target and tension events
|
|
//
|
|
// Revision 1.32 2002/07/01 21:31:49 Flayra
|
|
// - Regular update
|
|
//
|
|
// Revision 1.31 2002/06/25 17:57:15 Flayra
|
|
// - Removed old events, added new events, fixed infinite loop crash in PlayMeleeHitEffects, added parasite projectile
|
|
//
|
|
// Revision 1.30 2002/06/03 16:44:27 Flayra
|
|
// - Moved offense chamber firing into event, fixed duplicate empty sound (now it's an event, not server-side), weapons play view model anim in weapon (not event), started to add grenade event for grenade gun
|
|
//
|
|
// Revision 1.29 2002/05/28 17:36:55 Flayra
|
|
// - Don't play machine gun or pistol sounds louder when upgraded, changed welder from looping sound to periodic sound
|
|
//
|
|
// Revision 1.28 2002/05/23 02:33:42 Flayra
|
|
// - Post-crash checkin. Restored @Backup from around 4/16. Contains changes for last four weeks of development.
|
|
//
|
|
//===============================================================================
|
|
#include "cl_dll/hud.h"
|
|
#include "cl_dll/cl_util.h"
|
|
|
|
//#include "mod/AvHMarineWeapons.h"
|
|
//#include "mod/AvHAlienWeapons.h"
|
|
//#include "mod/AvHAlienAbilities.h"
|
|
|
|
#include "cl_dll/eventscripts.h"
|
|
#include "cl_dll/in_defs.h"
|
|
#include "cl_dll/ev_hldm.h"
|
|
#include "common/event_api.h"
|
|
#include "common/event_args.h"
|
|
#include "common/dlight.h"
|
|
#include "common/r_efx.h"
|
|
#include "mod/AvHMarineWeaponConstants.h"
|
|
#include "mod/AvHAlienWeaponConstants.h"
|
|
#include "mod/AvHParticleSystemManager.h"
|
|
#include "pm_shared/pm_defs.h"
|
|
#include "mod/AvHSpecials.h"
|
|
#include "mod/AvHEvents.h"
|
|
#include "mod/AvHSelectionHelper.h"
|
|
#include "pm_shared/pm_defs.h"
|
|
#include "mod/AvHPlayerUpgrade.h"
|
|
#include "mod/AvHSharedUtil.h"
|
|
#include "mod/AvHParticleConstants.h"
|
|
#include "mod/AvHMarineEquipmentConstants.h"
|
|
#include "mod/AvHAlienAbilityConstants.h"
|
|
#include "mod/AvHAlienEquipmentConstants.h"
|
|
#include "mod/AvHParticleTemplate.h"
|
|
#include "mod/AvHParticleTemplateClient.h"
|
|
#include "mod/AvHClientVariables.h"
|
|
#include "util/MathUtil.h"
|
|
#include "mod/AvHHulls.h"
|
|
|
|
//extern AvHKnife gKnife;
|
|
//extern AvHMachineGun gMachineGun;
|
|
//extern AvHPistol gPistol;
|
|
//extern AvHSonicGun gSonicGun;
|
|
//extern AvHHeavyMachineGun gHeavyMachineGun;
|
|
//extern AvHGrenadeGun gGrenadeGun;
|
|
//extern AvHWelder gWelder;
|
|
//extern AvHMine gMine;
|
|
//extern AvHSpitGun gSpitGun;
|
|
//extern AvHClaws gClaws;
|
|
//extern AvHSpore gSpores;
|
|
//extern AvHBite gBite;
|
|
//extern AvHBite2 gBite2;
|
|
//extern AvHSpikeGun gSpikeGun;
|
|
//extern AvHSwipe gSwipe;
|
|
//extern AvHWebSpinner gWebSpinner;
|
|
//extern AvHPrimalScream gPrimalScream;
|
|
//extern AvHParasiteGun gParasite;
|
|
//extern AvHUmbraGun gUmbra;
|
|
//extern AvHBlinkGun gBlink;
|
|
//extern AvHDivineWind gDivineWind;
|
|
//extern AvHParalysisGun gParalysisGun;
|
|
//extern AvHBileBombGun gBileBomb;
|
|
//extern AvHAcidRocketGun gAcidRocket;
|
|
//extern AvHHealingSpray gHealingSpray;
|
|
//extern AvHBabblerGun gBabblerGun;
|
|
|
|
// Alien abilities
|
|
//extern AvHLeap gLeap;
|
|
//extern AvHCharge gCharge;
|
|
|
|
|
|
// The sound constants are stored here, even though it's used on client
|
|
#define SND_CHANGE_PITCH (1<<7) // duplicated in protocol.h change sound pitch
|
|
#define SND_CHANGE_VOL (1<<6) // duplicated in protocol.h change sound vol
|
|
|
|
void V_PunchAxis( int axis, float punch );
|
|
//extern double gClientTimeLastUpdate;
|
|
void DrawCircleOnGroundAtPoint(vec3_t inOrigin, int inNumSides, int inStartAngle, int inRadius, float inR, float inG, float inB, float inA, bool inUseRedInstead, float inInnerRadius);
|
|
|
|
extern void ComputeGrenadeVelocity(Vector& inForward, Vector& inVelocity, Vector& outVelocity);
|
|
|
|
class LightType
|
|
{
|
|
public:
|
|
LightType(int inIndex, dlight_t* inLight) : mIndex(inIndex), mLight(inLight)
|
|
{}
|
|
|
|
int mIndex;
|
|
dlight_t* mLight;
|
|
};
|
|
|
|
typedef vector<LightType> DLightListType;
|
|
|
|
DLightListType gJetpackLights;
|
|
const float kArbitraryLargeLightTime = 2000.0f;
|
|
//SelectionListType gSelectionList;
|
|
extern playermove_t* pmove;
|
|
|
|
#include "pm_shared/pm_debug.h"
|
|
extern DebugPointListType gTriDebugLocations;
|
|
extern DebugPointListType gSquareDebugLocations;
|
|
extern DebugEntityListType gCubeDebugEntities;
|
|
|
|
AvHSelectionHelper gSelectionHelper;
|
|
extern AvHParticleTemplateListClient gParticleTemplateList;
|
|
|
|
extern const Vector g_vecZero;
|
|
|
|
//
|
|
// Macros to make these things easier, less error-prone and clearer
|
|
//
|
|
#define AVH_DECLARE_EVENT(s) \
|
|
extern "C" \
|
|
{\
|
|
void EV_##s(struct event_args_s *args); \
|
|
}
|
|
|
|
extern "C" Vector gPredictedPlayerOrigin;
|
|
|
|
#define AVH_NAME_TO_EVENT(s) "events/"#s".sc"
|
|
|
|
#define AVH_HOOK_EVENT(s) \
|
|
gEngfuncs.pfnHookEvent(AVH_NAME_TO_EVENT(s), EV_##s)
|
|
|
|
//#define AVH_DEFINE_EVENT(s, Class) \
|
|
//void EV_##s(struct event_args_s *args) \
|
|
//{\
|
|
// static Class theWeapon(AVH_NAME_TO_EVENT(s));
|
|
|
|
//#define AVH_DEFINE_EVENT_END \
|
|
///* TODO: Add cleanup or other default behavior? */ \
|
|
//}
|
|
|
|
//physent_t* GetEntity(int inPhysIndex);
|
|
|
|
//
|
|
// Declare events
|
|
//
|
|
|
|
//// Marine weapon events
|
|
AVH_DECLARE_EVENT(Knife)
|
|
AVH_DECLARE_EVENT(MachineGun)
|
|
AVH_DECLARE_EVENT(Pistol)
|
|
AVH_DECLARE_EVENT(SonicGun)
|
|
AVH_DECLARE_EVENT(HeavyMachineGun)
|
|
AVH_DECLARE_EVENT(GrenadeGun)
|
|
AVH_DECLARE_EVENT(Grenade)
|
|
|
|
// Alien weapon events
|
|
AVH_DECLARE_EVENT(SpitGun)
|
|
AVH_DECLARE_EVENT(OffenseChamber)
|
|
AVH_DECLARE_EVENT(Claws)
|
|
AVH_DECLARE_EVENT(Swipe)
|
|
//AVH_DECLARE_EVENT(EnsnareShoot)
|
|
//AVH_DECLARE_EVENT(EnsnareHit)
|
|
|
|
AVH_DECLARE_EVENT(SporeShoot)
|
|
AVH_DECLARE_EVENT(SporeCloud)
|
|
AVH_DECLARE_EVENT(UmbraGun)
|
|
AVH_DECLARE_EVENT(UmbraCloud)
|
|
|
|
AVH_DECLARE_EVENT(Bite)
|
|
AVH_DECLARE_EVENT(Bite2)
|
|
AVH_DECLARE_EVENT(SpikeGun)
|
|
//AVH_DECLARE_EVENT(LayEgg)
|
|
AVH_DECLARE_EVENT(BuildGun)
|
|
AVH_DECLARE_EVENT(HealingSpray)
|
|
AVH_DECLARE_EVENT(Metabolize)
|
|
AVH_DECLARE_EVENT(MetabolizeSuccess)
|
|
AVH_DECLARE_EVENT(SpinWeb)
|
|
//AVH_DECLARE_EVENT(Babbler)
|
|
AVH_DECLARE_EVENT(PrimalScream)
|
|
AVH_DECLARE_EVENT(Cocoon)
|
|
|
|
AVH_DECLARE_EVENT(Jetpack)
|
|
AVH_DECLARE_EVENT(Welder)
|
|
AVH_DECLARE_EVENT(WelderConst)
|
|
AVH_DECLARE_EVENT(WelderStart)
|
|
AVH_DECLARE_EVENT(WelderEnd)
|
|
//AVH_DECLARE_EVENT(OverwatchStart)
|
|
//AVH_DECLARE_EVENT(OverwatchTarget)
|
|
//AVH_DECLARE_EVENT(OverwatchTension)
|
|
//AVH_DECLARE_EVENT(OverwatchEnd)
|
|
AVH_DECLARE_EVENT(Regeneration)
|
|
AVH_DECLARE_EVENT(StartCloak);
|
|
AVH_DECLARE_EVENT(EndCloak);
|
|
//AVH_DECLARE_EVENT(WallJump)
|
|
AVH_DECLARE_EVENT(Flight)
|
|
AVH_DECLARE_EVENT(Select)
|
|
AVH_DECLARE_EVENT(Teleport)
|
|
AVH_DECLARE_EVENT(PhaseIn)
|
|
AVH_DECLARE_EVENT(SiegeHit)
|
|
AVH_DECLARE_EVENT(SiegeViewHit)
|
|
AVH_DECLARE_EVENT(StopScream)
|
|
AVH_DECLARE_EVENT(CommandPoints)
|
|
AVH_DECLARE_EVENT(AlienSightOn)
|
|
AVH_DECLARE_EVENT(AlienSightOff)
|
|
//AVH_DECLARE_EVENT(ParalysisGun)
|
|
//AVH_DECLARE_EVENT(ParalysisStart)
|
|
AVH_DECLARE_EVENT(ParasiteGun)
|
|
AVH_DECLARE_EVENT(BlinkSuccess)
|
|
AVH_DECLARE_EVENT(DivineWind)
|
|
AVH_DECLARE_EVENT(BileBomb)
|
|
AVH_DECLARE_EVENT(AcidRocket)
|
|
AVH_DECLARE_EVENT(Stomp);
|
|
AVH_DECLARE_EVENT(Devour);
|
|
AVH_DECLARE_EVENT(InvalidAction)
|
|
AVH_DECLARE_EVENT(Particle)
|
|
AVH_DECLARE_EVENT(DistressBeacon)
|
|
AVH_DECLARE_EVENT(LevelUp)
|
|
|
|
// Alien abilities
|
|
AVH_DECLARE_EVENT(Leap);
|
|
AVH_DECLARE_EVENT(Charge);
|
|
//AVH_DECLARE_EVENT(HiveHit)
|
|
AVH_DECLARE_EVENT(EmptySound)
|
|
AVH_DECLARE_EVENT(NumericalInfo)
|
|
AVH_DECLARE_EVENT(WeaponAnimation)
|
|
AVH_DECLARE_EVENT(Ability);
|
|
|
|
//extern "C"
|
|
//{
|
|
//void EV_FireGlock1( struct event_args_s *args );
|
|
//void EV_FireGlock2( struct event_args_s *args );
|
|
//}
|
|
|
|
//
|
|
// Hook 'em on Initialize
|
|
//
|
|
void Game_HookEvents( void )
|
|
{
|
|
// gEngfuncs.pfnHookEvent( "events/glock1.sc", EV_FireGlock1 );
|
|
// gEngfuncs.pfnHookEvent( "events/glock2.sc", EV_FireGlock2 );
|
|
|
|
// Hook marine weapon events
|
|
gEngfuncs.pfnHookEvent( kKNEventName, EV_Knife );
|
|
gEngfuncs.pfnHookEvent( kMGEventName, EV_MachineGun );
|
|
gEngfuncs.pfnHookEvent( kHGEventName, EV_Pistol );
|
|
gEngfuncs.pfnHookEvent( kSGEventName, EV_SonicGun );
|
|
gEngfuncs.pfnHookEvent( kHMGEventName, EV_HeavyMachineGun );
|
|
gEngfuncs.pfnHookEvent( kGGEventName, EV_GrenadeGun );
|
|
gEngfuncs.pfnHookEvent( kGREventName, EV_Grenade );
|
|
|
|
// Alien weapon events
|
|
gEngfuncs.pfnHookEvent( kSpitGEventName, EV_SpitGun );
|
|
gEngfuncs.pfnHookEvent( kOffenseChamberEventName, EV_OffenseChamber);
|
|
gEngfuncs.pfnHookEvent( kClawsEventName, EV_Claws );
|
|
gEngfuncs.pfnHookEvent( kSwipeEventName, EV_Swipe );
|
|
//gEngfuncs.pfnHookEvent( kEnsnareShootEventName, EV_EnsnareShoot);
|
|
//gEngfuncs.pfnHookEvent( kEnsnareHitEventName, EV_EnsnareHit);
|
|
|
|
gEngfuncs.pfnHookEvent( kSporeShootEventName, EV_SporeShoot);
|
|
gEngfuncs.pfnHookEvent( kSporeCloudEventName, EV_SporeCloud);
|
|
gEngfuncs.pfnHookEvent( kUmbraShootEventName, EV_UmbraGun);
|
|
gEngfuncs.pfnHookEvent( kUmbraCloudEventName, EV_UmbraCloud);
|
|
|
|
gEngfuncs.pfnHookEvent( kSpikeShootEventName, EV_SpikeGun);
|
|
gEngfuncs.pfnHookEvent( kBiteEventName, EV_Bite);
|
|
gEngfuncs.pfnHookEvent( kBite2EventName, EV_Bite2);
|
|
//gEngfuncs.pfnHookEvent( kEggLayerShootEventName, EV_LayEgg);
|
|
gEngfuncs.pfnHookEvent( kBuildingGunEventName, EV_BuildGun);
|
|
gEngfuncs.pfnHookEvent( kHealingSprayEventName, EV_HealingSpray);
|
|
gEngfuncs.pfnHookEvent( kMetabolizeEventName, EV_Metabolize);
|
|
gEngfuncs.pfnHookEvent( kMetabolizeSuccessEventName, EV_MetabolizeSuccess);
|
|
gEngfuncs.pfnHookEvent( kWebSpinnerShootEventName, EV_SpinWeb);
|
|
// gEngfuncs.pfnHookEvent( kBabblerGunEventName, EV_Babbler);
|
|
gEngfuncs.pfnHookEvent( kPrimalScreamShootEventName, EV_PrimalScream);
|
|
gEngfuncs.pfnHookEvent( kStopPrimalScreamSoundEvent, EV_StopScream);
|
|
|
|
gEngfuncs.pfnHookEvent( kJetpackEvent, EV_Jetpack );
|
|
gEngfuncs.pfnHookEvent( kWelderEventName, EV_Welder );
|
|
gEngfuncs.pfnHookEvent( kWelderStartEventName, EV_WelderStart );
|
|
gEngfuncs.pfnHookEvent( kWelderEndEventName, EV_WelderEnd );
|
|
gEngfuncs.pfnHookEvent( kWelderConstEventName, EV_WelderConst );
|
|
//gEngfuncs.pfnHookEvent( kStartOverwatchEvent, EV_OverwatchStart );
|
|
//gEngfuncs.pfnHookEvent( kEndOverwatchEvent, EV_OverwatchEnd );
|
|
gEngfuncs.pfnHookEvent( kRegenerationEvent, EV_Regeneration );
|
|
gEngfuncs.pfnHookEvent( kStartCloakEvent, EV_StartCloak );
|
|
gEngfuncs.pfnHookEvent( kEndCloakEvent, EV_EndCloak );
|
|
//gEngfuncs.pfnHookEvent( kWallJumpEvent, EV_WallJump );
|
|
gEngfuncs.pfnHookEvent( kFlightEvent, EV_Flight );
|
|
gEngfuncs.pfnHookEvent( kTeleportEvent, EV_Teleport );
|
|
gEngfuncs.pfnHookEvent( kPhaseInEvent, EV_PhaseIn );
|
|
gEngfuncs.pfnHookEvent( kSiegeHitEvent, EV_SiegeHit );
|
|
gEngfuncs.pfnHookEvent( kSiegeViewHitEvent, EV_SiegeViewHit );
|
|
gEngfuncs.pfnHookEvent( kCommanderPointsAwardedEvent, EV_CommandPoints );
|
|
gEngfuncs.pfnHookEvent( kAlienSightOnEvent, EV_AlienSightOn);
|
|
gEngfuncs.pfnHookEvent( kAlienSightOffEvent, EV_AlienSightOff);
|
|
// gEngfuncs.pfnHookEvent( kParalysisShootEventName, EV_ParalysisGun);
|
|
// gEngfuncs.pfnHookEvent( kParalysisStartEventName, EV_ParalysisStart);
|
|
gEngfuncs.pfnHookEvent( kParasiteShootEventName, EV_ParasiteGun);
|
|
gEngfuncs.pfnHookEvent( kBlinkEffectSuccessEventName, EV_BlinkSuccess);
|
|
gEngfuncs.pfnHookEvent( kDivineWindShootEventName, EV_DivineWind);
|
|
gEngfuncs.pfnHookEvent( kBileBombShootEventName, EV_BileBomb);
|
|
gEngfuncs.pfnHookEvent( kAcidRocketShootEventName, EV_AcidRocket);
|
|
gEngfuncs.pfnHookEvent( kStompShootEventName, EV_Stomp);
|
|
gEngfuncs.pfnHookEvent( kDevourShootEventName, EV_Devour);
|
|
|
|
gEngfuncs.pfnHookEvent( kLeapEventName, EV_Leap);
|
|
gEngfuncs.pfnHookEvent( kChargeEventName, EV_Charge);
|
|
gEngfuncs.pfnHookEvent( kAbilityEventName, EV_Ability);
|
|
|
|
//gEngfuncs.pfnHookEvent( kHiveHitEvent, EV_HiveHit );
|
|
gEngfuncs.pfnHookEvent( kEmptySoundEvent, EV_EmptySound );
|
|
gEngfuncs.pfnHookEvent( kNumericalInfoEvent, EV_NumericalInfo );
|
|
gEngfuncs.pfnHookEvent( kInvalidActionEvent, EV_InvalidAction);
|
|
gEngfuncs.pfnHookEvent( kParticleEvent, EV_Particle);
|
|
gEngfuncs.pfnHookEvent( kDistressBeaconEvent, EV_DistressBeacon);
|
|
gEngfuncs.pfnHookEvent( kWeaponAnimationEvent, EV_WeaponAnimation);
|
|
gEngfuncs.pfnHookEvent( kLevelUpEvent, EV_LevelUp);
|
|
}
|
|
|
|
#define LOUD_GUN_VOLUME 1000
|
|
#define NORMAL_GUN_VOLUME 600
|
|
#define QUIET_GUN_VOLUME 200
|
|
|
|
#define BRIGHT_GUN_FLASH 512
|
|
#define NORMAL_GUN_FLASH 256
|
|
#define DIM_GUN_FLASH 128
|
|
|
|
#define BIG_EXPLOSION_VOLUME 2048
|
|
#define NORMAL_EXPLOSION_VOLUME 1024
|
|
#define SMALL_EXPLOSION_VOLUME 512
|
|
|
|
#define WEAPON_ACTIVITY_VOLUME 64
|
|
|
|
#define VECTOR_CONE_1DEGREES Vector( 0.00873, 0.00873, 0.00873 )
|
|
#define VECTOR_CONE_2DEGREES Vector( 0.01745, 0.01745, 0.01745 )
|
|
#define VECTOR_CONE_3DEGREES Vector( 0.02618, 0.02618, 0.02618 )
|
|
#define VECTOR_CONE_4DEGREES Vector( 0.03490, 0.03490, 0.03490 )
|
|
#define VECTOR_CONE_5DEGREES Vector( 0.04362, 0.04362, 0.04362 )
|
|
#define VECTOR_CONE_6DEGREES Vector( 0.05234, 0.05234, 0.05234 )
|
|
#define VECTOR_CONE_7DEGREES Vector( 0.06105, 0.06105, 0.06105 )
|
|
#define VECTOR_CONE_8DEGREES Vector( 0.06976, 0.06976, 0.06976 )
|
|
#define VECTOR_CONE_9DEGREES Vector( 0.07846, 0.07846, 0.07846 )
|
|
#define VECTOR_CONE_10DEGREES Vector( 0.08716, 0.08716, 0.08716 )
|
|
#define VECTOR_CONE_15DEGREES Vector( 0.13053, 0.13053, 0.13053 )
|
|
#define VECTOR_CONE_20DEGREES Vector( 0.17365, 0.17365, 0.17365 )
|
|
|
|
// <<< cgc >>>
|
|
// This is duplicated here, make sure it's up to date
|
|
//#define SND_CHANGE_PITCH (1<<7) // duplicated in protocol.h change sound pitch
|
|
|
|
int& GetUpgradeState(int inIndex)
|
|
{
|
|
cl_entity_t* thePlayer = GetEntity(inIndex);
|
|
return thePlayer->curstate.iuser4;
|
|
}
|
|
|
|
// From Counter-strike
|
|
#define MAX_DEAD_PLAYER_MODELS 64
|
|
TEMPENTITY* g_DeadPlayerModels[MAX_DEAD_PLAYER_MODELS];
|
|
#define MAX_BODY_TIME 5
|
|
#define FTENT_BODYTRACE 0x00100000
|
|
#define FTENT_BODYGRAVITY 0x00200000
|
|
|
|
void RemoveAllDecals()
|
|
{
|
|
for ( int har = 0; har < (int)CVAR_GET_FLOAT( "r_decals" ); har++ )
|
|
gEngfuncs.pEfxAPI->R_DecalRemoveAll ( har );
|
|
|
|
//if ( g_pParticleMan )
|
|
// g_pParticleMan->ResetParticles();
|
|
|
|
if ( g_DeadPlayerModels == NULL )
|
|
return;
|
|
|
|
// Is the dead player model list already clean?
|
|
if ( g_DeadPlayerModels[0] == NULL )
|
|
return;
|
|
|
|
// Clear dead player model list
|
|
for ( int i = 0; i < MAX_DEAD_PLAYER_MODELS; i++ )
|
|
{
|
|
if ( g_DeadPlayerModels[i] )
|
|
{
|
|
g_DeadPlayerModels[i]->die = 0;
|
|
g_DeadPlayerModels[i] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
// lowers body into ground
|
|
void RemoveBody(TEMPENTITY* te, float frametime, float current_time)
|
|
{
|
|
if ( current_time >= te->entity.curstate.fuser2 + MAX_BODY_TIME + CVAR_GET_FLOAT( "cl_corpsestay" ) )
|
|
te->entity.origin[2] -= frametime * 5;
|
|
}
|
|
|
|
// Play a sound? Through up some smoke?
|
|
void HitBody( TEMPENTITY *ent, struct pmtrace_s *ptr )
|
|
{
|
|
if ( ptr->plane.normal.z > 0 )
|
|
ent->flags |= FTENT_BODYGRAVITY;
|
|
}
|
|
|
|
// From Counter-strike
|
|
void CreateCorpse ( Vector vOrigin, Vector vAngles, const char *pModel, float flAnimTime, int iSequence, int iBody )
|
|
{
|
|
int framecount = 255;
|
|
|
|
int iModelIndex = gEngfuncs.pEventAPI->EV_FindModelIndex ( pModel );
|
|
TEMPENTITY *pBody = gEngfuncs.pEfxAPI->R_TempModel( vOrigin, Vector( 0, 0, 0 ), vAngles, 100, gEngfuncs.pEventAPI->EV_FindModelIndex ( pModel ), 0 );
|
|
|
|
if ( pBody == NULL)
|
|
return;
|
|
|
|
pBody->flags |= FTENT_COLLIDEWORLD | FTENT_CLIENTCUSTOM | FTENT_SPRANIMATE | FTENT_BODYTRACE | FTENT_PERSIST;
|
|
pBody->frameMax = framecount;
|
|
|
|
pBody->entity.curstate.animtime = flAnimTime;
|
|
pBody->entity.curstate.framerate = 1;
|
|
pBody->entity.curstate.sequence = iSequence;
|
|
pBody->entity.curstate.frame = 0;
|
|
pBody->entity.curstate.body = iBody;
|
|
pBody->entity.curstate.renderamt = 255;
|
|
pBody->entity.curstate.fuser1 = gHUD.m_flTime + 1;
|
|
pBody->entity.curstate.fuser2 = gHUD.m_flTime + CVAR_GET_FLOAT ("cl_corpsestay"); // estimated end time of animation
|
|
|
|
// let entity sink 5 seconds then kill it after cl_corpsestay is over
|
|
pBody->die = gEngfuncs.GetClientTime() + pBody->entity.curstate.fuser2 + 5;
|
|
pBody->callback = RemoveBody;
|
|
pBody->hitcallback = HitBody;
|
|
pBody->bounceFactor = 0;
|
|
|
|
// Save body to list so we can remove after round ends
|
|
for ( int i = 0; i < 64; i ++ )
|
|
{
|
|
if ( g_DeadPlayerModels[ i ] == NULL )
|
|
{
|
|
g_DeadPlayerModels[ i ] = pBody;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PlayMeleeHitEffects(struct event_args_s* inArgs, int inRange, const string& inSoundName)
|
|
{
|
|
int thePlayer = inArgs->entindex;
|
|
|
|
vec3_t theForward, theRight, theUp;
|
|
gEngfuncs.pfnAngleVectors(inArgs->angles, theForward, theRight, theUp);
|
|
|
|
vec3_t theAimingDir;
|
|
for(int i = 0; i < 3; i++)
|
|
{
|
|
theAimingDir[i] = theForward[i];// + theRight[i] + theUp[i];
|
|
}
|
|
VectorNormalize(theAimingDir);
|
|
|
|
// Do a trace within certain range, ignoring ourselves
|
|
gEngfuncs.pEventAPI->EV_SetUpPlayerPrediction( false, true );
|
|
|
|
gEngfuncs.pEventAPI->EV_PushPMStates();
|
|
|
|
gEngfuncs.pEventAPI->EV_SetSolidPlayers(-1);
|
|
|
|
gEngfuncs.pEventAPI->EV_SetTraceHull( 2 );
|
|
|
|
Vector theStartPos;
|
|
//VectorMA(inArgs->origin, 17, theAimingDir, theStartPos);
|
|
VectorCopy(inArgs->origin, theStartPos);
|
|
//gTriDebugLocations.push_back(DebugPoint(theStartPos[0], theStartPos[1], theStartPos[2]));
|
|
//gTriDebugLocations.push_back(DebugPoint(theStartPos[0] + theAimingDir[0]*inRange, theStartPos[1]+ theAimingDir[1]*inRange, theStartPos[2] + theAimingDir[2]*inRange));
|
|
|
|
Vector theEndPos;
|
|
VectorMA(theStartPos, inRange, theAimingDir, theEndPos);
|
|
//gTriDebugLocations.push_back(DebugPoint(theEndPos[0], theEndPos[1], theEndPos[2]));
|
|
|
|
pmtrace_t tr;
|
|
bool theDone = false;
|
|
int theNumIterations = 0;
|
|
|
|
do
|
|
{
|
|
gEngfuncs.pEventAPI->EV_PlayerTrace(theStartPos, theEndPos, PM_NORMAL, thePlayer, &tr);
|
|
|
|
// Did we hit something marked as enemy?
|
|
int theHit = gEngfuncs.pEventAPI->EV_IndexFromTrace(&tr);
|
|
if(theHit == thePlayer)
|
|
{
|
|
// tr.endpos: -71.1894, 3274.23, -527.888
|
|
// theStartPos: -71.20, 3274.02, -517.89
|
|
// theAimingDir: -0.01169, -0.02157, .99969
|
|
// Infinite loop: this next line doesn't changea anything
|
|
VectorMA(tr.endpos, kHitOffsetAmount, theAimingDir, theStartPos);
|
|
}
|
|
else if(tr.fraction < 1.0f)
|
|
{
|
|
Vector theHitPos = tr.endpos;
|
|
|
|
// Play sound
|
|
gEngfuncs.pEventAPI->EV_PlaySound(thePlayer, theHitPos, CHAN_AUTO, inSoundName.c_str(), 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Play generic damage effect
|
|
//AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsMeleeDamage, theHitPos, &(tr.plane.normal));
|
|
|
|
theDone = true;
|
|
}
|
|
else
|
|
{
|
|
theDone = true;
|
|
}
|
|
|
|
// Prevent infinite loop
|
|
theNumIterations++;
|
|
|
|
if(theNumIterations > 20)
|
|
{
|
|
theDone = true;
|
|
}
|
|
|
|
} while(!theDone);
|
|
|
|
gEngfuncs.pEventAPI->EV_PopPMStates();
|
|
}
|
|
|
|
//
|
|
// Define them
|
|
//
|
|
void EV_Knife(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 1);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
theSoundToPlay = kKNFireSound1;
|
|
break;
|
|
case 1:
|
|
theSoundToPlay = kKNFireSound2;
|
|
break;
|
|
}
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// TODO: Now do a trace line and if we hit a wall, play the wall sound, if we hit a person, play a hit-flesh sound
|
|
// TODO: Add sparks where it hit
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
/*
|
|
float theHalfSpread = kKNXPunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
*/
|
|
}
|
|
}
|
|
|
|
//void EV_Grenade(struct event_args_s* inArgs)
|
|
//{
|
|
// int idx = inArgs->entindex;
|
|
// vec3_t up, right, forward;
|
|
// gEngfuncs.pfnAngleVectors(inArgs->angles, forward, right, up);
|
|
//
|
|
// // Play attack sound
|
|
// char* theSoundToPlay = kKNFireSound1;
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//
|
|
// // Create a temporary entity that bounces and dies after a certain amount of time
|
|
// int theModelIndex;
|
|
// struct model_s* theModel = gEngfuncs.CL_LoadModel("models/w_grenade.mdl", &theModelIndex);
|
|
// ASSERT(theModel);
|
|
// TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
// ASSERT(theTempEntity);
|
|
// if(theTempEntity)
|
|
// {
|
|
// vec3_t theStartPos, theEndPos, vecSrc;
|
|
// EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
// VectorMA(vecSrc, kGBarrelLength, forward, theStartPos);
|
|
//
|
|
// VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
// theTempEntity->die += kGrenDetonateTime;
|
|
// //theTempEntity->hitcallback = SpitHit;
|
|
// theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_PERSIST);
|
|
//
|
|
// vec3_t theSourceVelocity;
|
|
// VectorCopy(inArgs->velocity, theSourceVelocity);
|
|
//
|
|
// vec3_t theVelocity;
|
|
// ComputeGrenadeVelocity(forward, theSourceVelocity, theVelocity);
|
|
//
|
|
// VectorCopy(theVelocity, theTempEntity->entity.baseline.origin);
|
|
// VectorCopy(theVelocity, theTempEntity->entity.baseline.velocity);
|
|
// }
|
|
//
|
|
// // General x-punch axis
|
|
//// if (EV_IsLocal(idx))
|
|
//// {
|
|
//// gEngfuncs.pEventAPI->EV_WeaponAnimation(ANIM_FIRE1 + gEngfuncs.pfnRandomLong(0,2), 2);
|
|
//// }
|
|
//}
|
|
|
|
void EV_MachineGun(struct event_args_s* args)
|
|
{
|
|
// What to do about this static member?
|
|
static int tracerCount[ 32 ];
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = args->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = args->entindex;
|
|
VectorCopy( args->origin, origin );
|
|
VectorCopy( args->angles, angles );
|
|
VectorCopy( args->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
// Get upgrade
|
|
int theTracerFreq;
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(idx), NULL, &theTracerFreq);
|
|
|
|
// Vary flange effect more with higher upgrade
|
|
int theUpperBound = theUpgradeLevel*10;
|
|
int thePitch = 100 + (gEngfuncs.pfnRandomLong(0, theUpperBound) - theUpperBound);
|
|
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
// Add muzzle flash to current weapon model
|
|
if(theUpgradeLevel > 0)
|
|
{
|
|
EV_MuzzleFlash();
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(args->iparam2, 2);
|
|
}
|
|
|
|
// General ejecting ammo if any
|
|
int shell = gEngfuncs.pEventAPI->EV_FindModelIndex(kMGEjectModel);
|
|
vec3_t ShellVelocity;
|
|
vec3_t ShellOrigin;
|
|
|
|
EV_GetDefaultShellInfo(args, origin, velocity, ShellVelocity, ShellOrigin, forward, right, up, 20, -12, 4 );
|
|
|
|
// Only eject brass when upgraded
|
|
//if(theUpgradeLevel > 0)
|
|
//{
|
|
// VectorScale(ShellVelocity, theUpgradeLevel, ShellVelocity);
|
|
EV_EjectBrass(ShellOrigin, ShellVelocity, angles[ YAW ], shell, TE_BOUNCE_SHELL);
|
|
//}
|
|
|
|
// Play one of basic attack sounds
|
|
float theVolume = args->fparam1;
|
|
float theAttenuation = .8f + .3*theUpgradeLevel;
|
|
|
|
char* theSoundToPlay = kMGFireSound1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, theSoundToPlay, theVolume, theAttenuation, 0, thePitch);
|
|
|
|
EV_GetGunPosition( args, vecSrc, origin );
|
|
VectorCopy( forward, vecAiming );
|
|
|
|
//EV_HLDM_FireBullets( idx, forward, right, up, 1, vecSrc, vecAiming, VECTOR_CONE_6DEGREES, kMGRange, BULLET_PLAYER_MP5, 2, &tracerCount[idx-1] );
|
|
EV_HLDM_FireBulletsPlayer( idx, forward, right, up, 1, vecSrc, vecAiming, kMGRange, BULLET_PLAYER_MP5, theTracerFreq, &tracerCount[idx-1], kMGSpread, args->iparam1);
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
|
|
// Multiply punch by upgrade level
|
|
//float theHalfSpread = (kMGXPunch/4.0f)*(theUpgradeLevel+1);
|
|
|
|
// Changed to ignore upgrade level
|
|
float theHalfSpread = (kMGXPunch/4.0f);
|
|
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void EV_Pistol(struct event_args_s* args)
|
|
{
|
|
// What to do about this static member?
|
|
static int tracerCount[ 32 ];
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = args->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = args->entindex;
|
|
VectorCopy( args->origin, origin );
|
|
VectorCopy( args->angles, angles );
|
|
VectorCopy( args->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
// Get upgrade
|
|
int theTracerFreq;
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(idx), NULL, &theTracerFreq);
|
|
|
|
// Vary flange effect more with higher upgrade
|
|
int theUpperBound = theUpgradeLevel*10;
|
|
int thePitch = 100 + (gEngfuncs.pfnRandomLong(0, theUpperBound) - theUpperBound);
|
|
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
// Add muzzle flash to current weapon model
|
|
if(theUpgradeLevel > 0)
|
|
{
|
|
EV_MuzzleFlash();
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(args->iparam2, 2);
|
|
}
|
|
|
|
// General ejecting ammo if any
|
|
int shell = gEngfuncs.pEventAPI->EV_FindModelIndex(kHGEjectModel);
|
|
vec3_t ShellVelocity;
|
|
vec3_t ShellOrigin;
|
|
|
|
EV_GetDefaultShellInfo(args, origin, velocity, ShellVelocity, ShellOrigin, forward, right, up, 20, -12, 4 );
|
|
|
|
// Only eject brass when upgraded
|
|
//if(theUpgradeLevel > 0)
|
|
//{
|
|
// VectorScale(ShellVelocity, theUpgradeLevel, ShellVelocity);
|
|
EV_EjectBrass(ShellOrigin, ShellVelocity, angles[ YAW ], shell, TE_BOUNCE_SHELL);
|
|
//}
|
|
|
|
// Play one of basic attack sounds
|
|
float theVolume = args->fparam1;
|
|
float theAttenuation = .8f + .3*theUpgradeLevel;
|
|
|
|
char* theSoundToPlay = kHGFireSound1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, theSoundToPlay, theVolume, theAttenuation, 0, thePitch);
|
|
|
|
EV_GetGunPosition( args, vecSrc, origin );
|
|
VectorCopy( forward, vecAiming );
|
|
|
|
//EV_HLDM_FireBullets( idx, forward, right, up, 1, vecSrc, vecAiming, VECTOR_CONE_6DEGREES, kHGRange, BULLET_PLAYER_MP5, 2, &tracerCount[idx-1] );
|
|
EV_HLDM_FireBulletsPlayer( idx, forward, right, up, 1, vecSrc, vecAiming, kHGRange, BULLET_PLAYER_357, theTracerFreq, &tracerCount[idx-1], kHGSpread, args->iparam1);
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
// Multiply punch by upgrade level
|
|
//float theHalfSpread = (kHGXPunch/3.0f)*(theUpgradeLevel+1);
|
|
|
|
// Changed to ignore upgrade level
|
|
float theHalfSpread = (kHGXPunch/3.0f);
|
|
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_SonicGun(struct event_args_s* args)
|
|
{
|
|
// What to do about this static member?
|
|
static int tracerCount[ 32 ];
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = args->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = args->entindex;
|
|
VectorCopy( args->origin, origin );
|
|
VectorCopy( args->angles, angles );
|
|
VectorCopy( args->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
// Get upgrade
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(idx));
|
|
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
// Add muzzle flash to current weapon model
|
|
if(theUpgradeLevel > 0)
|
|
{
|
|
EV_MuzzleFlash();
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(args->iparam2, 2);
|
|
}
|
|
|
|
// General ejecting ammo if any
|
|
int shell = gEngfuncs.pEventAPI->EV_FindModelIndex(kSGEjectModel);
|
|
vec3_t ShellVelocity;
|
|
vec3_t ShellOrigin;
|
|
|
|
EV_GetDefaultShellInfo(args, origin, velocity, ShellVelocity, ShellOrigin, forward, right, up, 20, -12, 4 );
|
|
|
|
//if(theUpgradeLevel > 0)
|
|
//{
|
|
// VectorScale(ShellVelocity, theUpgradeLevel, ShellVelocity);
|
|
EV_EjectBrass(ShellOrigin, ShellVelocity, angles[ YAW ], shell, TE_BOUNCE_SHELL);
|
|
//}
|
|
|
|
float theAttenuation = .8f + .2*theUpgradeLevel;
|
|
//float theVolume = min(.85f + .05*theUpgradeLevel, 1.0f);
|
|
float theVolume = args->fparam1;
|
|
|
|
// TODO: Synch up with propagated random seed?
|
|
// Play one of basic attack sounds
|
|
//if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
char* theSoundToPlay = kSGFireSound1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound( idx, origin, CHAN_WEAPON, theSoundToPlay, theVolume, theAttenuation, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
EV_GetGunPosition( args, vecSrc, origin );
|
|
VectorCopy( forward, vecAiming );
|
|
|
|
//if ( gEngfuncs.GetMaxClients() > 1 )
|
|
//{
|
|
vec3_t theBarrelOffset, theBarrelTip;
|
|
VectorScale(forward, 30, theBarrelOffset);
|
|
VectorAdd(vecSrc, theBarrelOffset, theBarrelTip);
|
|
|
|
// if(theUpgradeLevel > 0)
|
|
// {
|
|
// AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsShotgun, theBarrelTip);
|
|
// }
|
|
//EV_HLDM_FireBullets( idx, forward, right, up, kSGBulletsPerShot, vecSrc, vecAiming, kSGRange, BULLET_PLAYER_BUCKSHOT, 0, &tracerCount[idx-1], VECTOR_CONE_20DEGREES.x, VECTOR_CONE_20DEGREES.y);
|
|
EV_HLDM_FireBulletsPlayer( idx, forward, right, up, BALANCE_VAR(kSGBulletsPerShot), vecSrc, vecAiming, kSGRange, BULLET_PLAYER_BUCKSHOT, 0, &tracerCount[idx-1], kSGSpread, args->iparam1);
|
|
//}
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
//float theHalfSpread = (kSGXPunch/3.0f)*(theUpgradeLevel+1);
|
|
|
|
// Changed to ignore upgrade level
|
|
float theHalfSpread = (kSGXPunch/3.0f);
|
|
|
|
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_HeavyMachineGun(struct event_args_s* args)
|
|
{
|
|
// What to do about this static member?
|
|
static int tracerCount[ 32 ];
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = args->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = args->entindex;
|
|
VectorCopy( args->origin, origin );
|
|
VectorCopy( args->angles, angles );
|
|
VectorCopy( args->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
int theTracerLevel;
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(idx), NULL, &theTracerLevel);
|
|
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
if(theUpgradeLevel > 0)
|
|
{
|
|
// Add muzzle flash to current weapon model
|
|
EV_MuzzleFlash();
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(args->iparam2, 2);
|
|
}
|
|
|
|
// General ejecting ammo if any
|
|
int shell = gEngfuncs.pEventAPI->EV_FindModelIndex(kHMGEjectModel);
|
|
vec3_t ShellVelocity;
|
|
vec3_t ShellOrigin;
|
|
|
|
EV_GetDefaultShellInfo(args, origin, velocity, ShellVelocity, ShellOrigin, forward, right, up, 20, -12, 4 );
|
|
|
|
//if(theUpgradeLevel > 0)
|
|
//{
|
|
// VectorScale(ShellVelocity, theUpgradeLevel, ShellVelocity);
|
|
EV_EjectBrass(ShellOrigin, ShellVelocity, angles[ YAW ], shell, TE_BOUNCE_SHELL);
|
|
//}
|
|
|
|
// TODO: Synch up with propagated random seed?
|
|
// Play one of basic attack sounds
|
|
int theHighIndex = theUpgradeLevel;
|
|
//int theRandomLong = gEngfuncs.pfnRandomLong(0, theHighIndex);
|
|
//float theVolume = min(.6 + .2*theUpgradeLevel, 1.0f);
|
|
float theVolume = args->fparam1;
|
|
float theAttenuation = .8f + .2*theUpgradeLevel;
|
|
int thePitch = 100;
|
|
if(theUpgradeLevel > 0)
|
|
{
|
|
int theVariance = theUpgradeLevel*6;
|
|
thePitch += (gEngfuncs.pfnRandomLong( 0, theVariance))/(theVariance/2);
|
|
}
|
|
|
|
char* theSoundToPlay = kHMGFireSound1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound( idx, origin, CHAN_WEAPON, theSoundToPlay, theVolume, theAttenuation, 0, thePitch);
|
|
|
|
EV_GetGunPosition( args, vecSrc, origin );
|
|
VectorCopy( forward, vecAiming );
|
|
|
|
//EV_HLDM_FireBullets( idx, forward, right, up, 1, vecSrc, vecAiming, kHMGRange, BULLET_PLAYER_357, theTracerLevel, &tracerCount[idx-1], theSpreadX, theSpreadY);
|
|
EV_HLDM_FireBulletsPlayer( idx, forward, right, up, 1, vecSrc, vecAiming, kHMGRange, BULLET_PLAYER_MP5, theTracerLevel, &tracerCount[idx-1], kHMGSpread, args->iparam1);
|
|
|
|
// if(theUpgradeLevel > 1)
|
|
// {
|
|
// if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
// {
|
|
// vec3_t theBarrelOffset, theBarrelTip;
|
|
// VectorScale(forward, 100, theBarrelOffset);
|
|
// VectorAdd(vecSrc, theBarrelOffset, theBarrelTip);
|
|
//
|
|
// AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsHeavyMGSmoke, vecSrc);
|
|
// }
|
|
// }
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
//float theHalfSpread = (kHMGXPunch/4.0f)*(theUpgradeLevel+1);
|
|
|
|
// Changed to ignore upgrade level
|
|
float theHalfSpread = (kHMGXPunch/4.0f);
|
|
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void GrenadeHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
{
|
|
char* theSoundToPlay = kGRHitSound;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(ent->entity.index, ptr->endpos, CHAN_AUTO, theSoundToPlay, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//ASSERT(ptr);
|
|
//ASSERT(ent);
|
|
//ent->die = gEngfuncs.GetClientTime();
|
|
}
|
|
|
|
void EV_GrenadeGun(struct event_args_s* inArgs)
|
|
{
|
|
// What to do about this static member?
|
|
static int tracerCount[ 32 ];
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = inArgs->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(idx));
|
|
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
// Add muzzle flash to current weapon model
|
|
if(theUpgradeLevel > 0)
|
|
{
|
|
EV_MuzzleFlash();
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
// // General ejecting ammo if any
|
|
// int shell = gEngfuncs.pEventAPI->EV_FindModelIndex(kGGEjectModel);
|
|
// vec3_t ShellVelocity;
|
|
// vec3_t ShellOrigin;
|
|
//
|
|
// EV_GetDefaultShellInfo(inArgs, origin, velocity, ShellVelocity, ShellOrigin, forward, right, up, 20, -12, 4 );
|
|
//
|
|
// //if(theUpgradeLevel > 0)
|
|
// //{
|
|
// // VectorScale(ShellVelocity, theUpgradeLevel, ShellVelocity);
|
|
// EV_EjectBrass(ShellOrigin, ShellVelocity, angles[ YAW ], shell, TE_BOUNCE_SHELL);
|
|
// //}
|
|
|
|
char* theSoundToPlay = kGGFireSound1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound( idx, origin, CHAN_WEAPON, theSoundToPlay, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
EV_GetGunPosition( inArgs, vecSrc, origin );
|
|
VectorCopy( forward, vecAiming );
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
//float theHalfSpread = (kGGXPunch/2.0f)*(theUpgradeLevel+1);
|
|
|
|
// Changed to ignore upgrade level
|
|
float theHalfSpread = (kGGXPunch/2.0f);
|
|
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
|
|
// Create temp entity for grenade (assumes grenade is created at inArgs->origin, with velocity inArgs->angles)
|
|
// ASSERT(inArgs);
|
|
//
|
|
// Vector theStartPos;
|
|
// VectorCopy(inArgs->origin, theStartPos);
|
|
//
|
|
// // Fire grenade
|
|
// int theModelIndex;
|
|
// struct model_s* theModel = gEngfuncs.CL_LoadModel(kGGAmmoModel, &theModelIndex);
|
|
// if(theModel)
|
|
// {
|
|
// TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
// if(theTempEntity)
|
|
// {
|
|
// VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
// theTempEntity->die += kSpitLifetime;
|
|
// theTempEntity->hitcallback = GrenadeHit;
|
|
// theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_PERSIST | FTENT_GRAVITY| FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP);
|
|
// theTempEntity->entity.curstate.framerate = 30;
|
|
// theTempEntity->frameMax = 4;//theModel->numframes;
|
|
//
|
|
// // Read origin as origin, and angles as velocity (from AvHAlienTurret::Shoot())
|
|
// Vector theStartVelocity;
|
|
// VectorCopy(inArgs->angles, theStartVelocity);
|
|
//
|
|
// // Temp entities interpret baseline origin as velocity.
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
// }
|
|
// }
|
|
}
|
|
|
|
void EV_Grenade(struct event_args_s* inArgs)
|
|
{
|
|
// What to do about this static member?
|
|
static int tracerCount[ 32 ];
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = inArgs->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(idx));
|
|
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
char* theSoundToPlay = NULL;
|
|
|
|
switch(inArgs->iparam2)
|
|
{
|
|
case 4:
|
|
case 7:
|
|
theSoundToPlay = kGRFireSound1;
|
|
break;
|
|
case 3:
|
|
case 6:
|
|
theSoundToPlay = kGRPrimeSound;
|
|
break;
|
|
}
|
|
|
|
if(theSoundToPlay)
|
|
gEngfuncs.pEventAPI->EV_PlaySound( idx, origin, CHAN_WEAPON, theSoundToPlay, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// EV_GetGunPosition( inArgs, vecSrc, origin );
|
|
// VectorCopy( forward, vecAiming );
|
|
//
|
|
// // Create grenade temp entity
|
|
// int theModelIndex;
|
|
// struct model_s* theModel = gEngfuncs.CL_LoadModel(kGRWModel, &theModelIndex);
|
|
// if(theModel)
|
|
// {
|
|
// TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
// if(theTempEntity)
|
|
// {
|
|
// vec3_t theStartPos, theEndPos;
|
|
// EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
// VectorMA(vecSrc, kGRBarrelLength, forward, theStartPos);
|
|
//
|
|
// VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
// theTempEntity->die += BALANCE_VAR(kGrenDetonateTime);
|
|
// theTempEntity->hitcallback = GrenadeHit;
|
|
// theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_GRAVITY | FTENT_ROTATE | FTENT_PERSIST);
|
|
// theTempEntity->clientIndex = inArgs->entindex; // Entity to ignore collisions with
|
|
//
|
|
// theTempEntity->entity.baseline.angles[0] = rand()%360;
|
|
//
|
|
// //theTempEntity->entity.curstate.framerate = 30;
|
|
// //theTempEntity->frameMax = 4;//theModel->numframes;
|
|
//
|
|
// // Temp entities interpret baseline origin as velocity.
|
|
// Vector theBaseVelocity;
|
|
// VectorScale(inArgs->velocity, kGrenadeParentVelocityScalar, theBaseVelocity);
|
|
//
|
|
// Vector theStartVelocity;
|
|
// VectorMA(theBaseVelocity, kGrenadeVelocity, forward, theStartVelocity);
|
|
//
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
// }
|
|
// }
|
|
}
|
|
|
|
char *EV_HLDM_DamageDecal( physent_t *pe );
|
|
|
|
void CreateDecal(struct pmtrace_s* inTrace)
|
|
{
|
|
physent_t *pe;
|
|
pe = gEngfuncs.pEventAPI->EV_GetPhysent( inTrace->ent );
|
|
if(pe)
|
|
{
|
|
char* inDecalName = EV_HLDM_DamageDecal( pe );
|
|
|
|
// Only decal brush models such as the world etc.
|
|
if ( inDecalName && inDecalName[0] && pe && ( pe->solid == SOLID_BSP || pe->movetype == MOVETYPE_PUSHSTEP ) )
|
|
{
|
|
if ( CVAR_GET_FLOAT( "r_decals" ) )
|
|
{
|
|
gEngfuncs.pEfxAPI->R_DecalShoot(
|
|
gEngfuncs.pEfxAPI->Draw_DecalIndex( gEngfuncs.pEfxAPI->Draw_DecalIndexFromName( inDecalName) ),
|
|
gEngfuncs.pEventAPI->EV_IndexFromTrace( inTrace ), 0, inTrace->endpos, 0 );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void SpitHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
{
|
|
char* theSoundToPlay = kSpitHitSound1;
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
theSoundToPlay = kSpitHitSound2;
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(ent->entity.index, ptr->endpos, CHAN_AUTO, theSoundToPlay, .6f, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
if(ptr && ent)
|
|
{
|
|
ent->die = gEngfuncs.GetClientTime();
|
|
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSpitHit, ptr->endpos, &(ptr->plane.normal));
|
|
|
|
// Create splat here too
|
|
CreateDecal(ptr);
|
|
}
|
|
}
|
|
|
|
void EV_SpitGun(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = inArgs->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = kSpitGFireSound1;
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
theSoundToPlay = kSpitGFireSound2;
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Fire spit glob
|
|
int theModelIndex;
|
|
struct model_s* theModel = gEngfuncs.CL_LoadModel(kSpitGunSprite, &theModelIndex);
|
|
if(theModel)
|
|
{
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
if(theTempEntity)
|
|
{
|
|
vec3_t theStartPos, theEndPos;
|
|
EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
VectorMA(vecSrc, kSpitGBarrelLength, forward, theStartPos);
|
|
|
|
// Create tiny spittle where it leaves
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSpitShoot, theStartPos);
|
|
|
|
VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
theTempEntity->die += kSpitLifetime;
|
|
theTempEntity->hitcallback = SpitHit;
|
|
theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP/* | FTENT_PERSIST*/);
|
|
theTempEntity->clientIndex = inArgs->entindex; // Entity to ignore collisions with
|
|
theTempEntity->entity.curstate.framerate = 30;
|
|
theTempEntity->frameMax = 4;//theModel->numframes;
|
|
|
|
// Temp entities interpret baseline origin as velocity.
|
|
Vector theBaseVelocity;
|
|
VectorScale(inArgs->velocity, kSpitParentVelocityScalar, theBaseVelocity);
|
|
|
|
Vector theStartVelocity;
|
|
VectorMA(theBaseVelocity, kSpitVelocity, forward, theStartVelocity);
|
|
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
}
|
|
}
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
float theHalfSpread = kSpitGXPunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void SpikeHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
{
|
|
//EV_HLDM_DecalGunshot( &tr, iBulletType );
|
|
|
|
//int theSprite = gEngfuncs.pEventAPI->EV_FindModelIndex(kSpikeGunHitSprite);
|
|
//TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->R_TempSprite(ptr->endpos, vec3_origin, .6f, theSprite, kRenderTransAdd, kRenderFxNoDissipation, .5f, .4f, FTENT_COLLIDEALL | FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP | FTENT_PERSIST);
|
|
//if(theTempEntity)
|
|
//{
|
|
// theTempEntity->entity.curstate.framerate = 30;
|
|
//}
|
|
|
|
// Play spike ricochet sounds
|
|
char* theRandomSound = NULL;
|
|
|
|
switch (gEngfuncs.pfnRandomLong(0, 2))
|
|
{
|
|
case 0:
|
|
theRandomSound = "weapons/a_ric1.wav";
|
|
break;
|
|
case 1:
|
|
theRandomSound = "weapons/a_ric2.wav";
|
|
break;
|
|
case 2:
|
|
theRandomSound = "weapons/a_ric3.wav";
|
|
break;
|
|
}
|
|
|
|
const float kAlienRicochetVolume = .25f;
|
|
ASSERT(theRandomSound);
|
|
gEngfuncs.pEventAPI->EV_PlaySound( 0, ptr->endpos, CHAN_STATIC, theRandomSound, kAlienRicochetVolume, ATTN_NORM, 0, 96 + gEngfuncs.pfnRandomLong(0,0xf) );
|
|
|
|
// Create spike hit particle system
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSpikeHitEffect, ptr->endpos);
|
|
|
|
// Kill off temp ent
|
|
ent->die = -1;
|
|
}
|
|
|
|
void EV_OffenseChamber(struct event_args_s* inArgs)
|
|
{
|
|
//static int theTracerCount[ 32 ];
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
// Play one of basic attack sounds
|
|
float theVolume = inArgs->fparam1;
|
|
float theAttenuation = 1.2f;
|
|
|
|
char* theSoundToPlay = kAlienTurretFire1;
|
|
|
|
int theUpperBound = 30;
|
|
int thePitch = 100 + (gEngfuncs.pfnRandomLong(0, theUpperBound) - theUpperBound/2);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, theSoundToPlay, theVolume, theAttenuation, 0, thePitch);
|
|
|
|
// Read origin as origin, and angles as direction (from AvHAlienTurret::Shoot())
|
|
Vector theStartPos;
|
|
VectorCopy(inArgs->origin, theStartPos);
|
|
|
|
Vector theNetworkDirToEnemy;
|
|
VectorCopy(inArgs->angles, theNetworkDirToEnemy);
|
|
|
|
// Take into account network precision
|
|
Vector theDirToEnemy;
|
|
VectorScale(theNetworkDirToEnemy, 1/100.0f, theDirToEnemy);
|
|
|
|
Vector theDirToEnemyAngles;
|
|
VectorAngles(theDirToEnemy, theDirToEnemyAngles);
|
|
|
|
Vector theInitialVelocity;
|
|
VectorScale(theDirToEnemy, kOffenseChamberSpikeVelocity, theInitialVelocity);
|
|
|
|
// Create spike projectile
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->R_TempModel(theStartPos, theDirToEnemy, theDirToEnemyAngles, 100, gEngfuncs.pEventAPI->EV_FindModelIndex(kSpikeProjectileModel), 0);
|
|
if(theTempEntity)
|
|
{
|
|
//vec3_t theStartPos, theEndPos;
|
|
//EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
//VectorMA(vecSrc, kSpikeBarrelLength, forward, theStartPos);
|
|
|
|
VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
theTempEntity->hitcallback = SpikeHit;
|
|
theTempEntity->flags = (FTENT_COLLIDEALL | FTENT_PERSIST | FTENT_COLLIDEKILL);
|
|
theTempEntity->die += kSpikeLifetime;
|
|
theTempEntity->clientIndex = inArgs->entindex; // Entity to ignore collisions with
|
|
|
|
//theTempEntity->entity.curstate.framerate = 30;
|
|
//theTempEntity->frameMax = 4;//theModel->numframes;
|
|
|
|
// Temp entities interpret baseline origin as velocity.
|
|
VectorCopy(theInitialVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theInitialVelocity, theTempEntity->entity.baseline.velocity);
|
|
|
|
// Set orientation
|
|
//VectorCopy(inArgs->angles, theTempEntity->entity.angles);
|
|
}
|
|
|
|
|
|
//
|
|
//
|
|
// ASSERT(inArgs);
|
|
//
|
|
// // Play attack animation and add muzzle flash
|
|
// int idx = inArgs->entindex;
|
|
//
|
|
// Vector theStartPos;
|
|
// VectorCopy(inArgs->origin, theStartPos);
|
|
//
|
|
// // Play attack sound
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(idx, theStartPos, CHAN_WEAPON, kAlienTurretFire1, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//
|
|
// // Fire spit glob
|
|
// int theModelIndex;
|
|
// struct model_s* theModel = gEngfuncs.CL_LoadModel(kAlienTurretSprite, &theModelIndex);
|
|
// if(theModel)
|
|
// {
|
|
// TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
// if(theTempEntity)
|
|
// {
|
|
// // Create tiny spittle where it leaves
|
|
// AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSpitShoot, theStartPos);
|
|
//
|
|
// VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
// theTempEntity->die += kSpitLifetime;
|
|
// theTempEntity->hitcallback = SpitHit;
|
|
// theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP/* | FTENT_PERSIST*/);
|
|
// theTempEntity->entity.curstate.framerate = 30;
|
|
// theTempEntity->frameMax = 4;//theModel->numframes;
|
|
//
|
|
// // Read origin as origin, and angles as direction (from AvHAlienTurret::Shoot())
|
|
// Vector theDirToEnemy;
|
|
// VectorCopy(inArgs->angles, theDirToEnemy);
|
|
//
|
|
// Vector theStartVelocity;
|
|
// VectorScale(theDirToEnemy, kAlienTurretProjectileVelocity, theStartVelocity);
|
|
//
|
|
// // Temp entities interpret baseline origin as velocity.
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
// }
|
|
// }
|
|
}
|
|
|
|
void EV_Claws(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 2);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
theSoundToPlay = kClawsSound1;
|
|
break;
|
|
case 1:
|
|
theSoundToPlay = kClawsSound2;
|
|
break;
|
|
case 2:
|
|
theSoundToPlay = kClawsSound3;
|
|
break;
|
|
}
|
|
|
|
const int kBasePitch = 94;
|
|
const int kVariablePitchRange = 0xF;
|
|
|
|
int theRandomPitch = gEngfuncs.pfnRandomLong( 0, kVariablePitchRange );
|
|
int thePitch = kBasePitch + (inArgs->fparam1*kClawsAdrenPitchFactor) + theRandomPitch;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, thePitch);
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
|
|
/*
|
|
float theHalfSpread = (inArgs->fparam1*kClawsPunch)/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
*/
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
int theRandomLong = gEngfuncs.pfnRandomLong(0, 1);
|
|
if(theRandomLong == 0)
|
|
{
|
|
PlayMeleeHitEffects(inArgs, kClawsRange, kClawsHitSound1);
|
|
}
|
|
else
|
|
{
|
|
PlayMeleeHitEffects(inArgs, kClawsRange, kClawsHitSound2);
|
|
}
|
|
}
|
|
|
|
|
|
void EV_Swipe(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 3);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
theSoundToPlay = kSwipeSound1;
|
|
break;
|
|
case 1:
|
|
theSoundToPlay = kSwipeSound2;
|
|
break;
|
|
case 2:
|
|
theSoundToPlay = kSwipeSound3;
|
|
break;
|
|
case 3:
|
|
theSoundToPlay = kSwipeSound4;
|
|
break;
|
|
}
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
|
|
/*
|
|
float theHalfSpread = kSwipePunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
*/
|
|
}
|
|
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
PlayMeleeHitEffects(inArgs, kSwipeRange, kSwipeHitSound1);
|
|
}
|
|
else
|
|
{
|
|
PlayMeleeHitEffects(inArgs, kSwipeRange, kSwipeHitSound2);
|
|
}
|
|
}
|
|
|
|
//void EnsnareHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
//{
|
|
// ent->die = gEngfuncs.GetClientTime();
|
|
//}
|
|
//
|
|
//void EV_EnsnareShoot(struct event_args_s* inArgs)
|
|
//{
|
|
// int idx = inArgs->entindex;
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, kEnsnareFireSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//
|
|
// vec3_t up, right, forward;
|
|
// gEngfuncs.pfnAngleVectors(inArgs->angles, forward, right, up);
|
|
//
|
|
// // Fire spit glob
|
|
// int theModelIndex;
|
|
// struct model_s* theModel = gEngfuncs.CL_LoadModel(kEnsnareSprite, &theModelIndex);
|
|
// ASSERT(theModel);
|
|
// TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
// if(theTempEntity)
|
|
// {
|
|
// vec3_t vecSrc, vecAiming;
|
|
// vec3_t theStartPos, theEndPos;
|
|
// EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
// VectorMA(vecSrc, kEnsnareBarrelLength, forward, theStartPos);
|
|
//
|
|
// // Create tiny ensnare bits where it leaves
|
|
// AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsEnsnareShoot, theStartPos);
|
|
//
|
|
// VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
// theTempEntity->die += 10.0f;
|
|
// theTempEntity->hitcallback = EnsnareHit;
|
|
// theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP | FTENT_PERSIST);
|
|
// theTempEntity->entity.curstate.framerate = 30;
|
|
// theTempEntity->frameMax = 4;//theModel->numframes;
|
|
//
|
|
// // Temp entities interpret baseline origin as velocity.
|
|
// Vector theBaseVelocity;
|
|
// VectorScale(inArgs->velocity, kEnsnareParentVelocityScalar, theBaseVelocity);
|
|
//
|
|
// Vector theStartVelocity;
|
|
// VectorMA(theBaseVelocity, kEnsnareVelocity, forward, theStartVelocity);
|
|
//
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
// }
|
|
//}
|
|
//
|
|
//void EV_EnsnareHit(struct event_args_s* inArgs)
|
|
//{
|
|
// int idx = inArgs->entindex;
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, kEnsnareHitSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//
|
|
// AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsEnsnareHit, inArgs->origin);
|
|
//}
|
|
|
|
void ShootCloud(struct event_args_s* inArgs, const char* inSpriteName)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, kSporeFireSound, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
// Create temp cloud projectile
|
|
int theModelIndex;
|
|
struct model_s* theModel = gEngfuncs.CL_LoadModel(inSpriteName, &theModelIndex);
|
|
if(theModel)
|
|
{
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
if(theTempEntity)
|
|
{
|
|
vec3_t angles;
|
|
VectorCopy( inArgs->angles, angles );
|
|
|
|
//vec3_t velocity
|
|
//VectorCopy( inArgs->velocity, velocity );
|
|
|
|
vec3_t forward, right, up;
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
vec3_t theStartPos, theEndPos, vecSrc;
|
|
EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
VectorMA(vecSrc, kSporeBarrelLength, forward, theStartPos);
|
|
|
|
VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
//theTempEntity->die += kSpitLifetime;
|
|
//theTempEntity->hitcallback = SpitHit;
|
|
theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP | FTENT_COLLIDEKILL);
|
|
theTempEntity->entity.curstate.framerate = 20;
|
|
theTempEntity->frameMax = 17;//theModel->numframes;
|
|
|
|
// Temp entities interpret baseline origin as velocity.
|
|
Vector theStartVelocity(0, 0, 0);
|
|
VectorMA(theStartVelocity, kShootCloudVelocity, forward, theStartVelocity);
|
|
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_SporeShoot(struct event_args_s* inArgs)
|
|
{
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSporeShoot, inArgs->origin);
|
|
|
|
ShootCloud(inArgs, kClientSporeSprite);
|
|
}
|
|
|
|
void EV_SporeCloud(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_AUTO, kSporeCloudSound, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSporeCloud, inArgs->origin);
|
|
}
|
|
|
|
void EV_UmbraGun(struct event_args_s* inArgs)
|
|
{
|
|
ShootCloud(inArgs, kClientUmbraSprite);
|
|
|
|
// if (EV_IsLocal(inArgs->entindex))
|
|
// {
|
|
// gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
// }
|
|
}
|
|
|
|
void EV_UmbraCloud(struct event_args_s* inArgs)
|
|
{
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kUmbraFireSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsUmbraCloud, inArgs->origin);
|
|
}
|
|
|
|
void ParticleCallback( struct particle_s* particle, float frametime )
|
|
{
|
|
int i;
|
|
|
|
for ( i = 0; i < 3; i++ )
|
|
{
|
|
particle->org[ i ] += particle->vel[ i ] * frametime;
|
|
}
|
|
}
|
|
|
|
|
|
void EV_Jetpack(struct event_args_s* inArgs)
|
|
{
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsJetpackEffect, inArgs->origin);
|
|
|
|
// Update jetpack sound every once in awhile
|
|
if(gEngfuncs.pfnRandomLong(0, 3) == 0)
|
|
{
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_BODY, kJetpackSound, .5f, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
|
|
// Use this as index instead?
|
|
//args->entindex;
|
|
|
|
// // Create a dlight for jetpack
|
|
// int theIndexToUse = gJetpackLights.size();
|
|
// dlight_t* dl = gEngfuncs.pEfxAPI->CL_AllocDlight(theIndexToUse);
|
|
// VectorCopy (args->origin, dl->origin);
|
|
// dl->radius = 180;
|
|
// dl->color.r = 180;
|
|
// dl->color.g = 180;
|
|
// dl->color.b = 250;
|
|
//
|
|
// // don't die for forseeable future
|
|
// dl->die = gEngfuncs.GetClientTime() + kArbitraryLargeLightTime;
|
|
// //dl->die += kArbitraryLargeLightTime;
|
|
//
|
|
// gJetpackLights.push_back(LightType(args->entindex, dl));
|
|
//
|
|
// gEngfuncs.pEventAPI->EV_PlaySound( args->entindex, args->origin, CHAN_BODY, kJetpackSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Create temporary smoke trail!
|
|
//void ( *R_Sprite_Trail ) ( int type, float * start, float * end, int modelIndex, int count, float life, float size, float amplitude, int renderamt, float speed );
|
|
//int theSmokeModelIndex = gEngfuncs.pEventAPI->EV_FindModelIndex( "sprites/smoke.spr" );
|
|
//gEngfuncs.pEfxAPI->R_PlayerSprites(args->entindex, theSmokeModelIndex, 20, 100);
|
|
|
|
//gEngfuncs.pEfxAPI->R_Sprite_Trail( TE_SPRITETRAIL, tr.endpos, fwd, theSmokeModelIndex, (int)(n * flDamage * 0.3), 0.1, gEngfuncs.pfnRandomFloat( 10, 20 ) / 100.0, 100, 255, 200 );
|
|
|
|
}
|
|
|
|
// Passing no param resets all sounds
|
|
void EndJetpackEffects(int inIndex)
|
|
{
|
|
for(DLightListType::iterator theIter = gJetpackLights.begin(); theIter != gJetpackLights.end(); theIter++)
|
|
{
|
|
if((theIter->mIndex == inIndex) || (inIndex == -1))
|
|
{
|
|
// Mark it for deletion by telling it to die a long time ago
|
|
if(theIter->mLight)
|
|
{
|
|
theIter->mLight->die = 0;
|
|
}
|
|
gJetpackLights.erase(theIter);
|
|
|
|
vec3_t theOrigin;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inIndex, theOrigin, CHAN_BODY, "common/null.wav", 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_EndJetpack(struct event_args_s* args)
|
|
{
|
|
// find dlight in list
|
|
//int theIndex = args->entindex;
|
|
//EndJetpackEffects(theIndex);
|
|
}
|
|
|
|
void DrawCircleOnGroundAtPoint(vec3_t inOrigin, int inNumSides, int inStartAngle, int inRadius, float inR, float inG, float inB, float inA, bool inUseRedInstead, float inInnerRadius)
|
|
{
|
|
static HSPRITE theGreenSprite = 0;
|
|
if(!theGreenSprite)
|
|
theGreenSprite = Safe_SPR_Load("sprites/green.spr");
|
|
|
|
static HSPRITE theRedSprite = 0;
|
|
if(!theRedSprite)
|
|
theRedSprite = Safe_SPR_Load("sprites/red.spr");
|
|
|
|
HSPRITE theSprite = theGreenSprite;
|
|
if(inUseRedInstead)
|
|
theSprite = theRedSprite;
|
|
|
|
if(gEngfuncs.pTriAPI->SpriteTexture((struct model_s *)gEngfuncs.GetSpritePointer(theSprite), 0))
|
|
{
|
|
gEngfuncs.pTriAPI->CullFace( TRI_NONE );
|
|
|
|
gEngfuncs.pTriAPI->RenderMode( kRenderGlow );
|
|
//gEngfuncs.pTriAPI->RenderMode( kRenderTransTexture );
|
|
gEngfuncs.pTriAPI->Color4f(inR, inG, inB, inA);
|
|
|
|
gEngfuncs.pTriAPI->Begin( TRI_TRIANGLES );
|
|
float theAngleIncrement = (2*M_PI)/inNumSides;
|
|
float theStartAngle = (2*M_PI)*(inStartAngle/360.0f);
|
|
|
|
for(int i = 0; i < inNumSides; i++)
|
|
{
|
|
for(int j = 0; j < 3; j++)
|
|
{
|
|
float thePoint[3];
|
|
float theCurrentPointAngle = theStartAngle + i*theAngleIncrement;
|
|
thePoint[0] = inOrigin[0] + inRadius*cos(theCurrentPointAngle);
|
|
thePoint[1] = inOrigin[1] + inRadius*sin(theCurrentPointAngle);
|
|
thePoint[2] = inOrigin[2];
|
|
|
|
float theNextPoint[3];
|
|
float theNextPointAngle = theStartAngle + (i+1)*theAngleIncrement;
|
|
theNextPoint[0] = inOrigin[0] + inRadius*cos(theNextPointAngle);
|
|
theNextPoint[1] = inOrigin[1] + inRadius*sin(theNextPointAngle);
|
|
theNextPoint[2] = inOrigin[2];
|
|
|
|
//gEngfuncs.pTriAPI->Color4f(inR, inG, inB, 1.0f);
|
|
//gEngfuncs.pTriAPI->Brightness( 1 );
|
|
gEngfuncs.pTriAPI->TexCoord2f(0, 0);
|
|
gEngfuncs.pTriAPI->Vertex3f( inOrigin[0], inOrigin[1], inOrigin[2] );
|
|
|
|
//gEngfuncs.pTriAPI->Color4f(inR, inG, inB, inA);
|
|
//gEngfuncs.pTriAPI->Brightness( 1 );
|
|
gEngfuncs.pTriAPI->TexCoord2f(0, 1);
|
|
gEngfuncs.pTriAPI->Vertex3f( theNextPoint[0], theNextPoint[1], theNextPoint[2] );
|
|
|
|
//gEngfuncs.pTriAPI->Color4f(inR, inG, inB, inA);
|
|
//gEngfuncs.pTriAPI->Brightness( 1 );
|
|
gEngfuncs.pTriAPI->TexCoord2f(1, 1);
|
|
gEngfuncs.pTriAPI->Vertex3f( thePoint[0], thePoint[1], thePoint[2] );
|
|
|
|
}
|
|
}
|
|
gEngfuncs.pTriAPI->End();
|
|
gEngfuncs.pTriAPI->RenderMode( kRenderNormal );
|
|
}
|
|
}
|
|
|
|
//void DrawOrderDirectionIndicator(const AvHOrder& inOrder)
|
|
//{
|
|
// static HSPRITE theSprite = 0;
|
|
// if(!theSprite)
|
|
// theSprite = Safe_SPR_Load("sprites/320questionmark.spr");
|
|
//
|
|
// int theCurrentFrame = 0;
|
|
//
|
|
// //
|
|
// vec3_t theScreenPos;
|
|
// vec3_t theWorldPos;
|
|
// inOrder.GetLocation(theWorldPos);
|
|
|
|
// gEngfuncs.pTriAPI->WorldToScreen((float*)theWorldPos, (float*)theScreenPos);
|
|
//
|
|
// int theBaseX = theScreenPos.x*ScreenWidth;
|
|
// int theBaseY = theScreenPos.y*ScreenHeight;
|
|
//
|
|
// int theSpriteWidth = SPR_Width(theSprite, theCurrentFrame);
|
|
// int theSpriteHeight = SPR_Height(theSprite, theCurrentFrame);
|
|
//
|
|
// int theX = min(max(theSpriteWidth, theBaseX), ScreenWidth - theSpriteWidth);
|
|
// int theY = min(max(theSpriteHeight, theBaseY), ScreenHeight - theSpriteHeight);
|
|
//
|
|
// SPR_Set(theSprite, 255, 255, 255);
|
|
// SPR_DrawHoles((int)0, theX, theY, NULL);
|
|
|
|
// gHUD.SetOrderPos(theWorldPos);
|
|
//}
|
|
|
|
//void DrawOrdersForPlayers(EntityListType& inPlayerList)
|
|
//{
|
|
// OrderListType theOrders = gHUD.GetOrderList();
|
|
//
|
|
// gHUD.ClearOrderPos();
|
|
//
|
|
// // Run through the order list type
|
|
// for(OrderListType::iterator theIter = theOrders.begin(); theIter != theOrders.end(); theIter++)
|
|
// {
|
|
// // For each one, if the order is for a player in the inPlayerList, draw it
|
|
// vec3_t theOrderLocation;
|
|
// theIter->GetLocation(theOrderLocation);
|
|
//
|
|
// if(theIter->GetOrderTargetType() == ORDERTARGETTYPE_TARGET)
|
|
// {
|
|
// int theTargetIndex = theIter->GetTargetIndex();
|
|
// cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(theTargetIndex);
|
|
// if(theEntity)
|
|
// {
|
|
// VectorCopy(theEntity->origin, theOrderLocation);
|
|
// }
|
|
// }
|
|
//
|
|
// // Draw dotted line or something from relevant player to dest?
|
|
// bool theDrawWaypoint = false;
|
|
// EntityListType thePlayerList = theIter->GetReceivers();
|
|
// for(EntityListType::iterator thePlayerIter = thePlayerList.begin(); thePlayerIter != thePlayerList.end(); thePlayerIter++)
|
|
// {
|
|
// cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(*thePlayerIter);
|
|
// if(theEntity)
|
|
// {
|
|
// EntityListType::iterator theSearchIter = std::find(inPlayerList.begin(), inPlayerList.end(), *thePlayerIter);
|
|
// if(theSearchIter != inPlayerList.end())
|
|
// {
|
|
// //gEngfuncs.pEfxAPI->R_ParticleLine((float*)theEntity->origin, (float*)theOrderLocation, 0, 255, 0, .05f);
|
|
// theDrawWaypoint = true;
|
|
// }
|
|
// }
|
|
// }
|
|
// if(theDrawWaypoint)
|
|
// {
|
|
// float r, g, b, a;
|
|
// theIter->GetOrderColor(r, g, b, a);
|
|
//
|
|
// // This doesn't really use the color yet
|
|
// DrawCircleOnGroundAtPoint(theOrderLocation, 11, 0, 20, r, g, b, a, false, 0.0f);
|
|
//
|
|
// // Draw HUD indicator at this position
|
|
// DrawOrderDirectionIndicator(*theIter);
|
|
// }
|
|
// }
|
|
//}
|
|
|
|
//void DrawBlips(const pVector& inView)
|
|
//{
|
|
// // Get enemy blips from HUD and draw 'em
|
|
// AvHVisibleBlipList& theEnemyBlipList = gHUD.GetEnemyBlipList();
|
|
// theEnemyBlipList.Draw(inView);
|
|
//
|
|
// // Get friendly blips from HUD and draw 'em
|
|
// AvHVisibleBlipList& theFriendlyBlipList = gHUD.GetFriendlyBlipList();
|
|
// theFriendlyBlipList.Draw(inView);
|
|
//}
|
|
|
|
void DrawDebugEffects()
|
|
{
|
|
for(DebugPointListType::iterator theIter = gTriDebugLocations.begin(); theIter != gTriDebugLocations.end(); theIter++)
|
|
{
|
|
vec3_t thePosition;
|
|
thePosition.x = theIter->x;
|
|
thePosition.y = theIter->y;
|
|
thePosition.z = theIter->z;
|
|
int theRadius = theIter->mSize;
|
|
DrawCircleOnGroundAtPoint(thePosition, 3, 0, theRadius, 1, 1, 0, .5f, false, 0.0f);
|
|
}
|
|
for(theIter = gSquareDebugLocations.begin(); theIter != gSquareDebugLocations.end(); theIter++)
|
|
{
|
|
vec3_t thePosition;
|
|
thePosition.x = theIter->x;
|
|
thePosition.y = theIter->y;
|
|
thePosition.z = theIter->z;
|
|
int theRadius = theIter->mSize;
|
|
DrawCircleOnGroundAtPoint(thePosition, 4, 0, theRadius, 1, 1, 0, .5f, false, 0.0f);
|
|
}
|
|
gSquareDebugLocations.clear();
|
|
|
|
// Run through particle systems, drawing a box around each one
|
|
|
|
// for(DebugEntityListType::iterator theCubeIter = gCubeDebugEntities.begin(); theCubeIter != gCubeDebugEntities.end(); theCubeIter++)
|
|
// {
|
|
// cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(*theCubeIter);
|
|
// if(theEntity)
|
|
// {
|
|
// Vector theStartPos = theEntity->curstate.origin + theEntity->curstate.mins;
|
|
// Vector theEndPos = theEntity->curstate.origin + theEntity->curstate.maxs;
|
|
// //gEngfuncs.pEfxAPI->R_ParticleBox( (float *)&(theStartPos), (float *)&(theEndPos), 5.0, 0, 255, .5f );
|
|
// float theMiddleX = (theStartPos.x + theEndPos.x)/2.0f;
|
|
// float theMiddleY = (theStartPos.y + theEndPos.y)/2.0f;
|
|
// Vector thePoint;
|
|
// thePoint.x = theMiddleX;
|
|
// thePoint.y = theMiddleY;
|
|
// thePoint.z = theStartPos.z;
|
|
// DrawCircleOnGroundAtPoint(thePoint, 4, 45, 64, .8f, .8f, .8f, .5f, false, 0.0f);
|
|
//
|
|
// thePoint.z = theEndPos.z;
|
|
// DrawCircleOnGroundAtPoint(thePoint, 4, 45, 64, .8f, .8f, .8f, .5f, false, 0.0f);
|
|
// }
|
|
// }
|
|
}
|
|
|
|
void DrawMarineLight(const pVector& inView, vec3_t& inStartPos, vec3_t& inEndPos)
|
|
{
|
|
gEngfuncs.pEfxAPI->R_RocketTrail(inStartPos, inEndPos, 1);
|
|
|
|
// // Draw triangle strip that faces camera
|
|
// const float kBeamAlpha = .7f;
|
|
// const float kBeamWidth = 10;
|
|
//
|
|
// pVector up(0, 0, 1);
|
|
// pVector right = inView ^ up;
|
|
// right.normalize();
|
|
// pVector nup = right ^ inView;
|
|
// //pVector nup = inView ^ right;
|
|
//
|
|
// // The particles should face you unless constrained (this is for rain and the like)
|
|
// if(this->mConstrainPitch)
|
|
// nup = up;
|
|
//
|
|
// //right *= size_scale/2.0f;
|
|
// //nup *= size_scale/2.0f;
|
|
//
|
|
// // Quad draws v0, v1, v2, v3
|
|
// //pVector V0 = -(right + nup);
|
|
//
|
|
// // Upper left
|
|
// pVector V0 = -(right + nup);
|
|
// //V0 = -(right - nup);
|
|
// V0 = nup - right;
|
|
//
|
|
// // Lower left
|
|
// pVector V1 = -(right + nup);
|
|
//
|
|
// // Lower right
|
|
// pVector V2 = right + nup;
|
|
// V2 = right - nup;
|
|
//
|
|
// // Upper right
|
|
// pVector V3 = right - nup;
|
|
// V3 = right + nup;
|
|
//
|
|
// // Tri draws v0, v4, v3
|
|
// //pVector V4 = nup;
|
|
// pVector V4 = -nup;
|
|
//
|
|
// // Load red sprite
|
|
// //if(gEngfuncs.pTriAPI->SpriteTexture((struct model_s *)gEngfuncs.GetSpritePointer(this->mSprite), theTextureOffsetToUse))
|
|
// //{
|
|
// gEngfuncs.pTriAPI->Begin( TRI_TRIANGLES );
|
|
//
|
|
// gEngfuncs.pTriAPI->Color4f(1.0f, 0.0f, 0.0f, kBeamAlpha);
|
|
//
|
|
// //gEngfuncs.pTriAPI->TexCoord2f(0,0);
|
|
// pVector ver = p + sV0;
|
|
// gEngfuncs.pTriAPI->Vertex3fv((float*)&ver);
|
|
//
|
|
// //gEngfuncs.pTriAPI->TexCoord2f(.5,1);
|
|
// ver = p + sV4;
|
|
// gEngfuncs.pTriAPI->Vertex3fv((float*)&ver);
|
|
//
|
|
// //gEngfuncs.pTriAPI->TexCoord2f(1,0);
|
|
// ver = p + sV3;
|
|
// gEngfuncs.pTriAPI->Vertex3fv((float*)&ver);
|
|
//
|
|
// gEngfuncs.pTriAPI->End();
|
|
// //}
|
|
}
|
|
|
|
void DrawMarineLights(const pVector& inView)
|
|
{
|
|
gEngfuncs.pEventAPI->EV_SetUpPlayerPrediction( false, false );
|
|
|
|
// 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 player is a marine
|
|
if(theEntity->iuser3 == AVH_USER3_MARINE_PLAYER)
|
|
{
|
|
// Get middle of head
|
|
//theEntity->model->
|
|
//model->attachment = model->origin;
|
|
|
|
// Get eyepiece attachment
|
|
cl_entity_t* theClientEntity = gEngfuncs.GetEntityByIndex(theEntity->info);
|
|
if(theClientEntity)
|
|
{
|
|
Vector theHeadPos = theClientEntity->attachment[2];
|
|
|
|
// Get look direction
|
|
Vector theForward, theRight, theUp;
|
|
|
|
gEngfuncs.pfnAngleVectors(theClientEntity->angles, theForward, theRight, theUp);
|
|
Vector theAiming;
|
|
for(int i = 0; i < 3; i++)
|
|
{
|
|
theAiming[i] = theForward[i] + theRight[i] + theUp[i];
|
|
}
|
|
|
|
// Build this line
|
|
Vector theStartPos = theHeadPos;
|
|
Vector theEndPos;
|
|
VectorMA(theStartPos, 8000, theAiming, theEndPos);
|
|
|
|
// Perform trace, ignore glass (!), ignore ourself
|
|
gEngfuncs.pEventAPI->EV_SetTraceHull(2);
|
|
//int theFoundEntity = theEntity->info;
|
|
int theFoundEntity = -1;
|
|
pmtrace_t tr;
|
|
|
|
bool theDone = false;
|
|
do
|
|
{
|
|
// NOTE: ** When implementing this again, put in check for infinite loop condition! **
|
|
|
|
// Run a traceline along this line until it hits
|
|
gEngfuncs.pEventAPI->EV_PlayerTrace(theStartPos, theEndPos, PM_GLASS_IGNORE, theFoundEntity, &tr);
|
|
|
|
//if((tr.fraction >= (1.0f - kFloatTolerance)) || (tr.fraction == 0.0f)/*|| (theNumIterations > 100)*//* || (tr.fraction < kFloatTolerance)*/)
|
|
//{
|
|
// theDone = true;
|
|
//}
|
|
//else
|
|
//{
|
|
// // Offset a bit so we don't hit again
|
|
// VectorMA(tr.endpos, kHitOffsetAmount, theForward, theStartPos);
|
|
//}
|
|
|
|
if(tr.startsolid)
|
|
{
|
|
VectorMA(tr.endpos, kHitOffsetAmount, theForward, theStartPos);
|
|
}
|
|
else
|
|
{
|
|
theDone = true;
|
|
}
|
|
|
|
} while(!theDone);
|
|
|
|
// Draw red additive light along this line
|
|
DrawMarineLight(inView, theStartPos, tr.endpos);
|
|
}
|
|
}
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_PopPMStates();
|
|
}
|
|
|
|
//void DrawRangeIndicator()
|
|
//{
|
|
// Vector theCenter;
|
|
// bool theIsValid;
|
|
// float theRadius;
|
|
//
|
|
// if(gHUD.GetRangeDrawingInfo(theCenter, theRadius, theIsValid))
|
|
// {
|
|
// const float theCircleWidth = 20.0f;
|
|
// float theInnerRadius = theRadius - theCircleWidth;
|
|
//
|
|
// static theAngle = 0;
|
|
// DrawCircleOnGroundAtPoint(theCenter, theRadius/200, theAngle++, theRadius, 1.0f, 1.0f, 1.0f, 1.0f, !theIsValid, theInnerRadius);
|
|
// }
|
|
//}
|
|
|
|
//void UpdateJetpackLights()
|
|
//{
|
|
// for(DLightListType::iterator theIter = gJetpackLights.begin(); theIter != gJetpackLights.end(); theIter++)
|
|
// {
|
|
// // Treat locally differently
|
|
// int theIndex = theIter->mIndex;
|
|
// vec3_t theOrigin;
|
|
//
|
|
// if(theIndex == 0)
|
|
// {
|
|
// VectorCopy(gPredictedPlayerOrigin, theOrigin);
|
|
// }
|
|
// else
|
|
// {
|
|
// cl_entity_t* thePlayer = gEngfuncs.GetEntityByIndex(theIndex);
|
|
// if(thePlayer)
|
|
// {
|
|
// // Get known position of entity
|
|
// VectorCopy(thePlayer->origin, theOrigin);
|
|
// }
|
|
// else
|
|
// {
|
|
// ASSERT(false);
|
|
// }
|
|
// }
|
|
//
|
|
// VectorCopy(theOrigin, theIter->mLight->origin);
|
|
//
|
|
// // Make sure it still won't die
|
|
// theIter->mLight->die = gEngfuncs.GetClientTime() + kArbitraryLargeLightTime;
|
|
//
|
|
// // Update jetpack sound every once in awhile
|
|
// if(gEngfuncs.pfnRandomLong(0, 70) == 0)
|
|
// {
|
|
// gEngfuncs.pEventAPI->EV_PlaySound( theIndex, theOrigin, CHAN_BODY, kJetpackSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
// }
|
|
// }
|
|
//}
|
|
|
|
void EV_WelderGeneralEffects(struct event_args_s* inArgs, const Vector& thePos)
|
|
{
|
|
int theIndex = inArgs->entindex;
|
|
cl_entity_t* thePlayer = GetEntity(theIndex);
|
|
|
|
// Don't play flashing effects if player is photosensitive
|
|
if(CVAR_GET_FLOAT(kvDynamicLights))
|
|
{
|
|
// Make flashing lights
|
|
int theLightIndex = 20; /* TODO: What do to about this index? */
|
|
dlight_t* theLight = gEngfuncs.pEfxAPI->CL_AllocDlight(theLightIndex);
|
|
VectorCopy (thePlayer->origin, theLight->origin);
|
|
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(theIndex));
|
|
|
|
theLight->radius = 180 + theUpgradeLevel*15;
|
|
theLight->color.r = 220;
|
|
theLight->color.g = 220;
|
|
theLight->color.b = 255;
|
|
|
|
// Have it die before it fires again, so it flickers on and off
|
|
theLight->die = gEngfuncs.GetClientTime() + BALANCE_VAR(kWelderROF)/2.2f;
|
|
}
|
|
|
|
// TODO: Apply burn mark
|
|
//gEngfuncs.pEfxAPI->R_DecalShoot( gEngfuncs.pEfxAPI->Draw_DecalIndex( gEngfuncs.pEfxAPI->Draw_DecalIndexFromName( "{shot1" ) ),
|
|
// gEngfuncs.pEventAPI->EV_IndexFromTrace(&theTraceResult),
|
|
// 0,
|
|
// theTraceResult.endpos,
|
|
// 0 );
|
|
|
|
// emit heavy smoke
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsWelderSmoke, thePos);
|
|
}
|
|
}
|
|
|
|
void EV_WelderHitEffects(struct event_args_s* inArgs, const Vector& thePos)
|
|
{
|
|
int theIndex = inArgs->entindex;
|
|
cl_entity_t* thePlayer = GetEntity(theIndex);
|
|
|
|
// Fire sparks
|
|
//Vector theVector(inPev->origin);
|
|
//float theRandomFloat = RANDOM_FLOAT(0.5F, 1.0f);
|
|
//float theHeight = theRandomFloat*(inPev->absmax.z - inPev->absmin.z);
|
|
|
|
//gEngfuncs.pEfxAPI->R_SparkShower(theTraceResult.endpos);
|
|
//gEngfuncs.pEfxAPI->R_SparkShower(theTraceResult.endpos);
|
|
|
|
// Fire heavy smoke
|
|
//if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
//{
|
|
// AvHParticleSystemManager::Instance()->CreateParticleSystem(5, theTraceResult.endpos);
|
|
//}
|
|
|
|
// Generate still plasma
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsWelderBluePlasma, thePos);
|
|
}
|
|
|
|
// Generate plasma shower
|
|
if(gEngfuncs.pfnRandomLong(0, 8) == 0)
|
|
{
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsWelderBluePlasmaDrops, thePos);
|
|
}
|
|
|
|
// Generate spark shower
|
|
float theFloatVec[3];
|
|
thePos.CopyToArray(theFloatVec);
|
|
|
|
int theUpgradeLevel = AvHPlayerUpgrade::GetWeaponUpgrade(AVH_USER3_MARINE_PLAYER, GetUpgradeState(theIndex));
|
|
for(int i = 0; i < theUpgradeLevel+1; i++)
|
|
{
|
|
gEngfuncs.pEfxAPI->R_SparkShower(theFloatVec);
|
|
}
|
|
}
|
|
|
|
void EV_Welder(struct event_args_s* inArgs)
|
|
{
|
|
// This event means the welder is on, but we don't know if we're hitting anything or are constructing
|
|
int theIndex = inArgs->entindex;
|
|
cl_entity_t* thePlayer = GetEntity(theIndex);
|
|
|
|
// Hot blue flame
|
|
|
|
// See if we're hitting something and playback smoke and sparks if so
|
|
// Trace to see where the sparks should emanate from
|
|
vec3_t vecSrc;
|
|
vec3_t origin;
|
|
vec3_t forward, right, up;
|
|
|
|
gEngfuncs.pfnAngleVectors(inArgs->angles, forward, right, up);
|
|
|
|
// Calculate theStartPos, taking into account the welder barrel length
|
|
vec3_t theStartPos, theEndPos;
|
|
EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
VectorMA(vecSrc, kWelderBarrelLength, forward, theStartPos);
|
|
|
|
// Calculate theEndPos, welder range along facing
|
|
theEndPos = theStartPos + forward * BALANCE_VAR(kWelderRange);
|
|
|
|
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 );
|
|
|
|
gEngfuncs.pEventAPI->EV_SetTraceHull( 2 );
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(theIndex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
// Don't hit ourselves, but how? Setting this to theIndex isn't right, we need the other kind of index, but how to get it?
|
|
// For now this function will cause the welder to hit the player using the welder when aiming down, sigh
|
|
int theEntityToIngore = -1;
|
|
|
|
struct pmtrace_s theTraceResult;
|
|
gEngfuncs.pEventAPI->EV_PlayerTrace(theStartPos, theEndPos, PM_NORMAL, theEntityToIngore, &theTraceResult);
|
|
if(theTraceResult.fraction != 1.0f)
|
|
{
|
|
// Adjust the trace so it's offset a bit towards us so particles aren't clipped away
|
|
//Vector theResult = theTraceResult.endpos;//theStartPos + forward*(theTraceResult.fraction - 0.001);
|
|
Vector theResult = theStartPos + forward*((theTraceResult.fraction - 0.1f)*BALANCE_VAR(kWelderRange));
|
|
|
|
EV_WelderGeneralEffects(inArgs, theResult);
|
|
EV_WelderHitEffects(inArgs, theResult);
|
|
|
|
// Update status bar if possible
|
|
physent_t* thePhysEnt = gEngfuncs.pEventAPI->EV_GetPhysent( theTraceResult.ent );
|
|
//if(!thePhysEnt || !thePhysEnt->player)
|
|
if(thePhysEnt /*&& !thePhysEnt->player*/)
|
|
{
|
|
bool theIsBuilding;
|
|
bool theIsResearching;
|
|
float thePercentage;
|
|
AvHSHUGetBuildResearchState(thePhysEnt->iuser3, thePhysEnt->iuser4, thePhysEnt->fuser1, theIsBuilding, theIsResearching, thePercentage);
|
|
|
|
// if(thePhysEnt->iuser3 == AVH_USER3_WELD)
|
|
// {
|
|
// if((thePercentage < 1.0f) && (thePercentage != -1))
|
|
// {
|
|
// gHUD.SetProgressStatus(thePercentage);
|
|
// }
|
|
// else
|
|
// {
|
|
// gHUD.HideProgressStatus();
|
|
// }
|
|
// }
|
|
}
|
|
}
|
|
// Fire light smoke from muzzle
|
|
else //if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
// TODO: emit this from the actual end of the barrel
|
|
//AvHParticleSystemManager::Instance()->CreateParticleSystem(5, theStartPos + forward*10);
|
|
|
|
EV_WelderGeneralEffects(inArgs, theStartPos + (forward*(.3f*BALANCE_VAR(kWelderRange))));
|
|
|
|
// Stop playing hit sound
|
|
//gEngfuncs.pEventAPI->EV_StopSound( theIndex, CHAN_WEAPON, kWeldingHitSound);
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_PopPMStates();
|
|
}
|
|
|
|
//void ClientPlayRandomConstructionSound(struct event_args_s* inArgs)
|
|
//{
|
|
// int theRandomInteger = gEngfuncs.pfnRandomLong(1, 5);
|
|
// char theSoundName[128];
|
|
// sprintf(theSoundName, kMarineConstructionSounds, theRandomInteger);
|
|
//
|
|
// int thePitch = gEngfuncs.pfnRandomLong(100, 125);
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_BODY, theSoundName, 1, ATTN_NORM, SND_CHANGE_PITCH | SND_CHANGE_VOL, thePitch);
|
|
//}
|
|
|
|
// This event means the welder is constructing something, it's fired by the server
|
|
void EV_WelderConst(struct event_args_s* inArgs)
|
|
{
|
|
int theIndex = inArgs->entindex;
|
|
cl_entity_t* thePlayer = GetEntity(theIndex);
|
|
|
|
switch(inArgs->iparam1)
|
|
{
|
|
case 0://Start playing loop sound.
|
|
gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_BODY, kWeldingHitSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
break;
|
|
case 1://Stop playing loop sound.
|
|
gEngfuncs.pEventAPI->EV_StopSound( inArgs->entindex, CHAN_BODY, kWeldingHitSound);
|
|
gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_AUTO, kWeldingStopSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
break;
|
|
}
|
|
}
|
|
|
|
void EV_WelderStart(struct event_args_s* inArgs)
|
|
{
|
|
int theIndex = inArgs->entindex;
|
|
cl_entity_t* thePlayer = GetEntity(theIndex);
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_WEAPON, kWeldingSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
|
|
void EV_WelderEnd(struct event_args_s* inArgs)
|
|
{
|
|
gEngfuncs.pEventAPI->EV_StopSound( inArgs->entindex, CHAN_WEAPON, kWeldingSound);
|
|
|
|
// Hide the status, it will be shown below if possible
|
|
gHUD.HideProgressStatus();
|
|
}
|
|
|
|
//void EV_OverwatchStart(struct event_args_s* inArgs)
|
|
//{
|
|
// cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
// gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_VOICE, kOverwatchStartSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//}
|
|
//
|
|
//void EV_OverwatchEnd(struct event_args_s* inArgs)
|
|
//{
|
|
// cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
// gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_VOICE, kOverwatchEndSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//}
|
|
|
|
void EV_Regeneration(struct event_args_s* inArgs)
|
|
{
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
ASSERT(thePlayer);
|
|
|
|
float theVolume = inArgs->fparam1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, thePlayer->origin, CHAN_AUTO, kRegenerationSound, theVolume, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
|
|
void EV_StartCloak(struct event_args_s* inArgs)
|
|
{
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
ASSERT(thePlayer);
|
|
|
|
float theVolume = inArgs->fparam1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, thePlayer->origin, CHAN_AUTO, kStartCloakSound, theVolume, .4, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
|
|
void EV_EndCloak(struct event_args_s* inArgs)
|
|
{
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
ASSERT(thePlayer);
|
|
|
|
float theVolume = inArgs->fparam1;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, thePlayer->origin, CHAN_AUTO, kEndCloakSound, theVolume, .4, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
|
|
//void EV_WallJump(struct event_args_s* inArgs)
|
|
//{
|
|
// cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
// gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_VOICE, kWallJumpSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//}
|
|
|
|
void EV_Flight(struct event_args_s* inArgs)
|
|
{
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, thePlayer->origin, CHAN_AUTO, kEndCloakSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
//gEngfuncs.pEventAPI->EV_PlaySound( inArgs->entindex, thePlayer->origin, CHAN_AUTO, kStartCloakSound/*kWingFlapSound*/, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
}
|
|
|
|
void TeleportCallback( struct tempent_s *ent, float frametime, float currenttime)
|
|
{
|
|
}
|
|
|
|
void EV_Teleport(struct event_args_s* inArgs)
|
|
{
|
|
// Create a temp entity of this player and fade him out!
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
|
|
// // Kill earlier effect so the sound plays right?
|
|
// int idx = inArgs->entindex;
|
|
// gEngfuncs.pEventAPI->EV_KillEvents(idx, kTeleportEvent);
|
|
|
|
// TEMPENTITY* theTempEnt = gEngfuncs.pEfxAPI->CL_TentEntAllocCustom( (float*)inArgs->origin, thePlayer->model, 0, TeleportCallback);
|
|
// if(theTempEnt)
|
|
// {
|
|
// VectorCopy(inArgs->origin, theTempEnt->entity.origin);
|
|
// VectorCopy(inArgs->angles, theTempEnt->entity.angles);
|
|
//
|
|
// theTempEnt->flags |= FTENT_FADEOUT;
|
|
// theTempEnt->entity.baseline.renderamt = 255;
|
|
// theTempEnt->entity.curstate.renderfx = kRenderFxStrobeFaster;
|
|
// theTempEnt->fadeSpeed = 1.3f;
|
|
// theTempEnt->entity.curstate.movetype = MOVETYPE_NONE;
|
|
//
|
|
// theTempEnt->entity.curstate.modelindex = thePlayer->curstate.modelindex;
|
|
// theTempEnt->entity.curstate.frame = thePlayer->curstate.frame;
|
|
// theTempEnt->entity.curstate.colormap = thePlayer->curstate.colormap;
|
|
// theTempEnt->entity.curstate.skin = thePlayer->curstate.skin;
|
|
// theTempEnt->entity.curstate.framerate = thePlayer->curstate.framerate;
|
|
// theTempEnt->entity.curstate.body = thePlayer->curstate.body;
|
|
// theTempEnt->entity.curstate.weaponmodel = thePlayer->curstate.weaponmodel;
|
|
// theTempEnt->entity.curstate.sequence = thePlayer->curstate.sequence;
|
|
// theTempEnt->entity.curstate.gaitsequence = thePlayer->curstate.gaitsequence;
|
|
// theTempEnt->entity.curstate.usehull = thePlayer->curstate.usehull;
|
|
// theTempEnt->entity.curstate.playerclass = thePlayer->curstate.playerclass;
|
|
// theTempEnt->entity.curstate.animtime = thePlayer->curstate.animtime;
|
|
//
|
|
// memcpy(&theTempEnt->entity.curstate.controller[0], &thePlayer->curstate.controller[0], 4 * sizeof( byte ));
|
|
// memcpy(&theTempEnt->entity.curstate.blending[0], &thePlayer->curstate.blending[0], 2 * sizeof( byte ));
|
|
|
|
if(CVAR_GET_FLOAT(kvDynamicLights))
|
|
{
|
|
dlight_t* theDLight = gEngfuncs.pEfxAPI->CL_AllocDlight(1);
|
|
VectorCopy(inArgs->origin, theDLight->origin);
|
|
theDLight->radius = 200;
|
|
theDLight->color.r = 180;
|
|
theDLight->color.g = 180;
|
|
theDLight->color.b = 250;
|
|
|
|
// don't die for forseeable future
|
|
theDLight->die = gEngfuncs.GetClientTime() + .25f;
|
|
}
|
|
|
|
// Draw cool particle system
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsTeleport, inArgs->origin);
|
|
// }
|
|
}
|
|
|
|
void EV_PhaseIn(struct event_args_s* inArgs)
|
|
{
|
|
// Create a temp entity of this player and fade him out!
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kPhaseInSound, 1, ATTN_NORM, 0, thePitch);
|
|
|
|
if(CVAR_GET_FLOAT(kvDynamicLights))
|
|
{
|
|
dlight_t* theDLight = gEngfuncs.pEfxAPI->CL_AllocDlight(1);
|
|
VectorCopy(inArgs->origin, theDLight->origin);
|
|
theDLight->radius = 200;
|
|
theDLight->color.r = 180;
|
|
theDLight->color.g = 180;
|
|
theDLight->color.b = 250;
|
|
|
|
// don't die for forseeable future
|
|
theDLight->die = gEngfuncs.GetClientTime() + 1.0f;
|
|
}
|
|
|
|
// Draw cool particle system
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsPhaseIn, inArgs->origin);
|
|
}
|
|
|
|
void EV_SiegeHit(struct event_args_s* inArgs)
|
|
{
|
|
char* theSoundToPlay = kSiegeHitSound1;
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 1)
|
|
{
|
|
theSoundToPlay = kSiegeHitSound2;
|
|
}
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_BODY, theSoundToPlay, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
gEngfuncs.pEfxAPI->R_BlobExplosion(inArgs->origin);
|
|
}
|
|
|
|
void EV_SiegeViewHit(struct event_args_s* inArgs)
|
|
{
|
|
const float kPunchRange = 5.0f;
|
|
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat(-kPunchRange, kPunchRange));
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat(-kPunchRange, kPunchRange));
|
|
V_PunchAxis(2, gEngfuncs.pfnRandomFloat(-kPunchRange, kPunchRange));
|
|
}
|
|
|
|
void EV_CommandPoints(struct event_args_s* inArgs)
|
|
{
|
|
if(CVAR_GET_FLOAT(kvDynamicLights))
|
|
{
|
|
dlight_t* theDLight = gEngfuncs.pEfxAPI->CL_AllocDlight(1);
|
|
VectorCopy(inArgs->origin, theDLight->origin);
|
|
theDLight->radius = 60;
|
|
theDLight->color.r = 100;
|
|
theDLight->color.g = 250;
|
|
theDLight->color.b = 100;
|
|
|
|
// don't die for forseeable future
|
|
theDLight->die = gEngfuncs.GetClientTime() + 1.0f;
|
|
}
|
|
}
|
|
|
|
void EV_AlienSightOn(struct event_args_s* inArgs)
|
|
{
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kAlienSightOnSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
}
|
|
|
|
void EV_AlienSightOff(struct event_args_s* inArgs)
|
|
{
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kAlienSightOffSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
}
|
|
|
|
void EV_ParasiteGun(struct event_args_s* inArgs)
|
|
{
|
|
// Sharp falloff, so it's only heard at very close and by the firer
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kParasiteFireSound, .4f*inArgs->fparam1, .2, 0, thePitch);
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void EV_BlinkSuccess(struct event_args_s* inArgs)
|
|
{
|
|
// Create particle at origin, but also at angles (interpret as dest)
|
|
//AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsBlinkEffect, inArgs->origin);
|
|
//AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsBlinkEffect, inArgs->angles);
|
|
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kBlinkSuccessSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
//gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->angles, CHAN_WEAPON, kBlinkSuccessSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
}
|
|
|
|
void EV_DivineWind(struct event_args_s* inArgs)
|
|
{
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
|
|
float theSilenceVolumeFactor = AvHPlayerUpgrade::GetSilenceVolumeLevel((AvHUser3)thePlayer->curstate.iuser3, thePlayer->curstate.iuser4);
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kDivineWindFireSound, theSilenceVolumeFactor, ATTN_IDLE, 0, thePitch);
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void BileBombHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
{
|
|
gEngfuncs.pEventAPI->EV_PlaySound(ent->entity.index, ptr->endpos, CHAN_AUTO, kBileBombHitSound, 1, ATTN_IDLE, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsBilebomb, ptr->endpos, &(ptr->plane.normal));
|
|
}
|
|
|
|
void EV_BileBomb(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = inArgs->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kBileBombFireSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
|
|
vec3_t theBombAngles;
|
|
VectorAngles(forward, theBombAngles);
|
|
|
|
theBombAngles.y -= 90;
|
|
|
|
// Fire bomb
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->R_TempModel(inArgs->origin, forward, theBombAngles, 100, gEngfuncs.pEventAPI->EV_FindModelIndex(kBileBombProjectileModel), 0);
|
|
if(theTempEntity)
|
|
{
|
|
theTempEntity->hitcallback = BileBombHit;
|
|
theTempEntity->flags = (FTENT_GRAVITY | FTENT_COLLIDEALL | FTENT_COLLIDEKILL | FTENT_PERSIST);
|
|
theTempEntity->die += 5;
|
|
theTempEntity->entity.curstate.framerate = 30;
|
|
//theTempEntity->frameMax = 4;//theModel->numframes;
|
|
theTempEntity->clientIndex = inArgs->entindex; // Entity to ignore collisions with
|
|
|
|
Vector theAiming = forward;// + up;
|
|
VectorNormalize(theAiming);
|
|
|
|
Vector theStartVelocity;
|
|
VectorMA(Vector(0, 0, 0), kBileBombVelocity, theAiming, theStartVelocity);
|
|
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
}
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
float theHalfSpread = kBileBombPunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
}
|
|
|
|
void AcidRocketHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
{
|
|
char* theSoundToPlay = kAcidRocketHitSound;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(ent->entity.index, ptr->endpos, CHAN_AUTO, theSoundToPlay, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsAcidHitEffect, ptr->endpos, &(ptr->plane.normal));
|
|
}
|
|
|
|
void EV_AcidRocket(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
// Figure out which weapon description to assocate this weapon with
|
|
int theWeaponIndex = inArgs->iparam1;
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
int thePitch = gEngfuncs.pfnRandomLong(75, 150);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_AUTO, kAcidRocketFireSound, inArgs->fparam1, ATTN_IDLE, 0, thePitch);
|
|
|
|
vec3_t theRocketAngles;
|
|
VectorAngles(forward, theRocketAngles);
|
|
|
|
vec3_t theGunPosition;
|
|
EV_GetGunPosition(inArgs, theGunPosition, inArgs->origin);
|
|
|
|
Vector theRocketOrigin;
|
|
VectorMA(theGunPosition, kAcidRocketBarrelLength, forward, theRocketOrigin);
|
|
|
|
// Projectile is rotated a bit
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->R_TempModel(theRocketOrigin, forward, theRocketAngles, 100, gEngfuncs.pEventAPI->EV_FindModelIndex(kAcidRocketProjectileModel), 0);
|
|
if(theTempEntity)
|
|
{
|
|
theTempEntity->hitcallback = AcidRocketHit;
|
|
theTempEntity->flags = (FTENT_COLLIDEALL | FTENT_COLLIDEKILL | FTENT_PERSIST);
|
|
theTempEntity->entity.curstate.framerate = 30;
|
|
theTempEntity->frameMax = 4;//theModel->numframes;
|
|
|
|
// Temp entities interpret baseline origin as velocity.
|
|
Vector theBaseVelocity;
|
|
VectorScale(inArgs->velocity, kAcidRocketParentVelocityScalar, theBaseVelocity);
|
|
|
|
Vector theStartVelocity;
|
|
VectorMA(theBaseVelocity, kAcidRocketVelocity, forward, theStartVelocity);
|
|
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
}
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
float theHalfSpread = kAcidRocketPunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void UpdateStomp(TEMPENTITY* inEntity, float frametime, float inCurrentTime)
|
|
{
|
|
float theTimeScalar = (inCurrentTime - inEntity->entity.curstate.fuser1)/(inEntity->entity.curstate.fuser2/inEntity->entity.curstate.fuser1);
|
|
theTimeScalar = max(min(theTimeScalar, 1.0f), 0.0f);
|
|
|
|
// Fade it out according to lifetime
|
|
int theAlpha = (1.0f - theTimeScalar)*kStompModelRenderAmount;
|
|
inEntity->entity.curstate.renderamt = theAlpha;
|
|
|
|
// Don't create at end of trail
|
|
if(theTimeScalar < .7f)
|
|
{
|
|
// Every once in a awhile, create a smoke puff
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsStompSmoke, inEntity->entity.origin);
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_Stomp(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
Vector theStartPos;
|
|
VectorCopy(inArgs->origin, theStartPos);
|
|
|
|
// Not sure why this is necessary, AvHStomp::GetStompOrigin() should take care of it
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer();
|
|
if(theLocalPlayer)
|
|
{
|
|
float theZOffset = theLocalPlayer->curstate.mins.z;
|
|
theStartPos.z += theZOffset;
|
|
}
|
|
}
|
|
|
|
// Initial velocity store in angles, to make sure event and server entity match up
|
|
Vector theStartVelocity;
|
|
VectorCopy(inArgs->angles, theStartVelocity);
|
|
|
|
char* theSoundToPlay = kStompFireSound;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, theStartPos, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
vec3_t up, right, forward, vecSrc;//, origin;
|
|
|
|
// VectorCopy( inArgs->origin, origin );
|
|
gEngfuncs.pfnAngleVectors(inArgs->angles, forward, right, up);
|
|
|
|
// Create stomp projectile that flies forward, going through objects and stunning anyone it touches
|
|
|
|
vec3_t theStompAngles;
|
|
VectorAngles(forward, theStompAngles);
|
|
|
|
// Always draw level
|
|
theStompAngles.x = 0;
|
|
|
|
// Projectile artwork is oriented down the wrong axis
|
|
theStompAngles.y -= 180;
|
|
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->R_TempModel(theStartPos, forward, theStompAngles, 100, gEngfuncs.pEventAPI->EV_FindModelIndex(kStompProjectileModel), 0);
|
|
if(theTempEntity)
|
|
{
|
|
// vec3_t theStartPos, theEndPos;
|
|
// EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
// VectorMA(vecSrc, kStompBarrelLength, forward, theStartPos);
|
|
|
|
VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
//theTempEntity->hitcallback = StompHit;
|
|
theTempEntity->flags = (FTENT_PERSIST | FTENT_CLIENTCUSTOM/*| FTENT_SMOKETRAIL | FTENT_FADEOUT*/);
|
|
theTempEntity->callback = UpdateStomp;
|
|
|
|
// Set time created and time to expire so we can blend it in UpdateStomp
|
|
theTempEntity->entity.curstate.fuser1 = gHUD.m_flTime;
|
|
theTempEntity->entity.curstate.fuser2 = gHUD.m_flTime + kStompProjectileLifetime; // estimated end time of animation
|
|
|
|
theTempEntity->die = gEngfuncs.GetClientTime() + kStompProjectileLifetime;
|
|
theTempEntity->bounceFactor = 0;
|
|
|
|
theTempEntity->entity.curstate.rendermode = kRenderTransAdd;
|
|
theTempEntity->entity.curstate.renderamt = kStompModelRenderAmount;
|
|
|
|
//theTempEntity->entity.curstate.framerate = 30;
|
|
//theTempEntity->frameMax = 4;//theModel->numframes;
|
|
|
|
// Temp entities interpret baseline origin as velocity.
|
|
|
|
// Zero out vertical component as it's a ground stomp
|
|
forward[2] = 0.0f;
|
|
|
|
// Send shockwave in direction player is looking, but always make it go kStompProjectilVelocity
|
|
VectorNormalize(forward);
|
|
|
|
Vector theStartVelocity;
|
|
VectorScale(forward, kStompProjectileVelocity, theStartVelocity);
|
|
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
|
|
// Set orientation
|
|
//VectorCopy(inArgs->angles, theTempEntity->entity.angles);
|
|
}
|
|
|
|
// Create stomp effect at Onos
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsStompEffect, theStartPos);
|
|
}
|
|
|
|
void EV_Devour(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
char* theSoundToPlay = kDevourFireSound;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void EV_Leap(struct event_args_s* inArgs)
|
|
{
|
|
//char* theSoundToPlay = kLeapSound;
|
|
|
|
//gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void EV_Charge(struct event_args_s* inArgs)
|
|
{
|
|
char* theSoundToPlay = kChargeSound;
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
//We're sending a leap event.
|
|
void EV_Ability(struct event_args_s* inArgs)
|
|
{
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
gHUD.SetAlienAbility((AvHMessageID)inArgs->iparam1);
|
|
}
|
|
|
|
|
|
|
|
//physent_t* GetEntity(int inPhysIndex)
|
|
//{
|
|
// return gEngfuncs.pEventAPI->EV_GetPhysent( inPhysIndex );
|
|
//}
|
|
|
|
|
|
void EV_Select(struct event_args_s* args)
|
|
{
|
|
gSelectionHelper.ProcessPendingSelections();
|
|
}
|
|
|
|
|
|
void EV_Bite(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 1);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
theSoundToPlay = kBiteSound;
|
|
break;
|
|
//case 1:
|
|
// theSoundToPlay = kBiteSound2;
|
|
// break;
|
|
}
|
|
|
|
|
|
const int kBasePitch = 80;
|
|
const int kVariablePitchRange = 0x0;
|
|
|
|
const int kBiteAdrenPitchFactor = 30;
|
|
|
|
int theRandomPitch = gEngfuncs.pfnRandomLong( 0, kVariablePitchRange );
|
|
int thePitch = kBasePitch + (inArgs->fparam1*kBiteAdrenPitchFactor) + theRandomPitch;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, thePitch);
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
float theHalfSpread = kBitePunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
PlayMeleeHitEffects(inArgs, BALANCE_VAR(kBiteRange), kBiteHitSound1);
|
|
}
|
|
else
|
|
{
|
|
PlayMeleeHitEffects(inArgs, BALANCE_VAR(kBiteRange), kBiteHitSound2);
|
|
}
|
|
}
|
|
|
|
|
|
void EV_Bite2(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 1);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
theSoundToPlay = kBite2Sound;
|
|
break;
|
|
//case 1:
|
|
// theSoundToPlay = kBiteSound2;
|
|
// break;
|
|
}
|
|
|
|
|
|
const int kBasePitch = 80;
|
|
const int kVariablePitchRange = 0x0;
|
|
|
|
const int kBiteAdrenPitchFactor = 30;
|
|
|
|
int theRandomPitch = gEngfuncs.pfnRandomLong( 0, kVariablePitchRange );
|
|
int thePitch = kBasePitch + (inArgs->fparam1*kBiteAdrenPitchFactor) + theRandomPitch;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, thePitch);
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
// gEngfuncs.pEventAPI->EV_WeaponAnimation(ANIM_FIRE1 + gEngfuncs.pfnRandomLong(0,2), 2);
|
|
|
|
float theHalfSpread = kBitePunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
if(gEngfuncs.pfnRandomLong(0, 1) == 0)
|
|
{
|
|
PlayMeleeHitEffects(inArgs, BALANCE_VAR(kBiteRange), kBiteHitSound1);
|
|
}
|
|
else
|
|
{
|
|
PlayMeleeHitEffects(inArgs, BALANCE_VAR(kBiteRange), kBiteHitSound2);
|
|
}
|
|
}
|
|
|
|
|
|
void EV_BuildGun(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 1);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
theSoundToPlay = kBuildingGunSound1;
|
|
break;
|
|
case 1:
|
|
theSoundToPlay = kBuildingGunSound2;
|
|
break;
|
|
}
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// General x-punch axis
|
|
if (EV_IsLocal(idx))
|
|
{
|
|
// gEngfuncs.pEventAPI->EV_WeaponAnimation(ANIM_FIRE1 + gEngfuncs.pfnRandomLong(0,2), 2);
|
|
|
|
float theHalfSpread = kBuildingGunPunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis(0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
V_PunchAxis(1, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_HealingSpray(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
// Play attack sound
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, kHealingSpraySound, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Load model
|
|
//int theModelIndex;
|
|
//struct model_s* theModel = gEngfuncs.CL_LoadModel(kWebProjectileSprite, &theModelIndex);
|
|
//if(theModel)
|
|
//{
|
|
// Create some spray effect
|
|
//gEngfuncs.pEfxAPI->R_Sprite_Spray(inArgs->origin, forward, theModelIndex, 15, 100, 2);
|
|
//vec3_t theEndPoint;
|
|
//VectorMA(inArgs->origin, kHealingSprayRange, forward, theEndPoint);
|
|
//gEngfuncs.pEfxAPI->R_BeamLightning(inArgs->origin, theEndPoint, theModelIndex, .5f, 10, 1.0f, 1.0f, 1.0f);
|
|
//}
|
|
|
|
|
|
vec3_t theStartPos;
|
|
EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
VectorMA(vecSrc, kHealingSprayBarrelLength, forward, theStartPos);
|
|
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsBacteriaSpray, theStartPos);
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void EV_Metabolize(struct event_args_s* inArgs)
|
|
{
|
|
char* theSoundToPlay = "";
|
|
|
|
int theRandomSound = gEngfuncs.pfnRandomLong(0, 2);
|
|
switch(theRandomSound)
|
|
{
|
|
case 0:
|
|
theSoundToPlay = kMetabolizeFireSound1;
|
|
break;
|
|
case 1:
|
|
theSoundToPlay = kMetabolizeFireSound2;
|
|
break;
|
|
case 2:
|
|
theSoundToPlay = kMetabolizeFireSound3;
|
|
break;
|
|
}
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, theSoundToPlay, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, kMetabolizeFireSound, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void EV_MetabolizeSuccess(struct event_args_s* inArgs)
|
|
{
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, kMetabolizeSuccessSound, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void WebHit(struct tempent_s* ent, struct pmtrace_s* ptr)
|
|
{
|
|
gEngfuncs.pEventAPI->EV_PlaySound(ent->entity.index, ptr->endpos, CHAN_AUTO, kWebStrandHitSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//gEngfuncs.pEventAPI->EV_PlaySound(ent->entity.index, ptr->endpos, CHAN_AUTO, kWebSpinSound1, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
if(ptr && ent)
|
|
{
|
|
ent->die = gEngfuncs.GetClientTime();
|
|
|
|
//AvHParticleSystemManager::Instance()->CreateParticleSystem(kpsSpitHit, ptr->endpos, &(ptr->plane.normal));
|
|
|
|
// Create splat here too
|
|
CreateDecal(ptr);
|
|
}
|
|
}
|
|
|
|
|
|
void EV_SpinWeb(struct event_args_s* inArgs)
|
|
{
|
|
ASSERT(inArgs);
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
if( EV_IsLocal( idx ) )
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
|
|
// Play attack sound
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, kWebSpinSound1, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Fire spit glob
|
|
int theModelIndex;
|
|
struct model_s* theModel = gEngfuncs.CL_LoadModel(kWebProjectileSprite, &theModelIndex);
|
|
if(theModel)
|
|
{
|
|
TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(gPredictedPlayerOrigin, theModel);
|
|
if(theTempEntity)
|
|
{
|
|
vec3_t theStartPos, theEndPos;
|
|
EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
VectorMA(vecSrc, kSpitGBarrelLength, forward, theStartPos);
|
|
|
|
VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
//theTempEntity->die += kSpitLifetime;
|
|
theTempEntity->hitcallback = WebHit;
|
|
theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP/* | FTENT_PERSIST*/);
|
|
theTempEntity->clientIndex = inArgs->entindex; // Entity to ignore collisions with
|
|
theTempEntity->entity.curstate.framerate = 30;
|
|
theTempEntity->frameMax = 4;//theModel->numframes;
|
|
|
|
// Temp entities interpret baseline origin as velocity.
|
|
Vector theBaseVelocity;
|
|
VectorScale(inArgs->velocity, kWebProjectileParentVelocityScalar, theBaseVelocity);
|
|
|
|
Vector theStartVelocity;
|
|
VectorMA(theBaseVelocity, kWebProjectileVelocity, forward, theStartVelocity);
|
|
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
}
|
|
}
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
float theHalfSpread = kWebGXPunch/2.0f;
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
//void EV_Babbler(struct event_args_s* inArgs)
|
|
//{
|
|
// ASSERT(inArgs);
|
|
//
|
|
// // Play attack animation and add muzzle flash
|
|
// int idx = inArgs->entindex;
|
|
//
|
|
// // Play attack sound
|
|
// gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_WEAPON, kBabblerGunSound, inArgs->fparam1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
//
|
|
// if ( EV_IsLocal( idx ) )
|
|
// {
|
|
// float theHalfSpread = kBabblerXPunch/2.0f;
|
|
// if(theHalfSpread > 0.0f)
|
|
// {
|
|
// V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
// }
|
|
//
|
|
// gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
// }
|
|
//}
|
|
|
|
|
|
|
|
void EV_PrimalScream(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
// Play attack sound
|
|
|
|
// You can hear this waaay far off
|
|
float theFalloffFactor = .5f;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, inArgs->origin, CHAN_VOICE, kPrimalScreamSound, 1.0f, theFalloffFactor, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
if (EV_IsLocal(inArgs->entindex))
|
|
{
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
void EV_StopScream(struct event_args_s* inArgs)
|
|
{
|
|
int idx = inArgs->entindex;
|
|
|
|
gEngfuncs.pEventAPI->EV_StopSound(idx, CHAN_VOICE, kPrimalScreamSound);
|
|
}
|
|
|
|
|
|
void EV_SpikeGun(struct event_args_s* inArgs)
|
|
{
|
|
//static int theTracerCount[ 32 ];
|
|
|
|
// Play attack animation and add muzzle flash
|
|
int idx;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
vec3_t velocity;
|
|
|
|
vec3_t vecSrc, vecAiming;
|
|
vec3_t up, right, forward;
|
|
|
|
idx = inArgs->entindex;
|
|
VectorCopy( inArgs->origin, origin );
|
|
VectorCopy( inArgs->angles, angles );
|
|
VectorCopy( inArgs->velocity, velocity );
|
|
|
|
//AngleVectors( angles, forward, right, up );
|
|
gEngfuncs.pfnAngleVectors(angles, forward, right, up);
|
|
|
|
// if ( EV_IsLocal( idx ) )
|
|
// {
|
|
// gEngfuncs.pEventAPI->EV_WeaponAnimation(5 + + gEngfuncs.pfnRandomLong(0,2), 2);
|
|
// }
|
|
|
|
// Play one of basic attack sounds
|
|
float theVolume = inArgs->fparam1;
|
|
float theAttenuation = 1.2f;
|
|
|
|
char* theSoundToPlay = kSpikeFireSound;
|
|
|
|
int theUpperBound = 30;
|
|
int thePitch = 100 + (gEngfuncs.pfnRandomLong(0, theUpperBound) - theUpperBound/2);
|
|
gEngfuncs.pEventAPI->EV_PlaySound(idx, origin, CHAN_WEAPON, theSoundToPlay, theVolume, theAttenuation, 0, thePitch);
|
|
|
|
EV_GetGunPosition( inArgs, vecSrc, origin );
|
|
VectorCopy( forward, vecAiming );
|
|
|
|
// Create ricochet and spike hit decals
|
|
EV_HLDM_FireBulletsPlayer( idx, forward, right, up, 1, vecSrc, vecAiming, kSpikeRange, BULLET_MONSTER_9MM, 0, NULL, kSpikeSpread, inArgs->iparam1);
|
|
|
|
// // Create two spike projectiles, flying forward until they hit something
|
|
// //for(int i = 0; i < 2; i++)
|
|
// //{
|
|
// vec3_t theEntStartPos = gPredictedPlayerOrigin;
|
|
//
|
|
// //TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->CL_TempEntAlloc(theEntStartPos, theModel);
|
|
// vec3_t theSpikeAngles;
|
|
// VectorAngles(forward, theSpikeAngles);
|
|
//
|
|
// TEMPENTITY* theTempEntity = gEngfuncs.pEfxAPI->R_TempModel(gPredictedPlayerOrigin, forward, theSpikeAngles, 100, gEngfuncs.pEventAPI->EV_FindModelIndex(kSpikeProjectileModel), 0);
|
|
// if(theTempEntity)
|
|
// {
|
|
// vec3_t theStartPos, theEndPos;
|
|
// EV_GetGunPosition(inArgs, vecSrc, inArgs->origin);
|
|
// VectorMA(vecSrc, kSpikeBarrelLength, forward, theStartPos);
|
|
//
|
|
// VectorCopy(theStartPos, theTempEntity->entity.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.prevstate.origin);
|
|
// VectorCopy(theStartPos, theTempEntity->entity.curstate.origin);
|
|
// theTempEntity->hitcallback = SpikeHit;
|
|
// theTempEntity->flags |= (FTENT_COLLIDEALL | FTENT_COLLIDEKILL/* | FTENT_PERSIST*/);
|
|
// //theTempEntity->entity.curstate.framerate = 30;
|
|
// //theTempEntity->frameMax = 4;//theModel->numframes;
|
|
//
|
|
// // Temp entities interpret baseline origin as velocity.
|
|
// Vector theStartVelocity;
|
|
// VectorScale(forward, kSpikeVelocity, theStartVelocity);
|
|
//
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.origin);
|
|
// VectorCopy(theStartVelocity, theTempEntity->entity.baseline.velocity);
|
|
//
|
|
// // Set orientation
|
|
// //VectorCopy(inArgs->angles, theTempEntity->entity.angles);
|
|
// }
|
|
// //}
|
|
|
|
////EV_HLDM_FireBullets( idx, forward, right, up, 1, vecSrc, vecAiming, kMGRange, BULLET_PLAYER_MP5, theTracerFreq, &tracerCount[idx-1] );
|
|
//EV_HLDM_FireBullets( idx, forward, right, up, 1, vecSrc, vecAiming, kSpikeRange, BULLET_MONSTER_9MM, 0, NULL /* &theTracerCount[0]*/, 0.0f, 0.0f);
|
|
|
|
// General x-punch axis
|
|
if ( EV_IsLocal( idx ) )
|
|
{
|
|
// Multiply punch by upgrade level
|
|
float theHalfSpread = (kSpikePunch/2.0f);
|
|
if(theHalfSpread > 0.0f)
|
|
{
|
|
V_PunchAxis( 0, gEngfuncs.pfnRandomFloat( -theHalfSpread, theHalfSpread ) );
|
|
}
|
|
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(inArgs->iparam2, 2);
|
|
}
|
|
}
|
|
|
|
//void EV_HiveHit(struct event_args_s* inArgs)
|
|
//{
|
|
// // when a hive comes under attack, let all the minions know it
|
|
//
|
|
// // iparam1 is the hurt sound to play
|
|
//
|
|
//
|
|
//
|
|
// const float kPunchRange = 3.0f;
|
|
//
|
|
// V_PunchAxis(0, gEngfuncs.pfnRandomFloat(-kPunchRange, kPunchRange));
|
|
// V_PunchAxis(1, gEngfuncs.pfnRandomFloat(-kPunchRange, kPunchRange));
|
|
// V_PunchAxis(2, gEngfuncs.pfnRandomFloat(-kPunchRange, kPunchRange));
|
|
//}
|
|
|
|
void EV_EmptySound(struct event_args_s* inArgs)
|
|
{
|
|
int thePitch = 100;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, kEmptySound, .8f, ATTN_NORM, 0, thePitch);
|
|
}
|
|
|
|
void EV_NumericalInfo(struct event_args_s* inArgs)
|
|
{
|
|
// Can't send easily send events to one team only, so only draw them for the right team
|
|
cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer();
|
|
if(theLocalPlayer)
|
|
{
|
|
int theTeamNumber = inArgs->iparam2;
|
|
if((theLocalPlayer->curstate.team == theTeamNumber) || (theTeamNumber == 0))
|
|
{
|
|
float theNumber = inArgs->fparam1;
|
|
int theEventType = inArgs->iparam1;
|
|
gHUD.AddNumericalInfoMessage(inArgs->origin, theNumber, theEventType);
|
|
}
|
|
}
|
|
}
|
|
|
|
void EV_InvalidAction(struct event_args_s* inArgs)
|
|
{
|
|
int thePitch = 100;
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_WEAPON, kInvalidSound, .8f, ATTN_NORM, 0, thePitch);
|
|
}
|
|
|
|
void EV_Particle(struct event_args_s* inArgs)
|
|
{
|
|
// Read template
|
|
uint32 theTemplateIndex = inArgs->iparam1;
|
|
|
|
// Lookup template
|
|
const AvHParticleTemplate* theTemplate = gParticleTemplateList.GetTemplateAtIndex(theTemplateIndex);
|
|
if(theTemplate)
|
|
{
|
|
string theTemplateName = theTemplate->GetName();
|
|
AvHParticleSystemManager::Instance()->CreateParticleSystem(theTemplateName, inArgs->origin);
|
|
}
|
|
}
|
|
|
|
void EV_DistressBeacon(struct event_args_s* inArgs)
|
|
{
|
|
// Play distress beacon sound
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_BODY, kDistressBeaconSound, 1, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Read time before distress beacon will end
|
|
int theDuration = inArgs->iparam1;
|
|
|
|
// Create blinking light that will die in that time
|
|
if(CVAR_GET_FLOAT(kvDynamicLights))
|
|
{
|
|
dlight_t* theLight = gEngfuncs.pEfxAPI->CL_AllocDlight(21);
|
|
VectorCopy(inArgs->origin, theLight->origin);
|
|
theLight->radius = 250;
|
|
theLight->color.r = 250;
|
|
theLight->color.g = 180;
|
|
theLight->color.b = 180;
|
|
|
|
// don't die for forseeable future
|
|
theLight->die = gEngfuncs.GetClientTime() + theDuration;
|
|
}
|
|
}
|
|
|
|
void EV_WeaponAnimation(struct event_args_s* inArgs)
|
|
{
|
|
// General x-punch axis
|
|
if(EV_IsLocal(inArgs->entindex))
|
|
{
|
|
int theAnimation = max(inArgs->iparam2, 0);
|
|
gEngfuncs.pEventAPI->EV_WeaponAnimation(theAnimation, 2);
|
|
}
|
|
}
|
|
|
|
|
|
void EV_LevelUp(struct event_args_s* inArgs)
|
|
{
|
|
// General x-punch axis
|
|
if(EV_IsLocal(inArgs->entindex))
|
|
{
|
|
}
|
|
|
|
|
|
cl_entity_t* thePlayer = GetEntity(inArgs->entindex);
|
|
// Play sound
|
|
int theIsMarine = inArgs->iparam1;
|
|
const char* theSound = theIsMarine ? kLevelUpMarineSound : kLevelUpAlienSound;
|
|
|
|
float theSilenceVolumeFactor = theIsMarine ? 1.0 : AvHPlayerUpgrade::GetSilenceVolumeLevel((AvHUser3)thePlayer->curstate.iuser3, thePlayer->curstate.iuser4);
|
|
|
|
gEngfuncs.pEventAPI->EV_PlaySound(inArgs->entindex, inArgs->origin, CHAN_VOICE, theSound, theSilenceVolumeFactor, ATTN_NORM, 0, 94 + gEngfuncs.pfnRandomLong( 0, 0xf ));
|
|
|
|
// Play nice particle effect?
|
|
}
|