//========= Copyright Valve Corporation, All rights reserved. ============// // // Purpose: // // $NoKeywords: $ // //=============================================================================// #ifdef CLIENT_DLL #include "cbase.h" #endif #include "shareddefs.h" #if !defined( USERCMD_H ) #define USERCMD_H #ifdef _WIN32 #pragma once #endif #include "mathlib/vector.h" #include "utlvector.h" #include "imovehelper.h" #include "checksum_crc.h" #ifndef CLIENT_DLL #include "baseanimating.h" #include "BaseAnimatingOverlay.h" #else #include "c_baseanimating.h" #include "c_baseanimatingoverlay.h" #endif #define MAX_LAYER_RECORDS (CBaseAnimatingOverlay::MAX_OVERLAYS) #define MAX_POSE_PARAMETERS (CBaseAnimating::NUM_POSEPAREMETERS) #define MAX_ENCODED_CONTROLLERS (MAXSTUDIOBONECTRLS) class bf_read; class bf_write; struct LayerRecord { int m_sequence; float m_cycle; float m_weight; int m_order; LayerRecord() { m_sequence = 0; m_cycle = 0; m_weight = 0; m_order = 0; } }; struct ClientSideAnimationData { float m_flAnimTime; // Player animation details, so we can get the legs in the right spot. LayerRecord m_layerRecords[MAX_LAYER_RECORDS]; int m_masterSequence; float m_masterCycle; float m_poseParameters[MAX_POSE_PARAMETERS]; float m_encodedControllers[MAX_ENCODED_CONTROLLERS]; }; class CEntityGroundContact { public: int entindex; float minheight; float maxheight; }; class CUserCmd { public: CUserCmd() { Reset(); } virtual ~CUserCmd() { }; void Reset() { command_number = 0; tick_count = 0; viewangles.Init(); forwardmove = 0.0f; sidemove = 0.0f; upmove = 0.0f; buttons = 0; impulse = 0; weaponselect = 0; weaponsubtype = 0; random_seed = 0; mousedx = 0; mousedy = 0; hasbeenpredicted = false; for (int i = 0; i < MAX_EDICTS; i++) { simulationtimes[i] = 0.0f; has_simulation[i] = false; has_animation[i] = false; } for (int i = 0; i <= MAX_PLAYERS; i++) { animationdata[i] = {}; } #if defined( HL2_DLL ) || defined( HL2_CLIENT_DLL ) entitygroundcontact.RemoveAll(); #endif } CUserCmd& operator =( const CUserCmd& src ) { if ( this == &src ) return *this; command_number = src.command_number; tick_count = src.tick_count; viewangles = src.viewangles; forwardmove = src.forwardmove; sidemove = src.sidemove; upmove = src.upmove; buttons = src.buttons; impulse = src.impulse; weaponselect = src.weaponselect; weaponsubtype = src.weaponsubtype; random_seed = src.random_seed; mousedx = src.mousedx; mousedy = src.mousedy; hasbeenpredicted = src.hasbeenpredicted; for (int i = 0; i < MAX_EDICTS; i++) { simulationtimes[i] = src.simulationtimes[i]; has_simulation[i] = src.has_simulation[i]; has_animation[i] = src.has_animation[i]; } for (int i = 0; i <= MAX_PLAYERS; i++) { animationdata[i] = src.animationdata[i]; } #if defined( HL2_DLL ) || defined( HL2_CLIENT_DLL ) entitygroundcontact = src.entitygroundcontact; #endif return *this; } CUserCmd( const CUserCmd& src ) { *this = src; } CRC32_t GetChecksum( void ) const { CRC32_t crc; CRC32_Init( &crc ); CRC32_ProcessBuffer( &crc, &command_number, sizeof( command_number ) ); CRC32_ProcessBuffer( &crc, &tick_count, sizeof( tick_count ) ); CRC32_ProcessBuffer( &crc, &viewangles, sizeof( viewangles ) ); CRC32_ProcessBuffer( &crc, &forwardmove, sizeof( forwardmove ) ); CRC32_ProcessBuffer( &crc, &sidemove, sizeof( sidemove ) ); CRC32_ProcessBuffer( &crc, &upmove, sizeof( upmove ) ); CRC32_ProcessBuffer( &crc, &buttons, sizeof( buttons ) ); CRC32_ProcessBuffer( &crc, &impulse, sizeof( impulse ) ); CRC32_ProcessBuffer( &crc, &weaponselect, sizeof( weaponselect ) ); CRC32_ProcessBuffer( &crc, &weaponsubtype, sizeof( weaponsubtype ) ); CRC32_ProcessBuffer( &crc, &random_seed, sizeof( random_seed ) ); CRC32_ProcessBuffer( &crc, &mousedx, sizeof( mousedx ) ); CRC32_ProcessBuffer(&crc, &mousedy, sizeof(mousedy)); CRC32_ProcessBuffer(&crc, has_simulation, sizeof(has_simulation)); CRC32_ProcessBuffer(&crc, has_animation, sizeof(has_animation)); CRC32_ProcessBuffer( &crc, simulationtimes, sizeof( simulationtimes ) ); CRC32_ProcessBuffer(&crc, animationdata, sizeof(animationdata)); CRC32_Final( &crc ); return crc; } // Allow command, but negate gameplay-affecting values void MakeInert( void ) { viewangles = vec3_angle; forwardmove = 0.f; sidemove = 0.f; upmove = 0.f; buttons = 0; impulse = 0; } // For matching server and client commands for debugging int command_number; // the tick the client created this command int tick_count; // Player instantaneous view angles. QAngle viewangles; // Intended velocities // forward velocity. float forwardmove; // sideways velocity. float sidemove; // upward velocity. float upmove; // Attack button states int buttons; // Impulse command issued. byte impulse; // Current weapon id int weaponselect; int weaponsubtype; int random_seed; // For shared random functions short mousedx; // mouse accum in x from create move short mousedy; // mouse accum in y from create move // Client only, tracks whether we've predicted this command at least once bool hasbeenpredicted; // TODO_ENHANCED: Lag compensate also other entities when needed. // Send simulation times for each players for lag compensation. bool has_simulation[MAX_EDICTS]; bool has_animation[MAX_EDICTS]; float simulationtimes[MAX_EDICTS]; ClientSideAnimationData animationdata[MAX_PLAYERS+1]; // Back channel to communicate IK state #if defined( HL2_DLL ) || defined( HL2_CLIENT_DLL ) CUtlVector< CEntityGroundContact > entitygroundcontact; #endif }; void ReadUsercmd( bf_read *buf, CUserCmd *move, CUserCmd *from ); void WriteUsercmd( bf_write *buf, const CUserCmd *to, const CUserCmd *from ); #endif // USERCMD_H