csgo-2018-source/game/server/portal2/prop_weightedcube.cpp
2021-07-24 21:11:47 -07:00

1582 lines
41 KiB
C++

//========= Copyright © 1996-2009, Valve Corporation, All rights reserved. ============//
//
// Purpose: First-class cube entity so we can query by type and generally make inferences
// that are harder to do without an entity of that type.
//
//=====================================================================================//
#include "cbase.h"
#include "props.h"
#include "ai_utils.h"
#include "physics_saverestore.h"
#include "phys_controller.h"
#include "portal_base2d.h"
#include "portal/weapon_physcannon.h"
#include "datacache/imdlcache.h"
#include "prop_weightedcube.h"
#include "portal_player.h"
#include "portal_player_shared.h"
#include "world.h"
#include "vcollide_parse.h"
#include "portal_gamestats.h"
#include "saverestore_utlvector.h"
#include "trigger_portal_cleanser.h"
#include "portal_mp_gamerules.h"
#include "cvisibilitymonitor.h"
ConVar reflector_cube_disabled_think_rate( "reflector_cube_disabled_think_rate", "0.1f", FCVAR_DEVELOPMENTONLY, "The rate at which the cube should think when it is disabled." );
ConVar reflector_cube_disabled_nudge_time( "reflector_cube_disabled_nudge_time", "0.5f", FCVAR_DEVELOPMENTONLY, "The amount of time the cube needs to be touched before it gets enabled again." );
ConVar reflector_cube_disabled_use_touch_check( "reflector_cube_disabled_use_touch_check", "0", FCVAR_DEVELOPMENTONLY, "Use touch checks to determine when to enable the cube." );
ConVar sv_portal2_pickup_hint_range( "sv_portal2_pickup_hint_range", "350.0", FCVAR_NONE );
// FIXME: Bring this back for DLC2
//extern ConVar sv_schrodinger_laser_world_aligned;
//Standard cube skins
enum StandardCubeSkinType_t
{
CUBE_STANDARD_CLEAN_SKIN = 0,
CUBE_STANDARD_CLEAN_ACTIVATED_SKIN = 2,
CUBE_STANDARD_RUSTED_SKIN = 3,
CUBE_STANDARD_RUSTED_ACTIVATED_SKIN = 5,
CUBE_STANDARD_BOUNCE_SKIN = 6,
CUBE_STANDARD_BOUNCE_ACTIVATED_SKIN = 10,
CUBE_STANDARD_SPEED_SKIN = 7,
CUBE_STANDARD_SPEED_ACTIVATED_SKIN = 11
};
//Companion cube skins
enum CompanionCubeSkinType_t
{
CUBE_COMPANION_CLEAN_SKIN = 1,
CUBE_COMPANION_CLEAN_ACTIVATED_SKIN = 4,
CUBE_COMPANION_BOUNCE_SKIN = 8,
CUBE_COMPANION_BOUNCE_ACTIVATED_SKIN = 8,
CUBE_COMPANION_SPEED_SKIN = 9,
CUBE_COMPANION_SPEED_ACTIVATED_SKIN = 9
};
//Reflective cubs skins
enum ReflectiveCubeSkinType_t
{
CUBE_REFLECTIVE_CLEAN_SKIN = 0,
CUBE_REFLECTIVE_RUSTED_SKIN = 1,
CUBE_REFLECTIVE_BOUNCE_SKIN = 2,
CUBE_REFLECTIVE_SPEED_SKIN = 3
};
//Sphere skins
enum WeightedSpherSkinType_t
{
CUBE_SPHERE_CLEAN_SKIN = 0,
CUBE_SPHERE_CLEAN_ACTIVATED_SKIN = 1,
CUBE_SPHERE_BOUNCE_SKIN = 2,
CUBE_SPHERE_BOUNCE_ACTIVATED_SKIN = 2,
CUBE_SPHERE_SPEED_SKIN = 3,
CUBE_SPHERE_SPEED_ACTIVATED_SKIN = 3
};
//Antique cube skins
enum AntiqueCubeSkinType_t
{
CUBE_ANTIQUE_CLEAN_SKIN = 0,
CUBE_ANTIQUE_BOUNCE_SKIN = 1,
CUBE_ANTIQUE_SPEED_SKIN = 2
};
//Schrodinger cube skins
enum SchrodingerCubeSkinType_t
{
CUBE_SCHRODINGER_CLEAN_SKIN = 4,
CUBE_SCHRODINGER_BOUNCE_SKIN = 5,
CUBE_SCHRODINGER_SPEED_SKIN = 6
};
const char SCHRODINGER_THINK_CONTEXT[] = "Schrodinger Think Context";
LINK_ENTITY_TO_CLASS( cube_rotationcontroller, CCubeRotationController );
//---------------------------------------------------------
// Save/Restore
//---------------------------------------------------------
BEGIN_DATADESC( CCubeRotationController )
DEFINE_FIELD( m_bEnabled, FIELD_BOOLEAN ),
DEFINE_FIELD( m_flSuspendTime, FIELD_TIME ),
DEFINE_FIELD( m_worldGoalAxis, FIELD_VECTOR ),
DEFINE_FIELD( m_localTestAxis, FIELD_VECTOR ),
DEFINE_PHYSPTR( m_pController ),
DEFINE_FIELD( m_angularLimit, FIELD_FLOAT ),
DEFINE_FIELD( m_pParent, FIELD_CLASSPTR ),
END_DATADESC()
IMPLEMENT_AUTO_LIST( IPropWeightedCubeAutoList );
CCubeRotationController::~CCubeRotationController()
{
if ( m_pController )
{
physenv->DestroyMotionController( m_pController );
m_pController = NULL;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CCubeRotationController::Spawn( void )
{
m_bEnabled = false;
// align the object's local Z axis
m_localTestAxis.Init( 1, 0, 0 );
// with the world's Z axis
m_worldGoalAxis.Init( 0, 0, 1 );
// recover from up to 25 degrees / sec angular velocity
m_angularLimit = 25;
m_flSuspendTime = 0;
SetMoveType( MOVETYPE_NONE );
}
//-----------------------------------------------------------------------------
// Purpose: Set the vector we'll try to match
//-----------------------------------------------------------------------------
void CCubeRotationController::SetAlignmentVector( const Vector &vecAlign )
{
m_worldGoalAxis = vecAlign;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CCubeRotationController::Activate( void )
{
BaseClass::Activate();
if ( m_pParent == NULL )
{
UTIL_Remove(this);
return;
}
IPhysicsObject *pPhys = m_pParent->VPhysicsGetObject();
if ( pPhys == NULL )
{
UTIL_Remove(this);
return;
}
//Setup the motion controller
if ( !m_pController )
{
m_pController = physenv->CreateMotionController( (IMotionEvent *)this );
m_pController->AttachObject( pPhys, true );
}
else
{
m_pController->SetEventHandler( this );
}
}
//-----------------------------------------------------------------------------
// Purpose: Simulation will be suspended after this amount of time
//-----------------------------------------------------------------------------
void CCubeRotationController::SuspendAfter( float flSuspendTime )
{
m_flSuspendTime = flSuspendTime;
}
//-----------------------------------------------------------------------------
// Purpose: Actual simulation for tip controller
//-----------------------------------------------------------------------------
IMotionEvent::simresult_e CCubeRotationController::Simulate( IPhysicsMotionController *pController, IPhysicsObject *pObject, float deltaTime, Vector &linear, AngularImpulse &angular )
{
if ( Enabled() == false )
return SIM_NOTHING;
// Don't simulate if we're being carried by the player
// if ( m_pParent->IsBeingCarriedByPlayer() )
// return SIM_NOTHING;
float flAngularLimit = m_angularLimit;
// If we were just dropped by a friendly player, stabilise better
/*
if ( m_pParent->WasJustDroppedByPlayer() )
{
// Increase the controller strength a little
flAngularLimit += 20;
}
else
*/
{
// If the turret has some vertical velocity, don't simulate
/*
Vector vecVelocity;
AngularImpulse angImpulse;
pObject->GetVelocity( &vecVelocity, &angImpulse );
if ( (vecVelocity.LengthSqr() > CNPC_FloorTurret::fMaxTipControllerVelocity) || (angImpulse.LengthSqr() > CNPC_FloorTurret::fMaxTipControllerAngularVelocity) )
return SIM_NOTHING;
*/
}
linear.Init();
AngularImpulse angVel;
pObject->GetVelocity( NULL, &angVel );
matrix3x4_t matrix;
// get the object's local to world transform
pObject->GetPositionMatrix( &matrix );
// Get the alignment axis in object space
Vector currentLocalTargetAxis;
VectorIRotate( m_worldGoalAxis, matrix, currentLocalTargetAxis );
float invDeltaTime = (1/deltaTime);
angular = ComputeRotSpeedToAlignAxes( m_localTestAxis, currentLocalTargetAxis, angVel, 1.0, invDeltaTime * invDeltaTime, flAngularLimit * invDeltaTime );
return SIM_LOCAL_ACCELERATION;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CCubeRotationController::Enable( bool state )
{
m_bEnabled = state;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CCubeRotationController::Enabled( void )
{
if ( m_flSuspendTime > gpGlobals->curtime )
return true;
return m_bEnabled;
}
CEG_NOINLINE CCubeRotationController * CCubeRotationController::CreateRotationController( CBaseEntity *pOwner )
{
if ( pOwner == NULL )
return NULL;
CCubeRotationController *pController = (CCubeRotationController *) Create( "cube_rotationcontroller", pOwner->GetAbsOrigin(), pOwner->GetAbsAngles() );
if ( pController != NULL )
{
pController->m_pParent = pOwner;
}
return pController;
}
CEG_PROTECT_STATIC_MEMBER_FUNCTION( CCubeRotationController_CreateRotationController, CCubeRotationController::CreateRotationController );
LINK_ENTITY_TO_CLASS( prop_weighted_cube, CPropWeightedCube );
BEGIN_DATADESC( CPropWeightedCube )
DEFINE_FIELD( m_vecCarryAngles, FIELD_VECTOR ),
DEFINE_FIELD( m_pController, FIELD_EHANDLE ),
DEFINE_FIELD( m_bMovementDisabled, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bActivated, FIELD_BOOLEAN ),
DEFINE_FIELD( m_bTouchedByPlayer, FIELD_BOOLEAN ),
DEFINE_FIELD( m_nCurrentPaintedType, FIELD_INTEGER ),
DEFINE_FIELD( m_bPickupDisabled, FIELD_BOOLEAN ),
DEFINE_SOUNDPATCH( m_pSchrodingerSound ),
DEFINE_THINKFUNC( SchrodingerThink ),
DEFINE_THINKFUNC( DisabledThink ),
DEFINE_THINKFUNC( TractorBeamThink ),
DEFINE_THINKFUNC( ExitTractorBeamThink ),
DEFINE_KEYFIELD( m_bRusted, FIELD_BOOLEAN, "SkinType" ),
DEFINE_KEYFIELD( m_nCubeType, FIELD_INTEGER, "CubeType" ),
DEFINE_KEYFIELD( m_bNewSkins, FIELD_BOOLEAN, "NewSkins" ),
DEFINE_INPUTFUNC( FIELD_VOID, "Dissolve", InputDissolve ),
DEFINE_INPUTFUNC( FIELD_VOID, "SilentDissolve", InputSilentDissolve ),
DEFINE_INPUTFUNC( FIELD_VOID, "PreDissolveJoke", InputPreDissolveJoke ),
DEFINE_INPUTFUNC( FIELD_VOID, "DisablePortalFunnel", InputDisablePortalFunnel ),
DEFINE_INPUTFUNC( FIELD_VOID, "EnablePortalFunnel", InputEnablePortalFunnel ),
DEFINE_INPUTFUNC( FIELD_VOID, "ExitDisabledState", InputExitDisabledState ),
DEFINE_INPUTFUNC( FIELD_INTEGER, "SetPaint", InputSetPaint ),
DEFINE_INPUTFUNC( FIELD_VOID, "DisablePickup", InputDisablePickup ),
DEFINE_INPUTFUNC( FIELD_VOID, "EnablePickup", InputEnablePickup ),
DEFINE_OUTPUT( m_OnFizzled, "OnFizzled" ),
DEFINE_OUTPUT( m_OnOrangePickUp, "OnOrangePickUp" ),
DEFINE_OUTPUT( m_OnBluePickUp, "OnBluePickUp" ),
DEFINE_OUTPUT( m_OnPainted, "OnPainted" ),
END_DATADESC()
//no new networked fields, just need entity specific virtual functions defined on the client
IMPLEMENT_SERVERCLASS_ST( CPropWeightedCube, DT_PropWeightedCube )
END_SEND_TABLE()
const char *CUBE_MODEL = "models/props/metal_box.mdl";
const char *CUBE_REFLECT_MODEL = "models/props/reflection_cube.mdl";
const char *CUBE_SPHERE_MODEL = "models/props_gameplay/mp_ball.mdl";
const char *CUBE_FX_FIZZLER_MODEL = "models/props/metal_box_fx_fizzler.mdl";
const char *CUBE_ANTIQUE_MODEL = "models/props_underground/underground_weighted_cube.mdl";
const char *CUBE_SCHRODINGER_MODEL = "models/props/reflection_cube.mdl";
CHandle< CPropWeightedCube > CPropWeightedCube::m_hSchrodingerDangling;
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CPropWeightedCube::CPropWeightedCube()
: m_bMovementDisabled( false ),
m_bRusted( false ),
m_bActivated( false ),
m_nCubeType( CUBE_STANDARD ),
m_bTouchedByPlayer( false )
{
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CEG_NOINLINE void CPropWeightedCube::Spawn( void )
{
// Start out with nothing
m_vecCarryAngles.Init(0,0,0);
ConvertOldSkins();
Precache();
m_nCurrentPaintedType = NO_POWER;
m_bPickupDisabled = false;
SetCubeType();
CEG_PROTECT_VIRTUAL_FUNCTION( CPropWeightedCube_Spawn );
m_nBouncyMaterialIndex = physprops->GetSurfaceIndex( "WeightedCube_Bounce" );
SetInteraction( PROPINTER_PHYSGUN_ALLOW_OVERHEAD );
BaseClass::Spawn();
SetCollisionGroup( COLLISION_GROUP_WEIGHTED_CUBE );
if ( m_nCubeType == CUBE_SCHRODINGER )
{
SetContextThink( &CPropWeightedCube::SchrodingerThink, gpGlobals->curtime + reflector_cube_disabled_think_rate.GetFloat(), SCHRODINGER_THINK_CONTEXT );
}
#if !defined( _GAMECONSOLE ) && !defined( NO_STEAM )
g_PortalGameStats.Event_CubeSpawn();
#endif
VisibilityMonitor_AddEntity_NotVisibleThroughGlass( this, sv_portal2_pickup_hint_range.GetFloat() - 50.0f, NULL, NULL );
SetFadeDistance( -1.0f, 0.0f );
SetGlobalFadeScale( 0.0f );
}
void CPropWeightedCube::Activate( void )
{
SetPaintedMaterial( (PaintPowerType)( m_PrePaintedPower ) );
#if 0
if ( !m_pSchrodingerSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
CPASAttenuationFilter filter( this );
m_pSchrodingerSound = controller.SoundCreate( filter, entindex(), "music.laser_node_02.play" );
controller.Play( m_pSchrodingerSound, 0, RandomFloat( 99, 101 ) );
}
#endif
BaseClass::Activate();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::UpdateOnRemove( void )
{
BaseClass::UpdateOnRemove();
if ( m_pController )
{
UTIL_Remove( m_pController );
}
CPropWeightedCube *pTwin = m_hSchrodingerTwin.Get();
if ( pTwin && !pTwin->IsMarkedForDeletion() )
{
CTriggerPortalCleanser::FizzleBaseAnimating( NULL, pTwin );
}
#if 0
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundDestroy( m_pSchrodingerSound );
m_pSchrodingerSound = NULL;
BaseClass::StopLoopingSounds();
#endif
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::Precache( void )
{
ConvertOldSkins();
switch ( m_nCubeType )
{
default:
case CUBE_STANDARD:
case CUBE_COMPANION:
PrecacheModel( CUBE_MODEL );
break;
case CUBE_REFLECTIVE:
PrecacheModel( CUBE_REFLECT_MODEL );
break;
case CUBE_SPHERE:
PrecacheModel( CUBE_SPHERE_MODEL );
break;
case CUBE_ANTIQUE:
PrecacheModel( CUBE_ANTIQUE_MODEL );
case CUBE_SCHRODINGER:
PrecacheModel( CUBE_SCHRODINGER_MODEL );
PrecacheScriptSound( "music.laser_node_02.play" );
PrecacheScriptSound( "prop_laser_catcher.poweron" );
PrecacheScriptSound( "prop_laser_catcher.poweroff" );
break;
}
PrecacheModel( CUBE_FX_FIZZLER_MODEL );
PrecacheScriptSound( "WeightedCube.JumpPowerActivateShort" );
PrecacheScriptSound( "WeightedCube.JumpPowerActivateLong" );
BaseClass::Precache();
}
int CPropWeightedCube::ObjectCaps( void )
{
int flags = (BaseClass::ObjectCaps()|FCAP_IMPULSE_USE);
if ( GetPaintedPower() == BOUNCE_POWER )
{
flags |= FCAP_USE_IN_RADIUS;
}
return flags;
}
int CPropWeightedCube::UpdateTransmitState()
{
if ( HasLaser() )
{
return SetTransmitState( FL_EDICT_ALWAYS );
}
return BaseClass::UpdateTransmitState();
}
void CPropWeightedCube::ConvertOldSkins( void )
{
//HACK HACK: Make the cubes choose skins using the new method even though the maps have not been updated to use them.
if( !m_bNewSkins )
{
if( m_nSkin > 1 )
{
m_nSkin--;
}
m_nCubeType = static_cast<WeightedCubeType_e>( m_nSkin.Get() );
m_bNewSkins = true;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::SetCubeType( void )
{
// FIXME: Remove for DLC2
if ( m_nCubeType == CUBE_SCHRODINGER )
{
m_nCubeType = CUBE_REFLECTIVE;
}
switch( m_nCubeType )
{
//Standard cube
case CUBE_STANDARD:
case CUBE_COMPANION:
{
SetModelName( MAKE_STRING( CUBE_MODEL ) );
break;
}
//Reflective cube
case CUBE_REFLECTIVE:
{
SetModelName( MAKE_STRING( CUBE_REFLECT_MODEL ) );
m_pController = CCubeRotationController::CreateRotationController( this );
AddSpawnFlags( SF_PHYSPROP_ENABLE_ON_PHYSCANNON );
break;
}
//Sphere
case CUBE_SPHERE:
{
SetModelName( MAKE_STRING( CUBE_SPHERE_MODEL ) );
break;
}
//Antique cube
case CUBE_ANTIQUE:
{
SetModelName( MAKE_STRING( CUBE_ANTIQUE_MODEL ) );
break;
}
//Schrodinger cube
case CUBE_SCHRODINGER:
{
SetModelName( MAKE_STRING( CUBE_SCHRODINGER_MODEL ) );
m_pController = CCubeRotationController::CreateRotationController( this );
AddSpawnFlags( SF_PHYSPROP_ENABLE_ON_PHYSCANNON );
if ( m_hSchrodingerDangling.Get() == NULL )
{
m_hSchrodingerDangling = this;
}
else
{
m_hSchrodingerDangling->m_hSchrodingerTwin = this;
m_hSchrodingerTwin = m_hSchrodingerDangling;
m_hSchrodingerDangling = NULL;
}
break;
}
}
SetCubeSkin();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::SetActivated( bool bActivate )
{
m_bActivated = bActivate;
SetCubeSkin();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::SetCubeSkin( void )
{
switch( m_nCubeType )
{
//Standard cube
case CUBE_STANDARD:
{
//Rusted cubes don't show paint
if( m_bRusted )
{
if( m_bActivated )
{
SetSkin( CUBE_STANDARD_RUSTED_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_STANDARD_RUSTED_SKIN );
}
}
else
{
switch( GetPaintedPower() )
{
//Bounce painted
case BOUNCE_POWER:
{
if( m_bActivated )
{
RANDOM_CEG_TEST_SECRET_PERIOD( 98, 106 );
SetSkin( CUBE_STANDARD_BOUNCE_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_STANDARD_BOUNCE_SKIN );
}
}
break;
//Speed painted
case SPEED_POWER:
{
if( m_bActivated )
{
SetSkin( CUBE_STANDARD_SPEED_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_STANDARD_SPEED_SKIN );
}
}
break;
//Not painted
default:
{
if( m_bActivated )
{
SetSkin( CUBE_STANDARD_CLEAN_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_STANDARD_CLEAN_SKIN );
}
}
break;
}
}
}
break;
//Companion cube
case CUBE_COMPANION:
{
switch( GetPaintedPower() )
{
//Bounce painted
case BOUNCE_POWER:
{
if( m_bActivated )
{
SetSkin( CUBE_COMPANION_BOUNCE_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_COMPANION_BOUNCE_SKIN );
}
}
break;
//Speed painted
case SPEED_POWER:
{
if( m_bActivated )
{
SetSkin( CUBE_COMPANION_SPEED_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_COMPANION_SPEED_SKIN );
}
}
break;
//Not painted
default:
{
if( m_bActivated )
{
SetSkin( CUBE_COMPANION_CLEAN_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_COMPANION_CLEAN_SKIN );
}
}
break;
}
}
break;
//Reflective cube
case CUBE_REFLECTIVE:
{
switch( GetPaintedPower() )
{
//Bounce painted
case BOUNCE_POWER:
{
if( m_bRusted )
{
// FIXME
SetSkin( CUBE_REFLECTIVE_BOUNCE_SKIN );
}
else
{
SetSkin( CUBE_REFLECTIVE_BOUNCE_SKIN );
}
}
break;
//Speed painted
case SPEED_POWER:
{
if( m_bRusted )
{
// FIXME
SetSkin( CUBE_REFLECTIVE_SPEED_SKIN );
}
else
{
SetSkin( CUBE_REFLECTIVE_SPEED_SKIN );
}
}
break;
//Not painted
default:
{
if( m_bRusted )
{
SetSkin( CUBE_REFLECTIVE_RUSTED_SKIN );
}
else
{
SetSkin( CUBE_REFLECTIVE_CLEAN_SKIN );
}
}
break;
}
}
break;
//Sphere
case CUBE_SPHERE:
{
switch( GetPaintedPower() )
{
//Bounce painted
case BOUNCE_POWER:
{
if( m_bActivated )
{
SetSkin( CUBE_SPHERE_BOUNCE_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_SPHERE_BOUNCE_SKIN );
}
}
break;
//Speed painted
case SPEED_POWER:
{
if( m_bActivated )
{
SetSkin( CUBE_SPHERE_SPEED_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_SPHERE_SPEED_SKIN );
}
}
break;
//Not painted
default:
{
if( m_bActivated )
{
SetSkin( CUBE_SPHERE_CLEAN_ACTIVATED_SKIN );
}
else
{
SetSkin( CUBE_SPHERE_CLEAN_SKIN );
}
}
break;
}
}
break;
//Antique cube
case CUBE_ANTIQUE:
{
switch( GetPaintedPower() )
{
//Bounce painted
case BOUNCE_POWER:
{
SetSkin( CUBE_ANTIQUE_BOUNCE_SKIN );
}
break;
//Speed painted
case SPEED_POWER:
{
SetSkin( CUBE_ANTIQUE_SPEED_SKIN );
}
break;
//Not painted
default:
{
SetSkin( CUBE_ANTIQUE_CLEAN_SKIN );
}
break;
}
}
break;
//Antique cube
case CUBE_SCHRODINGER:
{
switch( GetPaintedPower() )
{
//Bounce painted
case BOUNCE_POWER:
{
SetSkin( CUBE_SCHRODINGER_BOUNCE_SKIN );
}
break;
//Speed painted
case SPEED_POWER:
{
SetSkin( CUBE_SCHRODINGER_SPEED_SKIN );
}
break;
//Not painted
default:
{
SetSkin( CUBE_SCHRODINGER_CLEAN_SKIN );
}
break;
}
}
break;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::SetSkin( int skinNum )
{
m_nSkin = skinNum;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::InputDissolve( inputdata_t &in )
{
CTriggerPortalCleanser::FizzleBaseAnimating( NULL, this );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::InputSilentDissolve( inputdata_t &in )
{
OnFizzled();
UTIL_Remove( this );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::InputPreDissolveJoke( inputdata_t &in )
{
CBaseEntity *pEntity = gEntList.FindEntityByName( NULL, "@glados" );
if ( pEntity )
{
pEntity->RunScript( "CoopCubeFizzle()", "PreDissolveJoke" );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::InputDisablePortalFunnel( inputdata_t &in )
{
m_bAllowPortalFunnel = false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::InputEnablePortalFunnel( inputdata_t &in )
{
m_bAllowPortalFunnel = true;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
QAngle CPropWeightedCube::CalculatePreferredAngles( CBasePlayer *pPlayer )
{
return QAngle(0,0,0);
}
void CPropWeightedCube::UpdatePreferredAngles( CBasePlayer *pPlayer )
{
m_vecCarryAngles = CalculatePreferredAngles( pPlayer );
if( HasPreferredCarryAnglesForPlayer( pPlayer ) )
{
m_qPreferredPlayerCarryAngles = m_vecCarryAngles;
}
else
{
if( m_qPreferredPlayerCarryAngles.Get().x < FLT_MAX )
{
m_qPreferredPlayerCarryAngles.GetForModify().Init( FLT_MAX, FLT_MAX, FLT_MAX );
}
}
}
extern void ComputePlayerMatrix( CBasePlayer *pPlayer, matrix3x4_t &out );
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
QAngle CPropWeightedCube::PreferredCarryAngles( void )
{
static QAngle s_prefAngles;
s_prefAngles = m_vecCarryAngles;
CBasePlayer *pPlayer = GetPlayerHoldingEntity( this );
if ( pPlayer )
{
Vector vecRight;
pPlayer->GetVectors( NULL, &vecRight, NULL );
Quaternion qRotation;
AxisAngleQuaternion( vecRight, pPlayer->EyeAngles().x, qRotation );
matrix3x4_t tmp;
ComputePlayerMatrix( pPlayer, tmp );
QAngle qTemp = TransformAnglesToWorldSpace( s_prefAngles, tmp );
Quaternion qExisting;
AngleQuaternion( qTemp, qExisting );
Quaternion qFinal;
QuaternionMult( qRotation, qExisting, qFinal );
QuaternionAngles( qFinal, qTemp );
s_prefAngles = TransformAnglesToLocalSpace( qTemp, tmp );
}
return s_prefAngles;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::OnPhysGunPickup( CBasePlayer *pPhysGunUser, PhysGunPickup_t reason )
{
BaseClass::OnPhysGunPickup( pPhysGunUser, reason );
m_bMovementDisabled = false;
m_bTouchedByPlayer = true;
// Calculate our preferred angles on the first pickup
if ( reason == PICKED_UP_BY_CANNON || reason == PICKED_UP_BY_PLAYER )
{
UpdatePreferredAngles( pPhysGunUser );
if ( m_pController )
{
m_pController->Enable( false );
}
CPortal_Player *pPlayer = ToPortalPlayer( pPhysGunUser );
if ( pPlayer )
{
// Force a cool-down on the +USE key after a successful grab
pPlayer->SetUseKeyCooldownTime( 0.5f );
}
}
if ( pPhysGunUser )
{
if ( pPhysGunUser->GetTeamNumber() == TEAM_RED )
{
m_OnOrangePickUp.FireOutput( pPhysGunUser, this );
}
else if ( pPhysGunUser->GetTeamNumber() == TEAM_BLUE )
{
m_OnBluePickUp.FireOutput( pPhysGunUser, this );
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Turn on our rotation controller when we're dropped nicely
//-----------------------------------------------------------------------------
ConVar sv_box_physgundrop_angle_threshold("sv_box_physgundrop_angle_threshold", "70.f");
void CPropWeightedCube::OnPhysGunDrop( CBasePlayer *pPhysGunUser, PhysGunDrop_t reason )
{
BaseClass::OnPhysGunDrop( pPhysGunUser, reason );
// Only care about this if we're dropped, as opposed to launched or thrown
if ( reason != DROPPED_BY_PLAYER && reason != DROPPED_BY_CANNON )
return;
// Enable the controller for a short time
if ( m_pController )
{
m_pController->Activate();
Vector vecForward;
AngleVectors( GetAbsAngles(), &vecForward );
m_pController->SetAlignmentVector( vecForward );
m_pController->SuspendAfter( gpGlobals->curtime + 0.5f );
}
// When player drop the box and player's up is not world up, try to throw the box in the local down direction
bool bThrowBoxLocalDown = false;
CPortal_Player *pPortalPlayer = ToPortalPlayer( pPhysGunUser );
if ( pPortalPlayer && !AlmostEqual( DotProduct( pPortalPlayer->GetPortalPlayerLocalData().m_Up, Vector( 0, 0, 1 ) ), 1.f ) )
{
// check if player looks too far off the ground, then just drop the box with gravity
Vector vForward;
pPortalPlayer->GetVectors( &vForward, NULL, NULL );
float flLookAngle = RAD2DEG( acosf( DotProduct( vForward, pPortalPlayer->GetPortalPlayerLocalData().m_StickNormal ) ) );
bThrowBoxLocalDown = flLookAngle >= sv_box_physgundrop_angle_threshold.GetFloat();
if( bThrowBoxLocalDown )
{
float flDropSpeed = 400.f;
Vector vecDownVelocity = -( flDropSpeed * pPortalPlayer->GetPortalPlayerLocalData().m_Up );
IPhysicsObject *pPhysics = VPhysicsGetObject();
if ( pPhysics )
{
pPhysics->SetVelocityInstantaneous( &vecDownVelocity, NULL );
}
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Only bother with preferred carry angles if we're a reflective cube
//-----------------------------------------------------------------------------
bool CPropWeightedCube::HasPreferredCarryAnglesForPlayer( CBasePlayer *pPlayer )
{
return ( m_nCubeType == CUBE_REFLECTIVE ) || ( /*FIXME: Bring back for DLC2 !sv_schrodinger_laser_world_aligned.GetBool() && */ m_nCubeType == CUBE_SCHRODINGER && m_hLaser.Get() );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::NotifySystemEvent(CBaseEntity *pNotify, notify_system_event_t eventType, const notify_system_event_params_t &params )
{
// On teleport, we record a pointer to the portal we are arriving at
if ( eventType == NOTIFY_EVENT_TELEPORT )
{
CPortal_Base2D *pEnteredPortal = dynamic_cast<CPortal_Base2D*>( pNotify );
if ( pEnteredPortal && m_pController )
{
Vector vecWorldAign = pEnteredPortal->m_matrixThisToLinked.ApplyRotation( m_pController->GetAlignmentVector() );
vecWorldAign.NormalizeInPlace();
m_pController->SetAlignmentVector( vecWorldAign );
}
}
BaseClass::NotifySystemEvent( pNotify, eventType, params );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CPropWeightedCube::Paint( PaintPowerType paintType, const Vector &worldContactPt )
{
BaseClass::Paint( paintType, worldContactPt );
SetPaintedMaterial( paintType );
SetCubeSkin();
CPropWeightedCube *pTwin = m_hSchrodingerTwin.Get();
if ( pTwin && pTwin->GetPaintedPower() != paintType )
{
pTwin->Paint( paintType, worldContactPt );
}
}
void CPropWeightedCube::SetPaintedMaterial( PaintPowerType paintType )
{
if ( m_nCurrentPaintedType != paintType && paintType != NO_POWER )
{
m_OnPainted.FireOutput( this, this );
}
m_nCurrentPaintedType = paintType;
switch( paintType )
{
case BOUNCE_POWER:
{
//Set the box to be bouncy
IPhysicsObject* pPhysObject = VPhysicsGetObject();
if( pPhysObject )
{
pPhysObject->SetMaterialIndex( m_nBouncyMaterialIndex );
}
ExitDisabledState();
break;
}
case SPEED_POWER:
{
IPhysicsObject* pPhysObject = VPhysicsGetObject();
if( pPhysObject )
{
pPhysObject->SetMaterialIndex( BaseClass::GetSpeedMaterialIndex() );
}
break;
}
case PORTAL_POWER:
case REFLECT_POWER:
case NO_POWER:
default:
{
// Store our material index
IPhysicsObject* pPhysObject = VPhysicsGetObject();
if( pPhysObject )
{
pPhysObject->SetMaterialIndex( m_nOriginalMaterialIndex );
}
break;
}
}
}
CPropWeightedCube* CPropWeightedCube::GetSchrodingerTwin( void )
{
return m_hSchrodingerTwin;
}
void CPropWeightedCube::UpdateSchrodingerSound( void )
{
if ( !m_hSchrodingerTwin.Get() )
return;
float fDist = m_hSchrodingerTwin->GetDistanceToEntity( this );
if ( m_pSchrodingerSound )
{
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
controller.SoundChangeVolume( m_pSchrodingerSound, RemapValClamped( fDist, 350.0f, 0.0f, 0.0f, 1.0f ), 0.1 );
}
}
void CPropWeightedCube::SetLaser( CBaseEntity *pLaser )
{
m_hLaser = pLaser;
if ( pLaser )
{
CBasePlayer *pPlayer = GetPlayerHoldingEntity( this );
if ( pPlayer )
{
UpdatePreferredAngles( pPlayer );
}
if ( GetCubeType() == CUBE_SCHRODINGER )
{
// FIXME: Need a better sound for this
//EmitSound( "prop_laser_catcher.poweron" );
}
}
else
{
if ( GetCubeType() == CUBE_SCHRODINGER )
{
// FIXME: Need a better sound for this
//EmitSound( "prop_laser_catcher.poweroff" );
}
}
// need to update transmitstate to prevent laser going through box when box goes outside PVS
UpdateTransmitState();
}
bool CPropWeightedCube::ShouldEnterDisabledState( void )
{
IPhysicsObject *pPhysicsObject = VPhysicsGetObject();
if( pPhysicsObject )
{
if( !( pPhysicsObject->GetGameFlags() & FVPHYSICS_PLAYER_HELD ) && pPhysicsObject->IsAsleep() )
{
return true;
}
}
return false;
}
void CPropWeightedCube::EnterDisabledState( void )
{
if ( !m_bMovementDisabled )
{
IPhysicsObject *pPhysicsObject = VPhysicsGetObject();
if( pPhysicsObject )
{
pPhysicsObject->EnableMotion( false );
}
m_bMovementDisabled = true;
SetThink( &CPropWeightedCube::DisabledThink );
SetNextThink( gpGlobals->curtime + reflector_cube_disabled_think_rate.GetFloat() );
}
}
void CPropWeightedCube::ExitDisabledState( void )
{
if ( m_bMovementDisabled )
{
m_bMovementDisabled = false;
EnableMotion();
}
}
void CPropWeightedCube::InputExitDisabledState( inputdata_t &in )
{
ExitDisabledState();
}
void CPropWeightedCube::OnEnteredTractorBeam( void )
{
SetThink( &CPropWeightedCube::TractorBeamThink );
SetNextThink( gpGlobals->curtime );
}
void CPropWeightedCube::OnExitedTractorBeam( void )
{
SetThink( &CPropWeightedCube::ExitTractorBeamThink );
SetNextThink( gpGlobals->curtime );
}
void CPropWeightedCube::TractorBeamThink( void )
{
if ( m_bMovementDisabled )
return;
// Stop colliding with player and freeze any rotational speed
SetCollisionGroup( COLLISION_GROUP_PLAYER_HELD );
IPhysicsObject *pPhys = VPhysicsGetObject();
if ( pPhys )
{
AngularImpulse vZeroRotation( vec3_origin );
pPhys->SetVelocity( NULL, &vZeroRotation );
}
// Give players 2 seconds to get out of the way
SetThink( &CPropWeightedCube::ExitTractorBeamThink );
SetNextThink( gpGlobals->curtime + 2.0f );
}
void CPropWeightedCube::ExitTractorBeamThink( void )
{
bool bIntersectingPlayer = false;
for( int i = 1; i <= gpGlobals->maxClients; ++i )
{
CBasePlayer *pPlayer = UTIL_PlayerByIndex( i );
if ( pPlayer )
{
if ( Intersects( pPlayer ) )
{
bIntersectingPlayer = true;
break;
}
}
}
if ( bIntersectingPlayer )
{
SetThink( &CPropWeightedCube::ExitTractorBeamThink );
SetNextThink( gpGlobals->curtime + 0.2f );
}
else
{
// Start colliding with player
SetCollisionGroup( COLLISION_GROUP_WEIGHTED_CUBE );
}
}
void CPropWeightedCube::InputSetPaint( inputdata_t &in )
{
Paint( static_cast< PaintPowerType >( in.value.Int() ), Vector( 0.0f, 0.0f, 0.0f ) );
}
void CPropWeightedCube::StartTouch( CBaseEntity *pOther )
{
if( m_bMovementDisabled )
{
if( pOther->IsPlayer() )
{
Vector vecPlayerForward;
AngleVectors( pOther->EyeAngles(), &vecPlayerForward );
vecPlayerForward.NormalizeInPlace();
Vector vecCubeToPlayer = (GetAbsOrigin() - pOther->EyePosition()).Normalized();
float flPlayerLookDot = DotProduct( vecCubeToPlayer, vecPlayerForward );
float flCubeDirDot = DotProduct( Forward().Normalized(), vecPlayerForward );
//DevMsg( "Dot:%f, CubeDot:%f\n", flPlayerLookDot, flCubeDirDot );
//If the cube is in front of the player
if( ( flPlayerLookDot > 0.8f && flCubeDirDot > 0.8f ) || ( flPlayerLookDot > 0.85f ) )
{
ExitDisabledState();
}
}
}
if( pOther->IsPlayer() )
{
m_bTouchedByPlayer = true;
}
BaseClass::StartTouch( pOther );
}
void CPropWeightedCube::SchrodingerThink( void )
{
UpdateSchrodingerSound();
//Keep thinking
SetContextThink( &CPropWeightedCube::SchrodingerThink, gpGlobals->curtime + reflector_cube_disabled_think_rate.GetFloat(), SCHRODINGER_THINK_CONTEXT );
}
void CPropWeightedCube::DisabledThink( void )
{
bool hasPaintPower = false;
if( engine->HasPaintmap() )
{
if( GetPaintedPower() != NO_POWER )
{
hasPaintPower = true;
}
else
{
for( int i = 0; i < PAINT_POWER_TYPE_COUNT; ++i )
{
if( !IsInactivePower( GetPaintPower(i) ) )
{
hasPaintPower = true;
break;
}
}
}
}
//If the cube no longer has a laser attached to it or has a paint power
if( !HasLaser() || hasPaintPower )
{
ExitDisabledState();
return;
}
//Keep thinking
SetNextThink( gpGlobals->curtime + reflector_cube_disabled_think_rate.GetFloat() );
}
void CPropWeightedCube::InputDisablePickup( inputdata_t &in )
{
m_bPickupDisabled = true;
}
void CPropWeightedCube::InputEnablePickup( inputdata_t &in )
{
m_bPickupDisabled = false;
}
void CPropWeightedCube::Use( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
{
if( m_bPickupDisabled == false )
{
BaseClass::Use( pActivator, pCaller, useType, value );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool UTIL_IsWeightedCube( CBaseEntity *pEntity )
{
if ( pEntity == NULL )
return false;
return ( FClassnameIs( pEntity, "prop_weighted_cube" ) );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool UTIL_IsReflectiveCube( CBaseEntity *pEntity )
{
if ( UTIL_IsWeightedCube( pEntity ) == false )
return false;
CPropWeightedCube *pCube = assert_cast<CPropWeightedCube*>( pEntity );
return ( pCube && pCube->GetCubeType() == CUBE_REFLECTIVE );
}
#ifndef CLIENT_DLL
bool UTIL_IsSchrodinger( CBaseEntity *pEntity )
{
if ( !UTIL_IsWeightedCube( pEntity ) )
return false;
CPropWeightedCube *pCube = assert_cast<CPropWeightedCube*>( pEntity );
if ( !pCube )
return false;
return pCube->GetCubeType() == CUBE_SCHRODINGER;
}
CPropWeightedCube* UTIL_GetSchrodingerTwin( CBaseEntity *pEntity )
{
if ( !UTIL_IsSchrodinger( pEntity ) )
return NULL;
CPropWeightedCube *pCube = assert_cast<CPropWeightedCube*>( pEntity );
if ( !pCube )
return NULL;
return pCube->GetSchrodingerTwin();
}
#endif
#define PORTAL_REFLECTOR_CUBE_MODEL_NAME "models/props/reflectocube.mdl"
#define PORTAL_WEIGHT_BOX_MODEL_NAME "models/props/metal_box.mdl"
#ifndef CLIENT_DLL
//-----------------------------------------------------------------------------
// Creates a weighted cube of a specific type
//-----------------------------------------------------------------------------
void CPropWeightedCube::CreatePortalWeightedCube( WeightedCubeType_e objectType, bool bAtCursorPosition, const Vector &position )
{
MDLCACHE_CRITICAL_SECTION();
bool allowPrecache = CBaseEntity::IsPrecacheAllowed();
CBaseEntity::SetAllowPrecache( true );
// Try to create entity
CPropWeightedCube *entity = ( CPropWeightedCube* )CreateEntityByName("prop_weighted_cube");
if (entity)
{
//entity->PrecacheModel( PORTAL_REFLECTOR_CUBE_MODEL_NAME );
//entity->SetModel( PORTAL_REFLECTOR_CUBE_MODEL_NAME );
entity->SetName( MAKE_STRING("cube") );
entity->AddSpawnFlags( SF_PHYSPROP_ENABLE_PICKUP_OUTPUT );
entity->m_nCubeType = objectType;
entity->m_bNewSkins = true;
entity->Precache();
if ( !bAtCursorPosition )
{
entity->SetAbsOrigin( position );
}
DispatchSpawn(entity);
if ( bAtCursorPosition )
{
// Now attempt to drop into the world
CBasePlayer* pPlayer = UTIL_GetCommandClient();
trace_t tr;
Vector forward;
pPlayer->EyeVectors( &forward );
UTIL_TraceLine(pPlayer->EyePosition(),
pPlayer->EyePosition() + forward * MAX_TRACE_LENGTH,MASK_SOLID,
pPlayer, COLLISION_GROUP_WEIGHTED_CUBE, &tr );
if ( tr.fraction != 1.0 )
{
tr.endpos.z += 12;
entity->Teleport( &tr.endpos, NULL, NULL );
UTIL_DropToFloor( entity, MASK_SOLID );
}
}
// This entity should send its object caps to the client
entity->UpdateObjectCapsCache();
}
CBaseEntity::SetAllowPrecache( allowPrecache );
}
// Console command functions
void CC_Create_PortalWeightedCube()
{
CPropWeightedCube::CreatePortalWeightedCube( CUBE_STANDARD );
}
void CC_Create_PortalCompanionCube()
{
CPropWeightedCube::CreatePortalWeightedCube( CUBE_COMPANION );
}
void CC_Create_PortalReflectorCube()
{
CPropWeightedCube::CreatePortalWeightedCube( CUBE_REFLECTIVE );
}
void CC_Create_PortalWeightedSphere()
{
CPropWeightedCube::CreatePortalWeightedCube( CUBE_SPHERE );
}
void CC_Create_PortalWeightedAntique()
{
CPropWeightedCube::CreatePortalWeightedCube( CUBE_ANTIQUE );
}
void CC_Create_PortalWeightedSchrodinger()
{
CPropWeightedCube::CreatePortalWeightedCube( CUBE_SCHRODINGER );
}
// Console commands for creating cubes
static ConCommand ent_create_portal_reflector_cube("ent_create_portal_reflector_cube", CC_Create_PortalReflectorCube, "Creates a laser reflector cube cube where the player is looking.", FCVAR_GAMEDLL | FCVAR_CHEAT);
static ConCommand ent_create_portal_companion_cube("ent_create_portal_companion_cube", CC_Create_PortalCompanionCube, "Creates a companion cube where the player is looking.", FCVAR_GAMEDLL | FCVAR_CHEAT);
static ConCommand ent_create_portal_weighted_cube("ent_create_portal_weighted_cube", CC_Create_PortalWeightedCube, "Creates a standard cube where the player is looking.", FCVAR_GAMEDLL | FCVAR_CHEAT);
static ConCommand ent_create_portal_weighted_sphere("ent_create_portal_weighted_sphere", CC_Create_PortalWeightedSphere, "Creates a weighted sphere where the player is looking.", FCVAR_GAMEDLL | FCVAR_CHEAT);
static ConCommand ent_create_portal_weighted_antique("ent_create_portal_weighted_antique", CC_Create_PortalWeightedAntique, "Creates an antique cube where the player is looking.", FCVAR_GAMEDLL | FCVAR_CHEAT);
// FIXME: Bring this back for DLC2
//static ConCommand ent_create_portal_weighted_schrodinger("ent_create_portal_weighted_schrodinger", CC_Create_PortalWeightedSchrodinger, "Creates an Schrodinger cube where the player is looking.", FCVAR_GAMEDLL | FCVAR_CHEAT);
#endif // CLIENT_DLL