//========= Copyright Valve Corporation, All rights reserved. ============// // // Purpose: TF Sniper Rifle // //=============================================================================// #include "cbase.h" #include "tf_fx_shared.h" #include "tf_weapon_sniperrifle.h" #include "in_buttons.h" #include "tf_gamerules.h" // Client specific. #ifdef CLIENT_DLL #include "view.h" #include "beamdraw.h" #include "vgui/ISurface.h" #include <vgui/ILocalize.h> #include "vgui_controls/Controls.h" #include "hud_crosshair.h" #include "functionproxy.h" #include "materialsystem/imaterialvar.h" #include "toolframework_client.h" #include "input.h" #include "client_virtualreality.h" #include "sourcevr/isourcevirtualreality.h" // forward declarations void ToolFramework_RecordMaterialParams( IMaterial *pMaterial ); #else #include "tf_gamerules.h" #include "tf_fx.h" #endif #define TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC 50.0 #define TF_WEAPON_SNIPERRIFLE_UNCHARGE_PER_SEC 75.0 #define TF_WEAPON_SNIPERRIFLE_DAMAGE_MIN 50 #define TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX 150 #define TF_WEAPON_SNIPERRIFLE_RELOAD_TIME 1.5f #define TF_WEAPON_SNIPERRIFLE_ZOOM_TIME 0.3f #define TF_WEAPON_SNIPERRIFLE_NO_CRIT_AFTER_ZOOM_TIME 0.2f #define SNIPER_DOT_SPRITE_RED "effects/sniperdot_red.vmt" #define SNIPER_DOT_SPRITE_BLUE "effects/sniperdot_blue.vmt" #define SNIPER_CHARGE_BEAM_RED "tfc_sniper_charge_red" #define SNIPER_CHARGE_BEAM_BLUE "tfc_sniper_charge_blue" #ifdef CLIENT_DLL ConVar tf_sniper_fullcharge_bell( "tf_sniper_fullcharge_bell", "0", FCVAR_ARCHIVE ); #endif //============================================================================= // // Weapon Sniper Rifles tables. // IMPLEMENT_NETWORKCLASS_ALIASED( TFSniperRifle, DT_TFSniperRifle ) BEGIN_NETWORK_TABLE_NOBASE( CTFSniperRifle, DT_SniperRifleLocalData ) #if !defined( CLIENT_DLL ) SendPropFloat( SENDINFO(m_flChargedDamage), 0, SPROP_NOSCALE | SPROP_CHANGES_OFTEN ), #else RecvPropFloat( RECVINFO(m_flChargedDamage) ), #endif END_NETWORK_TABLE() BEGIN_NETWORK_TABLE( CTFSniperRifle, DT_TFSniperRifle ) #if !defined( CLIENT_DLL ) SendPropDataTable( "SniperRifleLocalData", 0, &REFERENCE_SEND_TABLE( DT_SniperRifleLocalData ), SendProxy_SendLocalWeaponDataTable ), #else RecvPropDataTable( "SniperRifleLocalData", 0, 0, &REFERENCE_RECV_TABLE( DT_SniperRifleLocalData ) ), #endif END_NETWORK_TABLE() BEGIN_PREDICTION_DATA( CTFSniperRifle ) #ifdef CLIENT_DLL DEFINE_PRED_FIELD( m_flUnzoomTime, FIELD_FLOAT, 0 ), DEFINE_PRED_FIELD( m_flRezoomTime, FIELD_FLOAT, 0 ), DEFINE_PRED_FIELD( m_bRezoomAfterShot, FIELD_BOOLEAN, 0 ), DEFINE_PRED_FIELD( m_flChargedDamage, FIELD_FLOAT, 0 ), #endif END_PREDICTION_DATA() LINK_ENTITY_TO_CLASS( tf_weapon_sniperrifle, CTFSniperRifle ); PRECACHE_WEAPON_REGISTER( tf_weapon_sniperrifle ); BEGIN_DATADESC( CTFSniperRifle ) DEFINE_FIELD( m_flUnzoomTime, FIELD_FLOAT ), DEFINE_FIELD( m_flRezoomTime, FIELD_FLOAT ), DEFINE_FIELD( m_bRezoomAfterShot, FIELD_BOOLEAN ), DEFINE_FIELD( m_flChargedDamage, FIELD_FLOAT ), END_DATADESC() //============================================================================= IMPLEMENT_NETWORKCLASS_ALIASED( TFSniperRifleDecap, DT_TFSniperRifleDecap ) BEGIN_NETWORK_TABLE( CTFSniperRifleDecap, DT_TFSniperRifleDecap ) END_NETWORK_TABLE() BEGIN_PREDICTION_DATA( CTFSniperRifleDecap ) END_PREDICTION_DATA() LINK_ENTITY_TO_CLASS( tf_weapon_sniperrifle_decap, CTFSniperRifleDecap ); PRECACHE_WEAPON_REGISTER( tf_weapon_sniperrifle_decap ); //============================================================================= IMPLEMENT_NETWORKCLASS_ALIASED( TFSniperRifleClassic, DT_TFSniperRifleClassic ) BEGIN_NETWORK_TABLE( CTFSniperRifleClassic, DT_TFSniperRifleClassic ) #if !defined( CLIENT_DLL ) SendPropBool( SENDINFO(m_bCharging) ), #else RecvPropBool( RECVINFO(m_bCharging) ), #endif END_NETWORK_TABLE() BEGIN_PREDICTION_DATA( CTFSniperRifleClassic ) #ifdef CLIENT_DLL DEFINE_PRED_FIELD( m_bCharging, FIELD_BOOLEAN, 0 ), #endif END_PREDICTION_DATA() LINK_ENTITY_TO_CLASS( tf_weapon_sniperrifle_classic, CTFSniperRifleClassic ); PRECACHE_WEAPON_REGISTER( tf_weapon_sniperrifle_classic ); //============================================================================= #ifdef STAGING_ONLY IMPLEMENT_NETWORKCLASS_ALIASED( TFSniperRifleRevolver, DT_TFSniperRifleRevolver ) BEGIN_NETWORK_TABLE( CTFSniperRifleRevolver, DT_TFSniperRifleRevolver ) END_NETWORK_TABLE() BEGIN_PREDICTION_DATA( CTFSniperRifleRevolver ) END_PREDICTION_DATA() LINK_ENTITY_TO_CLASS( tf_weapon_sniperrifle_revolver, CTFSniperRifleRevolver ); PRECACHE_WEAPON_REGISTER( tf_weapon_sniperrifle_revolver ); #endif // STAGING_ONLY //============================================================================= // // Weapon Sniper Rifles functions. // //----------------------------------------------------------------------------- // Purpose: Constructor. //----------------------------------------------------------------------------- CTFSniperRifle::CTFSniperRifle() { // Server specific. #ifdef GAME_DLL m_hSniperDot = NULL; #else m_bPlayedBell = false; #endif m_bCurrentShotIsHeadshot = false; m_flChargedDamage = 0.0f; m_flChargePerSec = TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC; m_bWasAimedAtEnemy = false; } //----------------------------------------------------------------------------- // Purpose: Destructor. //----------------------------------------------------------------------------- CTFSniperRifle::~CTFSniperRifle() { // Server specific. #ifdef GAME_DLL DestroySniperDot(); #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::Spawn() { m_iAltFireHint = HINT_ALTFIRE_SNIPERRIFLE; BaseClass::Spawn(); ResetTimers(); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::Precache() { BaseClass::Precache(); PrecacheModel( SNIPER_DOT_SPRITE_RED ); PrecacheModel( SNIPER_DOT_SPRITE_BLUE ); PrecacheScriptSound( "doomsday.warhead" ); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ResetTimers( void ) { m_flUnzoomTime = -1; m_flRezoomTime = -1; m_bRezoomAfterShot = false; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::Reload( void ) { // We currently don't reload. return true; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::CanHolster( void ) const { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer ) { // TF_COND_MELEE_ONLY need to be able to immediately holster and switch to melee weapon if ( pPlayer->m_Shared.InCond( TF_COND_MELEE_ONLY ) ) return true; // don't allow us to holster this weapon if we're in the process of zooming and // we've just fired the weapon (next primary attack is only 1.5 seconds after firing) if ( ( pPlayer->GetFOV() < pPlayer->GetDefaultFOV() ) && ( m_flNextPrimaryAttack > gpGlobals->curtime ) ) return false; } return BaseClass::CanHolster(); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::Holster( CBaseCombatWeapon *pSwitchingTo ) { // Server specific. #ifdef GAME_DLL // Destroy the sniper dot. DestroySniperDot(); #endif CTFPlayer *pPlayer = ToTFPlayer( GetPlayerOwner() ); if ( pPlayer && pPlayer->m_Shared.InCond( TF_COND_ZOOMED ) ) { ZoomOut(); } m_flChargedDamage = 0.0f; #ifdef CLIENT_DLL m_bPlayedBell = false; #endif ResetTimers(); return BaseClass::Holster( pSwitchingTo ); } void CTFSniperRifle::WeaponReset( void ) { BaseClass::WeaponReset(); ZoomOut(); m_bCurrentShotIsHeadshot = false; m_flChargePerSec = TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::OwnerCanJump( void ) { return gpGlobals->curtime > m_flUnzoomTime; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::HandleZooms( void ) { // Get the owning player. CTFPlayer *pPlayer = ToTFPlayer( GetOwner() ); if ( !pPlayer ) return; // Handle the zoom when taunting. if ( pPlayer->m_Shared.InCond( TF_COND_TAUNTING ) ) { if ( pPlayer->m_Shared.InCond( TF_COND_AIMING ) ) { ToggleZoom(); } //Don't rezoom in the middle of a taunt. ResetTimers(); } if ( m_flUnzoomTime > 0 && gpGlobals->curtime > m_flUnzoomTime ) { if ( m_bRezoomAfterShot ) { ZoomOutIn(); m_bRezoomAfterShot = false; } else { ZoomOut(); } m_flUnzoomTime = -1; } if ( m_flRezoomTime > 0 ) { if ( gpGlobals->curtime > m_flRezoomTime ) { ZoomIn(); m_flRezoomTime = -1; } } if ( ( pPlayer->m_nButtons & IN_ATTACK2 ) && ( m_flNextSecondaryAttack <= gpGlobals->curtime ) ) { // If we're in the process of rezooming, just cancel it if ( m_flRezoomTime > 0 || m_flUnzoomTime > 0 ) { // Prevent them from rezooming in less time than they would have m_flNextSecondaryAttack = m_flRezoomTime + TF_WEAPON_SNIPERRIFLE_ZOOM_TIME; m_flRezoomTime = -1; } else { Zoom(); } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ItemPostFrame( void ) { // If we're lowered, we're not allowed to fire if ( m_bLowered ) return; // Get the owning player. CTFPlayer *pPlayer = ToTFPlayer( GetOwner() ); if ( !pPlayer ) return; if ( !CanAttack() ) { if ( IsZoomed() ) { ToggleZoom(); } return; } HandleZooms(); #ifdef GAME_DLL // Update the sniper dot position if we have one if ( m_hSniperDot ) { UpdateSniperDot(); } #endif // Start charging when we're zoomed in, and allowed to fire if ( pPlayer->m_Shared.IsJumping() ) { // Unzoom if we're jumping if ( IsZoomed() ) { ToggleZoom(); } m_flChargedDamage = 0.0f; m_bRezoomAfterShot = false; m_flRezoomTime = -1.f; } else if ( m_flNextSecondaryAttack <= gpGlobals->curtime ) { // Don't start charging in the time just after a shot before we unzoom to play rack anim. if ( pPlayer->m_Shared.InCond( TF_COND_AIMING ) && !m_bRezoomAfterShot ) { float fSniperRifleChargePerSec = m_flChargePerSec; ApplyChargeSpeedModifications( fSniperRifleChargePerSec ); fSniperRifleChargePerSec += SniperRifleChargeRateMod(); // we don't want sniper charge rate to go too high. fSniperRifleChargePerSec = clamp( fSniperRifleChargePerSec, 0, 2.f * TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC ); m_flChargedDamage = MIN( m_flChargedDamage + gpGlobals->frametime * fSniperRifleChargePerSec, TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX ); #ifdef CLIENT_DLL // play the recharged bell if we're fully charged if ( IsFullyCharged() && !m_bPlayedBell ) { m_bPlayedBell = true; if ( tf_sniper_fullcharge_bell.GetBool() ) { C_TFPlayer::GetLocalTFPlayer()->EmitSound( "TFPlayer.ReCharged" ); } } #endif } else { m_flChargedDamage = MAX( 0, m_flChargedDamage - gpGlobals->frametime * TF_WEAPON_SNIPERRIFLE_UNCHARGE_PER_SEC ); } } // Fire. if ( pPlayer->m_nButtons & IN_ATTACK ) { Fire( pPlayer ); } // Idle. if ( !( ( pPlayer->m_nButtons & IN_ATTACK) || ( pPlayer->m_nButtons & IN_ATTACK2 ) ) ) { // No fire buttons down or reloading if ( !ReloadOrSwitchWeapons() && ( m_bInReload == false ) ) { WeaponIdle(); } } // Sniper Rage (Hitman's heatmaker) // Activate on 'R' // no longer need full charge if ( (pPlayer->m_nButtons & IN_RELOAD ) && pPlayer->m_Shared.GetRageMeter() > 1.0f ) { int iBuffType = 0; CALL_ATTRIB_HOOK_INT( iBuffType, set_buff_type ); if ( iBuffType > 0 ) { pPlayer->m_Shared.ActivateRageBuff( pPlayer, iBuffType ); } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::PlayWeaponShootSound( void ) { if ( TFGameRules()->GameModeUsesUpgrades() ) { PlayUpgradedShootSound( "Weapon_Upgrade.DamageBonus" ); } if ( !IsFullyCharged() ) { float flDamageBonus = 1.0f; CALL_ATTRIB_HOOK_FLOAT( flDamageBonus, sniper_full_charge_damage_bonus ); if ( flDamageBonus > 1.0f ) { WeaponSound( SPECIAL3 ); return; } } BaseClass::PlayWeaponShootSound(); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::Lower( void ) { if ( BaseClass::Lower() ) { if ( IsZoomed() ) { ToggleZoom(); } return true; } return false; } //----------------------------------------------------------------------------- // Purpose: Secondary attack. //----------------------------------------------------------------------------- void CTFSniperRifle::Zoom( void ) { // Don't allow the player to zoom in while jumping CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer && pPlayer->m_Shared.IsJumping() ) { if ( pPlayer->GetFOV() >= 75 ) return; } ToggleZoom(); // at least 0.1 seconds from now, but don't stomp a previous value m_flNextPrimaryAttack = MAX( m_flNextPrimaryAttack, gpGlobals->curtime + 0.1 ); m_flNextSecondaryAttack = gpGlobals->curtime + TF_WEAPON_SNIPERRIFLE_ZOOM_TIME; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ZoomOutIn( void ) { ZoomOut(); CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer && pPlayer->ShouldAutoRezoom() ) { float flRezoomDelay = 0.9f; if ( !UsesClipsForAmmo1() ) { // Since sniper rifles don't actually use clips the fast reload hook also affects unzoom and zoom delays ApplyScopeSpeedModifications( flRezoomDelay ); } m_flRezoomTime = gpGlobals->curtime + flRezoomDelay; } else { m_flNextSecondaryAttack = gpGlobals->curtime + 1.0f; } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ZoomIn( void ) { // Start aiming. CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) return; if ( pPlayer->GetAmmoCount( m_iPrimaryAmmoType ) <= 0 ) return; BaseClass::ZoomIn(); pPlayer->m_Shared.AddCond( TF_COND_AIMING ); pPlayer->TeamFortress_SetSpeed(); #ifdef GAME_DLL // Create the sniper dot. CreateSniperDot(); pPlayer->ClearExpression(); #endif } //----------------------------------------------------------------------------- bool CTFSniperRifle::IsZoomed( void ) { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer ) { return pPlayer->m_Shared.InCond( TF_COND_ZOOMED ); } return false; } //----------------------------------------------------------------------------- // // Have we been zoomed in long enough for our shot to do max damage // bool CTFSniperRifle::IsFullyCharged( void ) const { return m_flChargedDamage >= TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ZoomOut( void ) { BaseClass::ZoomOut(); // Stop aiming CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) return; pPlayer->m_Shared.RemoveCond( TF_COND_AIMING ); pPlayer->TeamFortress_SetSpeed(); #ifdef GAME_DLL // Destroy the sniper dot. DestroySniperDot(); pPlayer->ClearExpression(); #endif // if we are thinking about zooming, cancel it m_flUnzoomTime = -1; m_flRezoomTime = -1; m_bRezoomAfterShot = false; m_flChargedDamage = 0.0f; #ifdef CLIENT_DLL m_bPlayedBell = false; #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ApplyScopeSpeedModifications( float &flBaseRef ) { CALL_ATTRIB_HOOK_FLOAT( flBaseRef, fast_reload ); // Prototype hack CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer ) { if ( pPlayer->m_Shared.GetCarryingRuneType() == RUNE_HASTE || pPlayer->m_Shared.GetCarryingRuneType() == RUNE_PRECISION ) { flBaseRef *= 0.5f; } else if ( pPlayer->m_Shared.GetCarryingRuneType() == RUNE_KING || pPlayer->m_Shared.InCond( TF_COND_KING_BUFFED ) ) { flBaseRef *= 0.75f; } int iMaster = 0; CALL_ATTRIB_HOOK_INT_ON_OTHER( pPlayer, iMaster, ability_master_sniper ); if ( iMaster ) { flBaseRef *= RemapValClamped( iMaster, 1, 2, 0.6f, 0.3f ); } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ApplyChargeSpeedModifications( float &flBaseRef ) { CALL_ATTRIB_HOOK_FLOAT( flBaseRef, mult_sniper_charge_per_sec ); CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer ) { Vector vForward; AngleVectors( pPlayer->EyeAngles() + pPlayer->GetPunchAngle(), &vForward ); Vector vShootPos = pPlayer->Weapon_ShootPosition(); trace_t tr; UTIL_TraceLine( vShootPos, vShootPos + vForward * m_pWeaponInfo->GetWeaponData( m_iWeaponMode ).m_flRange, MASK_BLOCKLOS_AND_NPCS, pPlayer, COLLISION_GROUP_NONE, &tr ); CTFPlayer *pTarget = ToTFPlayer( tr.m_pEnt ); if ( pTarget && pTarget->IsAlive() && pTarget->GetTeamNumber() != pPlayer->GetTeamNumber() && !( pTarget->m_Shared.IsStealthed() && !pTarget->m_Shared.InCond( TF_COND_STEALTHED_BLINK ) ) ) { CALL_ATTRIB_HOOK_FLOAT( flBaseRef, mult_sniper_charge_per_sec_with_enemy_under_crosshair ); int nBeep = 0; CALL_ATTRIB_HOOK_FLOAT( nBeep, sniper_beep_with_enemy_under_crosshair ); if ( nBeep > 0 && !m_bWasAimedAtEnemy ) { pPlayer->EmitSound( "doomsday.warhead" ); } m_bWasAimedAtEnemy = true; } else { m_bWasAimedAtEnemy = false; } if ( pPlayer && ( pPlayer->m_Shared.GetCarryingRuneType() == RUNE_PRECISION || pPlayer->m_Shared.GetCarryingRuneType() == RUNE_HASTE ) ) { flBaseRef *= 3.0f; } else if ( pPlayer->m_Shared.GetCarryingRuneType() == RUNE_KING || pPlayer->m_Shared.InCond( TF_COND_KING_BUFFED ) ) { flBaseRef *= 1.5f; } // Prototype hack int iMaster = 0; CALL_ATTRIB_HOOK_INT_ON_OTHER( pPlayer, iMaster, ability_master_sniper ); if ( iMaster ) { flBaseRef *= RemapValClamped( iMaster, 1, 2, 1.5f, 3.f ); } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::MustBeZoomedToFire( void ) { int iModOnlyFireZoomed = 0; CALL_ATTRIB_HOOK_INT( iModOnlyFireZoomed, sniper_only_fire_zoomed ); return ( iModOnlyFireZoomed != 0 ); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::HandleNoScopeFireDeny( void ) { if ( m_flNextEmptySoundTime < gpGlobals->curtime ) { WeaponSound( SPECIAL2 ); #ifdef CLIENT_DLL ParticleProp()->Init( this ); ParticleProp()->Create( "dxhr_sniper_fizzle", PATTACH_POINT_FOLLOW, "muzzle" ); #endif m_flNextEmptySoundTime = gpGlobals->curtime + 0.5; } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- ETFDmgCustom CTFSniperRifle::GetPenetrateType() const { if ( IsFullyCharged() ) { int iPenetrate = 0; CALL_ATTRIB_HOOK_INT( iPenetrate, sniper_penetrate_players_when_charged ); if ( iPenetrate > 0 ) return TF_DMG_CUSTOM_PENETRATE_ALL_PLAYERS; } return BaseClass::GetPenetrateType(); } #ifdef SIXENSE //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- float CTFSniperRifle::GetRezoomTime() const { return m_flRezoomTime; } #endif //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::Fire( CTFPlayer *pPlayer ) { // Check the ammo. We don't use clip ammo, check the primary ammo type. if ( pPlayer->GetAmmoCount( m_iPrimaryAmmoType ) <= 0 ) { HandleFireOnEmpty(); return; } // Some weapons can only fire while zoomed if ( MustBeZoomedToFire() ) { if ( !IsZoomed() ) { HandleNoScopeFireDeny(); return; } } if ( m_flNextPrimaryAttack > gpGlobals->curtime ) return; // Fire the sniper shot. PrimaryAttack(); if ( IsZoomed() ) { // If we have more bullets, zoom out, play the bolt animation and zoom back in if ( pPlayer->GetAmmoCount( m_iPrimaryAmmoType ) > 0 ) { // do not zoom out if we're under rage or about to enter it if ( !( pPlayer->m_Shared.InCond( TF_COND_SNIPERCHARGE_RAGE_BUFF ) ) ) { float flUnzoomDelay = 0.5f; if ( !UsesClipsForAmmo1() ) { // Since sniper rifles don't actually use clips the fast reload hook also affects unzoom and zoom delays ApplyScopeSpeedModifications( flUnzoomDelay ); } SetRezoom( true, flUnzoomDelay ); // zoom out in 0.5 seconds, then rezoom } } else { //just zoom out SetRezoom( false, 0.5f ); // just zoom out in 0.5 seconds } } else { float flZoomDelay = SequenceDuration(); // Since sniper rifles don't actually use clips the fast reload hook also affects zoom delays ApplyScopeSpeedModifications( flZoomDelay ); // Prevent primary fire preventing zooms m_flNextSecondaryAttack = gpGlobals->curtime + flZoomDelay; } m_flChargedDamage = 0.0f; #ifdef GAME_DLL if ( m_hSniperDot ) { m_hSniperDot->ResetChargeTime(); } #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::SetRezoom( bool bRezoom, float flDelay ) { m_flUnzoomTime = gpGlobals->curtime + flDelay; m_bRezoomAfterShot = bRezoom; } //----------------------------------------------------------------------------- // Purpose: // Output : float //----------------------------------------------------------------------------- float CTFSniperRifle::GetProjectileDamage( void ) { float flDamage = MAX( m_flChargedDamage, TF_WEAPON_SNIPERRIFLE_DAMAGE_MIN ); float flDamageMod = 1.f; CALL_ATTRIB_HOOK_FLOAT( flDamageMod, mult_dmg ); flDamage *= flDamageMod; if ( TFGameRules() && TFGameRules()->IsPowerupMode() ) { CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer && pPlayer->m_Shared.GetCarryingRuneType() == RUNE_PRECISION ) { flDamage *= 2.f; } } if ( IsFullyCharged() ) { CALL_ATTRIB_HOOK_FLOAT( flDamage, sniper_full_charge_damage_bonus ); } return flDamage; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- int CTFSniperRifle::GetDamageType( void ) const { // Only do hit location damage if we're zoomed CTFPlayer *pPlayer = ToTFPlayer( GetPlayerOwner() ); if ( pPlayer && pPlayer->m_Shared.InCond( TF_COND_ZOOMED ) ) return BaseClass::GetDamageType(); int nDamageType = BaseClass::GetDamageType() & ~DMG_USE_HITLOCATIONS; return nDamageType; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::CreateSniperDot( void ) { // Server specific. #ifdef GAME_DLL // Check to see if we have already been created? if ( m_hSniperDot ) return; // Get the owning player (make sure we have one). CBaseCombatCharacter *pPlayer = GetOwner(); if ( !pPlayer ) return; // Create the sniper dot, but do not make it visible yet. m_hSniperDot = CSniperDot::Create( GetAbsOrigin(), pPlayer, true ); m_hSniperDot->ChangeTeam( pPlayer->GetTeamNumber() ); #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::DestroySniperDot( void ) { // Server specific. #ifdef GAME_DLL // Destroy the sniper dot. if ( m_hSniperDot ) { UTIL_Remove( m_hSniperDot ); m_hSniperDot = NULL; } #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::UpdateSniperDot( void ) { // Server specific. #ifdef GAME_DLL CBasePlayer *pPlayer = ToBasePlayer( GetOwner() ); if ( !pPlayer ) return; // Get the start and endpoints. Vector vecMuzzlePos = pPlayer->Weapon_ShootPosition(); Vector forward; pPlayer->EyeVectors( &forward ); Vector vecEndPos = vecMuzzlePos + ( forward * MAX_TRACE_LENGTH ); trace_t trace; UTIL_TraceLine( vecMuzzlePos, vecEndPos, ( MASK_SHOT & ~CONTENTS_WINDOW ), GetOwner(), COLLISION_GROUP_NONE, &trace ); // Update the sniper dot. if ( m_hSniperDot ) { CBaseEntity *pEntity = NULL; if ( trace.DidHitNonWorldEntity() ) { pEntity = trace.m_pEnt; if ( !pEntity || !pEntity->m_takedamage ) { pEntity = NULL; } } m_hSniperDot->Update( pEntity, trace.endpos, trace.plane.normal ); } #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::CanFireCriticalShot( bool bIsHeadshot ) { m_bCurrentAttackIsCrit = false; m_bCurrentShotIsHeadshot = false; if ( !BaseClass::CanFireCriticalShot( bIsHeadshot ) ) return false; CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer && pPlayer->m_Shared.IsCritBoosted() ) { m_bCurrentShotIsHeadshot = bIsHeadshot; return true; } // If we don't auto crit on a headshot, use standard criteria to determine other crits. if ( GetRifleType() == RIFLE_JARATE ) { return false; // Never auto crit on headshot. } // can only fire a crit shot if this is a headshot, unless we're critboosted if ( !bIsHeadshot ) return false; int iFullChargeHeadShotPenalty = 0; CALL_ATTRIB_HOOK_INT( iFullChargeHeadShotPenalty, sniper_no_headshot_without_full_charge ); if ( iFullChargeHeadShotPenalty != 0 ) { if ( !IsFullyCharged() ) return false; } int iCanCritNoScope = 0; CALL_ATTRIB_HOOK_INT( iCanCritNoScope, sniper_crit_no_scope ); if ( iCanCritNoScope == 0 ) { if ( pPlayer ) { // no crits if they're not zoomed if ( pPlayer->GetFOV() >= pPlayer->GetDefaultFOV() ) { return false; } // no crits for 0.2 seconds after starting to zoom if ( ( gpGlobals->curtime - pPlayer->GetFOVTime() ) < TF_WEAPON_SNIPERRIFLE_NO_CRIT_AFTER_ZOOM_TIME ) { return false; } } } m_bCurrentAttackIsCrit = true; m_bCurrentShotIsHeadshot = bIsHeadshot; return true; } //----------------------------------------------------------------------------- // Purpose: Our owner was stunned. //----------------------------------------------------------------------------- void CTFSniperRifle::OnControlStunned( void ) { BaseClass::OnControlStunned(); ZoomOut(); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- int CTFSniperRifle::GetCustomDamageType() const { if ( IsJarateRifle() ) { return TF_DMG_CUSTOM_PENETRATE_NONBURNING_TEAMMATE; } return TF_DMG_CUSTOM_PENETRATE_MY_TEAM; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::Detach( void ) { if ( IsZoomed() ) { ToggleZoom(); } BaseClass::Detach(); } #ifdef GAME_DLL //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::OnPlayerKill( CTFPlayer *pVictim, const CTakeDamageInfo &info ) { BaseClass::OnPlayerKill( pVictim, info ); if ( m_iConsecutiveKills == 3 ) { CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer ) { pPlayer->AwardAchievement( ACHIEVEMENT_TF_SNIPER_RIFLE_NO_MISSING ); } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::OnBulletFire( int iEnemyPlayersHit ) { BaseClass::OnBulletFire( iEnemyPlayersHit ); // Did we completely miss? CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if( iEnemyPlayersHit == 0 && pPlayer && pPlayer->m_Shared.InCond( TF_COND_AIMING ) ) { EconEntity_OnOwnerKillEaterEventNoPartner( assert_cast<CEconEntity *>( this ), pPlayer, kKillEaterEvent_NEGATIVE_SniperShotsMissed ); } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifle::ExplosiveHeadShot( CTFPlayer *pAttacker, CTFPlayer *pVictim ) { if ( !pAttacker ) return; if ( !pVictim ) return; int iExplosiveShot = 0; CALL_ATTRIB_HOOK_INT_ON_OTHER ( pAttacker, iExplosiveShot, explosive_sniper_shot ); // Stun the source float flStunDuration = 1.f + ( ( (float)iExplosiveShot - 1.f ) * 0.5f ); float flStunAmt = pVictim->IsMiniBoss() ? 0.5f : RemapValClamped( iExplosiveShot, 1, 3, 0.5f, 0.8f ); pVictim->m_Shared.StunPlayer( flStunDuration, flStunAmt, TF_STUN_MOVEMENT, pAttacker ); // Generate an explosion and look for nearby bots float flDmgRange = 125.f + iExplosiveShot * 25.f; float flDmg = 130.f + iExplosiveShot * 20.f; CBaseEntity *pObjects[ 32 ]; int nCount = UTIL_EntitiesInSphere( pObjects, ARRAYSIZE( pObjects ), pVictim->GetAbsOrigin(), flDmgRange, FL_CLIENT ); for ( int i = 0; i < nCount; i++ ) { if ( !pObjects[i] ) continue; if ( !pObjects[i]->IsAlive() ) continue; if ( pObjects[i] == pVictim ) continue; if ( pAttacker->InSameTeam( pObjects[i] ) ) continue; if ( !pVictim->FVisible( pObjects[i], MASK_OPAQUE ) ) continue; CTFPlayer *pTFPlayer = static_cast<CTFPlayer *>( pObjects[i] ); if ( !pTFPlayer ) continue; if ( pTFPlayer->m_Shared.InCond( TF_COND_PHASE ) || pTFPlayer->m_Shared.InCond( TF_COND_PASSTIME_INTERCEPTION ) ) continue; if ( pTFPlayer->m_Shared.IsInvulnerable() ) continue; // Stun flStunAmt = pTFPlayer->IsMiniBoss() ? 0.5f : RemapValClamped( iExplosiveShot, 1, 3, 0.5f, 0.8f ); pTFPlayer->m_Shared.StunPlayer( flStunDuration, flStunAmt, TF_STUN_MOVEMENT, pAttacker ); // DoT pTFPlayer->m_Shared.MakeBleed( pAttacker, this, 0.1f, flDmg ); // Shoot a beam at them CPVSFilter filter( pTFPlayer->WorldSpaceCenter() ); Vector vStart = pVictim->EyePosition(); Vector vEnd = pTFPlayer->EyePosition(); te_tf_particle_effects_control_point_t controlPoint = { PATTACH_ABSORIGIN, vEnd }; TE_TFParticleEffectComplex( filter, 0.0f, "dxhr_arm_muzzleflash", vStart, QAngle( 0, 0, 0 ), NULL, &controlPoint, pTFPlayer, PATTACH_CUSTOMORIGIN ); pTFPlayer->EmitSound( "Weapon_Upgrade.ExplosiveHeadshot" ); } } #endif //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::IsJarateRifle( void ) const { return GetJarateTimeInternal() > 0.f; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- float CTFSniperRifle::GetJarateTime( void ) const { if ( m_flChargedDamage > 0.f ) { return GetJarateTimeInternal(); } else return 0.0f; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- float CTFSniperRifle::GetJarateTimeInternal( void ) const { float flMaxJarateTime = 0.0f; CALL_ATTRIB_HOOK_FLOAT( flMaxJarateTime, jarate_duration ); if ( flMaxJarateTime > 0 ) { const float flMinJarateTime = 2.f; float flDuration = RemapValClamped( m_flChargedDamage, TF_WEAPON_SNIPERRIFLE_DAMAGE_MIN, TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX, flMinJarateTime, flMaxJarateTime ); return flDuration; } return 0.f; } //----------------------------------------------------------------------------- // Purpose: UI Progress (same as GetProgress() without the division by 100.0f) //----------------------------------------------------------------------------- bool CTFSniperRifle::IsRageFull( void ) { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) { return false; } return ( pPlayer->m_Shared.GetRageMeter() >= 100.0f ); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::EffectMeterShouldFlash( void ) { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) { return false; } if ( pPlayer && ( IsRageFull() || pPlayer->m_Shared.IsRageDraining() ) ) { return true; } return false; } //----------------------------------------------------------------------------- // Purpose: UI Progress //----------------------------------------------------------------------------- float CTFSniperRifle::GetProgress( void ) { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) { return 0.f; } return pPlayer->m_Shared.GetRageMeter() / 100.0f; } //============================================================================= // // Client specific functions. // #ifdef CLIENT_DLL //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifle::ShouldEjectBrass() { if ( GetJarateTimeInternal() > 0.f ) return false; else return true; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- float CTFSniperRifle::GetHUDDamagePerc( void ) { return (m_flChargedDamage / TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX); } //----------------------------------------------------------------------------- // Returns the sniper chargeup from 0 to 1 //----------------------------------------------------------------------------- class CProxySniperRifleCharge : public CResultProxy { public: void OnBind( void *pC_BaseEntity ); }; void CProxySniperRifleCharge::OnBind( void *pC_BaseEntity ) { Assert( m_pResult ); C_TFPlayer *pPlayer = C_TFPlayer::GetLocalTFPlayer(); if ( GetSpectatorTarget() != 0 && GetSpectatorMode() == OBS_MODE_IN_EYE ) { pPlayer = (C_TFPlayer *)UTIL_PlayerByIndex( GetSpectatorTarget() ); } if ( pPlayer ) { CTFSniperRifle *pWeapon = assert_cast<CTFSniperRifle*>(pPlayer->GetActiveTFWeapon()); if ( pWeapon ) { float flChargeValue = ( ( 1.0 - pWeapon->GetHUDDamagePerc() ) * 0.8 ) + 0.6; VMatrix mat, temp; Vector2D center( 0.5, 0.5 ); MatrixBuildTranslation( mat, -center.x, -center.y, 0.0f ); // scale { Vector2D scale( 1.0f, 0.25f ); MatrixBuildScale( temp, scale.x, scale.y, 1.0f ); MatrixMultiply( temp, mat, mat ); } MatrixBuildTranslation( temp, center.x, center.y, 0.0f ); MatrixMultiply( temp, mat, mat ); // translation { Vector2D translation( 0.0f, flChargeValue ); MatrixBuildTranslation( temp, translation.x, translation.y, 0.0f ); MatrixMultiply( temp, mat, mat ); } m_pResult->SetMatrixValue( mat ); } } if ( ToolsEnabled() ) { ToolFramework_RecordMaterialParams( GetMaterial() ); } } EXPOSE_INTERFACE( CProxySniperRifleCharge, IMaterialProxy, "SniperRifleCharge" IMATERIAL_PROXY_INTERFACE_VERSION ); #endif //============================================================================= // // Laser Dot functions. // IMPLEMENT_NETWORKCLASS_ALIASED( SniperDot, DT_SniperDot ) BEGIN_NETWORK_TABLE( CSniperDot, DT_SniperDot ) #ifdef CLIENT_DLL RecvPropFloat( RECVINFO( m_flChargeStartTime ) ), #else SendPropTime( SENDINFO( m_flChargeStartTime ) ), #endif END_NETWORK_TABLE() LINK_ENTITY_TO_CLASS( env_sniperdot, CSniperDot ); BEGIN_DATADESC( CSniperDot ) DEFINE_FIELD( m_vecSurfaceNormal, FIELD_VECTOR ), DEFINE_FIELD( m_hTargetEnt, FIELD_EHANDLE ), END_DATADESC() //----------------------------------------------------------------------------- // Purpose: Constructor. //----------------------------------------------------------------------------- CSniperDot::CSniperDot( void ) { m_vecSurfaceNormal.Init(); m_hTargetEnt = NULL; #ifdef CLIENT_DLL m_hSpriteMaterial = NULL; m_laserBeamEffect = NULL; #endif ResetChargeTime(); } //----------------------------------------------------------------------------- // Purpose: Destructor. //----------------------------------------------------------------------------- CSniperDot::~CSniperDot( void ) { #ifdef CLIENT_DLL if ( m_laserBeamEffect ) { ParticleProp()->StopEmissionAndDestroyImmediately( m_laserBeamEffect ); m_laserBeamEffect = NULL; } #endif } //----------------------------------------------------------------------------- // Purpose: // Input : &origin - // Output : CSniperDot //----------------------------------------------------------------------------- CSniperDot *CSniperDot::Create( const Vector &origin, CBaseEntity *pOwner, bool bVisibleDot ) { // Client specific. #ifdef CLIENT_DLL return NULL; // Server specific. #else // Create the sniper dot entity. CSniperDot *pDot = static_cast<CSniperDot*>( CBaseEntity::Create( "env_sniperdot", origin, QAngle( 0.0f, 0.0f, 0.0f ) ) ); if ( !pDot ) return NULL; //Create the graphic pDot->SetMoveType( MOVETYPE_NONE ); pDot->AddSolidFlags( FSOLID_NOT_SOLID ); pDot->AddEffects( EF_NOSHADOW ); UTIL_SetSize( pDot, -Vector( 4.0f, 4.0f, 4.0f ), Vector( 4.0f, 4.0f, 4.0f ) ); // Set owner. pDot->SetOwnerEntity( pOwner ); // Force updates even though we don't have a model. pDot->AddEFlags( EFL_FORCE_CHECK_TRANSMIT ); return pDot; #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CSniperDot::Update( CBaseEntity *pTarget, const Vector &vecOrigin, const Vector &vecNormal ) { SetAbsOrigin( vecOrigin ); m_vecSurfaceNormal = vecNormal; m_hTargetEnt = pTarget; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- Vector CSniperDot::GetChasePosition() { return GetAbsOrigin() - m_vecSurfaceNormal * 10; } //============================================================================= // // Client specific functions. // #ifdef CLIENT_DLL bool CSniperDot::GetRenderingPositions( C_TFPlayer *pPlayer, Vector &vecAttachment, Vector &vecEndPos, float &flSize ) { if ( !pPlayer ) return false; // Get the sprite rendering position. flSize = 6.0; bool bScaleSizeByDistance = false; const float c_fMaxSizeDistVR = 384.0f; const float c_flMaxSizeDistUnzoomed = 200.0f; if ( !pPlayer->IsDormant() ) { Vector vecDir; QAngle angles; float flDist = MAX_TRACE_LENGTH; // Always draw the dot in front of our faces when in first-person. if ( pPlayer->IsLocalPlayer() ) { // Take our view position and orientation vecAttachment = CurrentViewOrigin(); vecDir = CurrentViewForward(); // Clamp the forward distance for the sniper's firstperson flDist = c_fMaxSizeDistVR; flSize = 2.0; // Make the dot bigger when charging and not zoomed in (The Classic) if ( pPlayer->m_Shared.InCond( TF_COND_AIMING ) && !pPlayer->m_Shared.InCond( TF_COND_ZOOMED )) { flSize = 4.0f; bScaleSizeByDistance = true; } if ( UseVR() ) { // The view direction is not exactly the same as the weapon direction because of stereo, calibration, etc. g_ClientVirtualReality.OverrideWeaponHudAimVectors ( &vecAttachment, &vecDir ); // No clamping, thanks - we need the distance to be correct so that // vergence works properly, and we'll scale the size up accordingly. flDist = MAX_TRACE_LENGTH; bScaleSizeByDistance = true; } } else { // Take the owning player eye position and direction. vecAttachment = pPlayer->EyePosition(); QAngle anglesEye = pPlayer->EyeAngles(); AngleVectors( anglesEye, &vecDir ); } trace_t tr; CTraceFilterIgnoreFriendlyCombatItems filter( pPlayer, COLLISION_GROUP_NONE, pPlayer->GetTeamNumber() ); UTIL_TraceLine( vecAttachment, vecAttachment + ( vecDir * flDist ), MASK_SHOT, &filter, &tr ); // Backup off the hit plane, towards the source vecEndPos = tr.endpos + vecDir * -4; if ( UseVR() ) { float fDist = ( vecEndPos - vecAttachment ).Length(); if ( fDist > c_fMaxSizeDistVR ) { // Scale the dot up so it's still visible in first person. flSize *= ( fDist * ( 1.0f / c_fMaxSizeDistVR ) ); } } else if ( bScaleSizeByDistance ) { float fDist = ( vecEndPos - vecAttachment ).Length(); if ( fDist > c_flMaxSizeDistUnzoomed ) { // Scale the dot up so it's still visible in first person. flSize *= ( fDist * ( 1.0f / c_flMaxSizeDistUnzoomed ) ); } } } else { // Just use our position if we can't predict it otherwise. vecAttachment = GetAbsOrigin(); vecEndPos = GetAbsOrigin(); } return true; } //----------------------------------------------------------------------------- // Purpose: // TFTODO: Make the sniper dot get brighter the more damage it will do. //----------------------------------------------------------------------------- int CSniperDot::DrawModel( int flags ) { // Get the owning player. C_TFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( !pPlayer ) return -1; Vector vecAttachement; Vector vecEndPos; float flSize; if ( !GetRenderingPositions( pPlayer, vecAttachement, vecEndPos, flSize ) ) { return -1; } // Draw our laser dot in space. CMatRenderContextPtr pRenderContext( materials ); pRenderContext->Bind( m_hSpriteMaterial, this ); CTFWeaponBase *pBaseWeapon = pPlayer->GetActiveTFWeapon(); CTFSniperRifle *pWeapon = dynamic_cast< CTFSniperRifle* >( pBaseWeapon ); float flLifeTime = gpGlobals->curtime - m_flChargeStartTime; float flChargePerSec = TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC; if ( pWeapon ) { pWeapon->ApplyChargeSpeedModifications( flChargePerSec ); } // Sniper Rage if ( pPlayer->m_Shared.InCond( TF_COND_SNIPERCHARGE_RAGE_BUFF ) ) { flChargePerSec *= 1.25f; } float flStrength; if ( pWeapon ) { flStrength = pWeapon->GetHUDDamagePerc(); // FIXME: We should find out what's causing this and fix it. AssertMsg1( flStrength >= ( 0.0f - FLT_EPSILON ) && flStrength <= ( 1.0f + FLT_EPSILON ), "GetHUDDamagePerc returned out of range value: %f", flStrength ); flStrength = clamp( flStrength, 0.1f, 1.0f ); } else { flStrength = RemapValClamped( flLifeTime, 0.0, TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX / flChargePerSec, 0.1f, 1.0f ); } color32 innercolor = { 255, 255, 255, 255 }; color32 outercolor = { 255, 255, 255, 128 }; DrawSprite( vecEndPos, flSize, flSize, outercolor ); DrawSprite( vecEndPos, flSize * flStrength, flSize * flStrength, innercolor ); // Successful. return 1; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CSniperDot::ShouldDraw( void ) { if ( IsEffectActive( EF_NODRAW ) ) return false; // Don't draw the sniper dot when in thirdperson. if ( ::input->CAM_IsThirdPerson() ) return false; return true; } void CSniperDot::ClientThink( void ) { // snipers have laser sights in PvE mode if ( TFGameRules()->IsPVEModeActive() && GetTeamNumber() == TF_TEAM_PVE_INVADERS ) { C_TFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer ) { if ( !m_laserBeamEffect ) { m_laserBeamEffect = ParticleProp()->Create( "laser_sight_beam", PATTACH_ABSORIGIN_FOLLOW ); } if ( m_laserBeamEffect ) { m_laserBeamEffect->SetSortOrigin( m_laserBeamEffect->GetRenderOrigin() ); m_laserBeamEffect->SetControlPoint( 2, Vector( 0, 0, 255 ) ); Vector vecAttachment; Vector vecEndPos; float flSize; if ( pPlayer->GetAttachment( "eye_1", vecAttachment ) ) { m_laserBeamEffect->SetControlPoint( 1, vecAttachment ); } else if ( GetRenderingPositions( pPlayer, vecAttachment, vecEndPos, flSize ) ) { m_laserBeamEffect->SetControlPoint( 1, vecAttachment ); } } } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CSniperDot::OnDataChanged( DataUpdateType_t updateType ) { if ( updateType == DATA_UPDATE_CREATED ) { if ( GetTeamNumber() == TF_TEAM_BLUE ) { m_hSpriteMaterial.Init( SNIPER_DOT_SPRITE_BLUE, TEXTURE_GROUP_CLIENT_EFFECTS ); } else { m_hSpriteMaterial.Init( SNIPER_DOT_SPRITE_RED, TEXTURE_GROUP_CLIENT_EFFECTS ); } SetNextClientThink( CLIENT_THINK_ALWAYS ); } } #endif // CLIENT_DLL #ifdef GAME_DLL void CTFSniperRifleDecap::OnPlayerKill( CTFPlayer *pVictim, const CTakeDamageInfo &info ) { BaseClass::OnPlayerKill( pVictim, info ); CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer && IsHeadshot( info.GetDamageCustom() ) ) { // If we got a headshot kill, increment our number of decapitations. int iDecaps = pPlayer->m_Shared.GetDecapitations() + 1; pPlayer->m_Shared.SetDecapitations( iDecaps ); } } #endif // GAME_DLL static const int MAX_HEAD_BONUS = 6; //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- float CTFSniperRifleDecap::SniperRifleChargeRateMod() { return ( .25f * ( MIN( GetCount(), MAX_HEAD_BONUS ) - 2 ) ) * TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- int CTFSniperRifleDecap::GetCount( void ) { CTFPlayer *pPlayer = ToTFPlayer( GetOwnerEntity() ); if ( pPlayer ) { return pPlayer->m_Shared.GetDecapitations(); } return 0; } //----------------------------------------------------------------------------- // Purpose: Constructor. //----------------------------------------------------------------------------- CTFSniperRifleClassic::CTFSniperRifleClassic() { m_bCharging = false; #ifdef CLIENT_DLL m_pChargedEffect = NULL; #endif } //----------------------------------------------------------------------------- // Purpose: Destructor. //----------------------------------------------------------------------------- CTFSniperRifleClassic::~CTFSniperRifleClassic() { #ifdef CLIENT_DLL if ( m_pChargedEffect ) { ParticleProp()->StopEmissionAndDestroyImmediately( m_pChargedEffect ); m_pChargedEffect = NULL; } #endif } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::Precache() { BaseClass::Precache(); PrecacheParticleSystem( SNIPER_CHARGE_BEAM_RED ); PrecacheParticleSystem( SNIPER_CHARGE_BEAM_BLUE ); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::ZoomOut( void ) { CTFWeaponBaseGun::ZoomOut(); // intentionally skipping CTFSniperRifle::ZoomOut() } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::ZoomIn( void ) { // Start aiming. CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) return; if ( pPlayer->GetAmmoCount( m_iPrimaryAmmoType ) <= 0 ) return; CTFWeaponBaseGun::ZoomIn(); // intentionally skipping CTFSniperRifle::ZoomIn() } //----------------------------------------------------------------------------- // Purpose: Secondary attack. //----------------------------------------------------------------------------- void CTFSniperRifleClassic::Zoom( void ) { ToggleZoom(); // at least 0.1 seconds from now, but don't stomp a previous value m_flNextSecondaryAttack = gpGlobals->curtime + TF_WEAPON_SNIPERRIFLE_ZOOM_TIME; } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::HandleZooms( void ) { // Get the owning player. CTFPlayer *pPlayer = ToTFPlayer( GetOwner() ); if ( !pPlayer ) return; // Handle the zoom when taunting. if ( pPlayer->m_Shared.InCond( TF_COND_TAUNTING ) ) { if ( IsZoomed() ) { ToggleZoom(); return; } } if ( ( pPlayer->m_nButtons & IN_ATTACK2 ) && ( m_flNextSecondaryAttack <= gpGlobals->curtime ) ) { Zoom(); } } #ifdef CLIENT_DLL void CTFSniperRifleClassic::OnDataChanged( DataUpdateType_t updateType ) { BaseClass::OnDataChanged( updateType ); ManageChargeBeam(); } #endif //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::ItemPostFrame( void ) { // If we're lowered, we're not allowed to fire if ( m_bLowered ) return; // Get the owning player. CTFPlayer *pPlayer = ToTFPlayer( GetOwner() ); if ( !pPlayer ) return; if ( !CanAttack() ) { if ( IsZoomed() ) { ToggleZoom(); } WeaponReset(); return; } HandleZooms(); #ifdef GAME_DLL // Update the sniper dot position if we have one if ( m_hSniperDot ) { UpdateSniperDot(); } #endif if ( pPlayer->m_Shared.InCond( TF_COND_TAUNTING ) ) { WeaponReset(); return; } if ( ( pPlayer->m_nButtons & IN_ATTACK ) && ( m_flNextPrimaryAttack <= gpGlobals->curtime ) ) { if ( !m_bCharging ) { pPlayer->m_Shared.AddCond( TF_COND_AIMING ); pPlayer->TeamFortress_SetSpeed(); m_bCharging = true; #ifdef GAME_DLL // Create the sniper dot. CreateSniperDot(); pPlayer->ClearExpression(); #endif } float fSniperRifleChargePerSec = m_flChargePerSec; ApplyChargeSpeedModifications( fSniperRifleChargePerSec ); fSniperRifleChargePerSec += SniperRifleChargeRateMod(); // we don't want sniper charge rate to go too high. fSniperRifleChargePerSec = clamp( fSniperRifleChargePerSec, 0, 2.f * TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC ); m_flChargedDamage = MIN( m_flChargedDamage + gpGlobals->frametime * fSniperRifleChargePerSec, TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX ); #ifdef CLIENT_DLL // play the recharged bell if we're fully charged if ( IsFullyCharged() && !m_bPlayedBell ) { m_bPlayedBell = true; if ( tf_sniper_fullcharge_bell.GetBool() ) { C_TFPlayer::GetLocalTFPlayer()->EmitSound( "TFPlayer.ReCharged" ); } } #endif } else if ( m_bCharging ) { if ( pPlayer->GetGroundEntity() ) { Fire( pPlayer ); } else { pPlayer->EmitSound( "Player.DenyWeaponSelection" ); } WeaponReset(); } else { // Idle. // No fire buttons down or reloading if ( !ReloadOrSwitchWeapons() && ( m_bInReload == false ) ) { WeaponIdle(); } } } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- int CTFSniperRifleClassic::GetDamageType( void ) const { return CTFWeaponBaseGun::GetDamageType(); // intentionally skipping CTFSniperRifle::GetDamageType() } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifleClassic::Holster( CBaseCombatWeapon *pSwitchingTo ) { WeaponReset(); return BaseClass::Holster( pSwitchingTo ); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifleClassic::Deploy( void ) { WeaponReset(); return BaseClass::Deploy(); } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::WeaponReset( void ) { m_flChargedDamage = 0.0f; m_bCharging = false; #ifdef CLIENT_DLL ManageChargeBeam(); #endif CTFPlayer *pPlayer = ToTFPlayer( GetOwner() ); if ( pPlayer ) { pPlayer->m_Shared.RemoveCond( TF_COND_AIMING ); pPlayer->TeamFortress_SetSpeed(); } #ifdef GAME_DLL // Destroy the sniper dot. DestroySniperDot(); if ( pPlayer ) { pPlayer->ClearExpression(); } #else m_bPlayedBell = false; #endif m_bCurrentShotIsHeadshot = false; m_flChargePerSec = TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC; CTFWeaponBase::WeaponReset(); // intentionally skipping CTFSniperRifle::WeaponReset() } //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- bool CTFSniperRifleClassic::Lower( void ) { if ( BaseClass::Lower() ) { WeaponReset(); return true; } return false; } #ifdef CLIENT_DLL //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::ManageChargeBeam( void ) { if ( m_bCharging ) { if ( !m_pChargedEffect ) { m_pChargedEffect = ParticleProp()->Create( ( GetTeamNumber() == TF_TEAM_RED ) ? SNIPER_CHARGE_BEAM_RED : SNIPER_CHARGE_BEAM_BLUE, PATTACH_POINT_FOLLOW, "laser" ); } } else { if ( m_pChargedEffect ) { ParticleProp()->StopEmissionAndDestroyImmediately( m_pChargedEffect ); m_pChargedEffect = NULL; } } } #endif //----------------------------------------------------------------------------- // Purpose: //----------------------------------------------------------------------------- void CTFSniperRifleClassic::Detach( void ) { WeaponReset(); BaseClass::Detach(); } #ifdef STAGING_ONLY // ******************************************************************************************************** // CTFSniperRifleRevolver // ******************************************************************************************************** void CTFSniperRifleRevolver::PrimaryAttack() { BaseClass::PrimaryAttack(); #ifdef GAME_DLL // Head bob CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer /*&& pPlayer->m_Shared.InCond( TF_COND_ZOOMED )*/ ) { float impulse = RandomFloat( -2.0f, -1.0f ); if ( pPlayer->GetFlags() & FL_DUCKING ) { impulse = RandomFloat( -0.5f, -0.2f ); } pPlayer->ApplyPunchImpulseX( impulse ); } float flCharge = (m_flChargedDamage / TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX); if ( flCharge > 0.99 ) { // Only at full charge do you get fast attack speed // reduce the time between attacks float flCurrTime = gpGlobals->curtime; //float flTimeBetweenShots = m_flNextPrimaryAttack - flCurrTime; //float flTime = RemapVal( flCharge, 0.0, 1, flTimeBetweenShots, 0.2 ); m_flNextPrimaryAttack = flCurrTime + 0.3; } #endif } //----------------------------------------------------------------------------- float CTFSniperRifleRevolver::GetProjectileDamage() { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( pPlayer && pPlayer->m_Shared.InCond( TF_COND_ZOOMED ) ) { float flCharge = ( m_flChargedDamage / TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX ); if ( flCharge > 0.99 ) { return 75.0f; // Full Charge dmg bonus is less then the normal one (150) } return 50.0f; } return 40.0; } //----------------------------------------------------------------------------- bool CTFSniperRifleRevolver::Reload( void ) { CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) return false; // do not reload if zoomed unless you are empty if ( m_iClip1 > 0 && pPlayer->m_Shared.InCond( TF_COND_ZOOMED ) ) return false; bool bReload = CTFWeaponBaseGun::Reload(); // intentionally skipping CTFSniperRifle::Reload(). if ( bReload && m_iClip1 <= 0 && m_iClip1 != -1 ) { if ( pPlayer->m_Shared.InCond( TF_COND_ZOOMED ) ) { ZoomOut(); m_bRezoomAfterShot = pPlayer->ShouldAutoRezoom(); } } return bReload; } //----------------------------------------------------------------------------- void CTFSniperRifleRevolver::ZoomIn( void ) { // Start aiming. CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) return; if ( m_iClip1 <= 0 && m_iClip1 != -1 ) return; pPlayer->m_Shared.AddCond( TF_COND_AIMING ); pPlayer->TeamFortress_SetSpeed(); m_flChargedDamage = 0; #ifdef GAME_DLL // Create the sniper dot. CreateSniperDot(); pPlayer->ClearExpression(); #endif CTFWeaponBaseGun::ZoomIn(); // intentionally skipping CTFSniperRifle::ZoomIn() } //----------------------------------------------------------------------------- void CTFSniperRifleRevolver::ZoomOut( void ) { // Start aiming. CTFPlayer *pPlayer = GetTFPlayerOwner(); if ( !pPlayer ) return; pPlayer->m_Shared.RemoveCond( TF_COND_AIMING ); pPlayer->TeamFortress_SetSpeed(); m_flChargedDamage = 0; #ifdef GAME_DLL // Destroy the sniper dot. DestroySniperDot(); pPlayer->ClearExpression(); #endif CTFWeaponBaseGun::ZoomOut(); // intentionally skipping CTFSniperRifle::ZoomOut() } //----------------------------------------------------------------------------- void CTFSniperRifleRevolver::ItemPostFrame( void ) { // If we're lowered, we're not allowed to fire if ( m_bLowered ) return; // Get the owning player. CTFPlayer *pPlayer = ToTFPlayer( GetOwner() ); if ( !pPlayer ) return; if ( !CanAttack() ) { if ( IsZoomed() ) { ToggleZoom(); } WeaponReset(); } if ( m_bRezoomAfterShot && m_iClip1 > 0 ) { Zoom(); m_bRezoomAfterShot = false; } HandleZooms(); #ifdef GAME_DLL // Update the sniper dot position if we have one if ( m_hSniperDot ) { UpdateSniperDot(); } #endif if ( pPlayer->m_Shared.InCond( TF_COND_TAUNTING ) ) { WeaponReset(); return; } // Handle Charge Meter if ( m_flNextSecondaryAttack <= gpGlobals->curtime ) { // Don't start charging in the time just after a shot before we unzoom to play rack anim. if ( pPlayer->m_Shared.InCond( TF_COND_AIMING ) && !m_bRezoomAfterShot ) { float fSniperRifleChargePerSec = m_flChargePerSec; ApplyChargeSpeedModifications( fSniperRifleChargePerSec ); fSniperRifleChargePerSec += SniperRifleChargeRateMod(); // we don't want sniper charge rate to go too high. fSniperRifleChargePerSec = clamp( fSniperRifleChargePerSec, 0, 2.f * TF_WEAPON_SNIPERRIFLE_CHARGE_PER_SEC ); m_flChargedDamage = MIN( m_flChargedDamage + gpGlobals->frametime * fSniperRifleChargePerSec, TF_WEAPON_SNIPERRIFLE_DAMAGE_MAX ); #ifdef CLIENT_DLL // play the recharged bell if we're fully charged if ( IsFullyCharged() && !m_bPlayedBell ) { m_bPlayedBell = true; if ( tf_sniper_fullcharge_bell.GetBool() ) { C_TFPlayer::GetLocalTFPlayer()->EmitSound( "TFPlayer.ReCharged" ); } } #endif } else { m_flChargedDamage = MAX( 0, m_flChargedDamage - gpGlobals->frametime * TF_WEAPON_SNIPERRIFLE_UNCHARGE_PER_SEC ); } } return CTFWeaponBaseGun::ItemPostFrame(); // intentionally skipping CTFSniperRifle::ItemPostFrame(). This should just fire the gun } //----------------------------------------------------------------------------- bool CTFSniperRifleRevolver::CanFireCriticalShot( bool bIsHeadshot ) { return CTFSniperRifle::CanFireCriticalShot( bIsHeadshot ); // Skip TFC Sniper Rifle } //----------------------------------------------------------------------------- // ConVar tf_sniper_bolt_speed( "tf_sniper_bolt_speed", "3000", FCVAR_REPLICATED, "Dev Convar - Speed of projectile for Revolver Sniper"); ConVar tf_sniper_bolt_gravity( "tf_sniper_bolt_gravity", "0.1", FCVAR_REPLICATED, "Dev Convar - Gravity of projectile for Revolver Sniper"); float CTFSniperRifleRevolver::GetProjectileSpeed( void ) { //return 4900.0; return tf_sniper_bolt_speed.GetFloat(); } //----------------------------------------------------------------------------- float CTFSniperRifleRevolver::GetProjectileGravity( void ) { //return 0.1; return tf_sniper_bolt_gravity.GetFloat(); } #endif // STAGING_ONLY