2020-12-21 16:56:44 -08:00
|
|
|
/*
|
2021-02-27 17:31:27 -08:00
|
|
|
* Copyright (c) 2016-2021 Marco Hladik <marco@icculus.org>
|
2020-12-21 16:56:44 -08:00
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2021-02-27 17:31:27 -08:00
|
|
|
/* all potential SendFlags bits we can possibly send */
|
|
|
|
enumflags
|
|
|
|
{
|
|
|
|
PLAYER_KEEPALIVE,
|
|
|
|
PLAYER_MODELINDEX,
|
|
|
|
PLAYER_ORIGIN,
|
|
|
|
PLAYER_ORIGIN_Z,
|
|
|
|
PLAYER_ANGLES_X,
|
|
|
|
PLAYER_ANGLES_Y,
|
|
|
|
PLAYER_ANGLES_Z,
|
|
|
|
PLAYER_VELOCITY,
|
|
|
|
PLAYER_VELOCITY_Z,
|
|
|
|
PLAYER_FLAGS,
|
|
|
|
PLAYER_WEAPON,
|
|
|
|
PLAYER_ITEMS,
|
|
|
|
PLAYER_HEALTH,
|
|
|
|
PLAYER_ARMOR,
|
|
|
|
PLAYER_MOVETYPE,
|
|
|
|
PLAYER_VIEWOFS,
|
|
|
|
PLAYER_BASEFRAME,
|
|
|
|
PLAYER_FRAME,
|
|
|
|
PLAYER_AMMO1,
|
|
|
|
PLAYER_AMMO2,
|
|
|
|
PLAYER_AMMO3,
|
|
|
|
PLAYER_UNUSED1,
|
|
|
|
PLAYER_UNUSED2
|
|
|
|
};
|
|
|
|
|
2020-12-21 16:56:44 -08:00
|
|
|
noref int input_sequence;
|
|
|
|
class player:base_player
|
|
|
|
{
|
|
|
|
#ifdef CLIENT
|
|
|
|
/* External model */
|
|
|
|
entity p_model;
|
|
|
|
int p_hand_bone;
|
|
|
|
int p_model_bone;
|
|
|
|
float lastweapon;
|
|
|
|
|
|
|
|
virtual void(void) draw;
|
|
|
|
virtual float() predraw;
|
|
|
|
virtual void(void) postdraw;
|
2021-02-27 17:31:27 -08:00
|
|
|
virtual void(float) ReceiveEntity;
|
|
|
|
virtual void(void) PredictPreFrame;
|
|
|
|
virtual void(void) PredictPostFrame;
|
2020-12-21 16:56:44 -08:00
|
|
|
#else
|
2021-02-27 17:31:27 -08:00
|
|
|
virtual void(void) EvaluateEntity;
|
|
|
|
virtual float(entity, float) SendEntity;
|
2020-12-21 16:56:44 -08:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2021-02-27 17:31:27 -08:00
|
|
|
#ifdef CLIENT
|
|
|
|
//void Weapons_AmmoUpdate(entity);
|
|
|
|
/*
|
|
|
|
=================
|
|
|
|
player::ReceiveEntity
|
|
|
|
=================
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
player::ReceiveEntity(float new)
|
|
|
|
{
|
|
|
|
float fl;
|
|
|
|
|
|
|
|
/* seed for our prediction table */
|
|
|
|
sequence = servercommandframe;
|
|
|
|
|
|
|
|
fl = readfloat();
|
|
|
|
|
|
|
|
/* HACK: we need to make this more reliable */
|
|
|
|
if (fl == UPDATE_ALL) {
|
|
|
|
/* we respawned */
|
|
|
|
gravity = __NULL__;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fl & PLAYER_MODELINDEX)
|
|
|
|
modelindex = readshort();
|
|
|
|
|
|
|
|
if (fl & PLAYER_ORIGIN) {
|
|
|
|
origin[0] = readcoord();
|
|
|
|
origin[1] = readcoord();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fl & PLAYER_ORIGIN_Z)
|
|
|
|
origin[2] = readcoord();
|
|
|
|
if (fl & PLAYER_ANGLES_X)
|
|
|
|
pitch = readfloat();
|
|
|
|
if (fl & PLAYER_ANGLES_Y)
|
|
|
|
angles[1] = readfloat();
|
|
|
|
if (fl & PLAYER_ANGLES_Z)
|
|
|
|
angles[2] = readfloat();
|
|
|
|
|
|
|
|
if (fl & PLAYER_VELOCITY) {
|
|
|
|
velocity[0] = readcoord();
|
|
|
|
velocity[1] = readcoord();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fl & PLAYER_VELOCITY_Z)
|
|
|
|
velocity[2] = readcoord();
|
|
|
|
if (fl & PLAYER_FLAGS) {
|
|
|
|
flags = readfloat();
|
|
|
|
gflags = readfloat();
|
|
|
|
}
|
|
|
|
if (fl & PLAYER_WEAPON)
|
|
|
|
activeweapon = readbyte();
|
|
|
|
if (fl & PLAYER_ITEMS)
|
|
|
|
g_items = (__variant)readfloat();
|
|
|
|
if (fl & PLAYER_HEALTH)
|
|
|
|
health = readbyte();
|
|
|
|
if (fl & PLAYER_ARMOR)
|
|
|
|
armor = readbyte();
|
|
|
|
if (fl & PLAYER_MOVETYPE)
|
|
|
|
movetype = readbyte();
|
|
|
|
if (fl & PLAYER_VIEWOFS)
|
|
|
|
view_ofs[2] = readfloat();
|
|
|
|
if (fl & PLAYER_BASEFRAME)
|
|
|
|
baseframe = readbyte();
|
|
|
|
if (fl & PLAYER_FRAME) {
|
|
|
|
frame = readbyte();
|
|
|
|
frame1time = 0.0f;
|
|
|
|
frame2time = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fl & PLAYER_AMMO1) {
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fl & PLAYER_AMMO2) {
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fl & PLAYER_AMMO3) {
|
|
|
|
}
|
|
|
|
|
|
|
|
//if (fl & PLAYER_AMMO1 || fl & PLAYER_AMMO2 || fl & PLAYER_AMMO3)
|
|
|
|
// Weapons_AmmoUpdate(this);
|
|
|
|
|
|
|
|
setorigin(this, origin);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
=================
|
|
|
|
player::PredictPostFrame
|
|
|
|
|
|
|
|
Save the last valid server values away in the _net variants of each field
|
|
|
|
so we can roll them back later.
|
|
|
|
=================
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
player::PredictPreFrame(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
=================
|
|
|
|
player::PredictPostFrame
|
|
|
|
|
|
|
|
Where we roll back our values to the ones last sent/verified by the server.
|
|
|
|
=================
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
player::PredictPostFrame(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
void
|
|
|
|
player::EvaluateEntity(void)
|
|
|
|
{
|
|
|
|
SendFlags |= PLAYER_KEEPALIVE;
|
|
|
|
|
|
|
|
if (old_modelindex != modelindex)
|
|
|
|
SendFlags |= PLAYER_MODELINDEX;
|
|
|
|
|
|
|
|
if (old_origin[0] != origin[0])
|
|
|
|
SendFlags |= PLAYER_ORIGIN;
|
|
|
|
|
|
|
|
if (old_origin[1] != origin[1])
|
|
|
|
SendFlags |= PLAYER_ORIGIN;
|
|
|
|
|
|
|
|
if (old_origin[2] != origin[2])
|
|
|
|
SendFlags |= PLAYER_ORIGIN_Z;
|
|
|
|
|
|
|
|
if (old_angles[0] != v_angle[0])
|
|
|
|
SendFlags |= PLAYER_ANGLES_X;
|
|
|
|
|
|
|
|
if (old_angles[1] != angles[1])
|
|
|
|
SendFlags |= PLAYER_ANGLES_Y;
|
|
|
|
|
|
|
|
if (old_angles[2] != angles[2])
|
|
|
|
SendFlags |= PLAYER_ANGLES_Z;
|
|
|
|
|
|
|
|
if (old_velocity[0] != velocity[0])
|
|
|
|
SendFlags |= PLAYER_VELOCITY;
|
|
|
|
|
|
|
|
if (old_velocity[1] != velocity[1])
|
|
|
|
SendFlags |= PLAYER_VELOCITY;
|
|
|
|
|
|
|
|
if (old_velocity[2] != velocity[2])
|
|
|
|
SendFlags |= PLAYER_VELOCITY_Z;
|
|
|
|
|
|
|
|
if (old_flags != flags)
|
|
|
|
SendFlags |= PLAYER_FLAGS;
|
|
|
|
|
|
|
|
if (old_gflags != gflags)
|
|
|
|
SendFlags |= PLAYER_FLAGS;
|
|
|
|
|
|
|
|
if (old_activeweapon != activeweapon)
|
|
|
|
SendFlags |= PLAYER_WEAPON;
|
|
|
|
|
|
|
|
if (old_items != g_items)
|
|
|
|
SendFlags |= PLAYER_ITEMS;
|
|
|
|
|
|
|
|
if (old_health != health)
|
|
|
|
SendFlags |= PLAYER_HEALTH;
|
|
|
|
|
|
|
|
if (old_armor != armor)
|
|
|
|
SendFlags |= PLAYER_ARMOR;
|
|
|
|
|
|
|
|
if (old_movetype != movetype)
|
|
|
|
SendFlags |= PLAYER_MOVETYPE;
|
|
|
|
|
|
|
|
if (old_viewofs != view_ofs[2])
|
|
|
|
SendFlags |= PLAYER_VIEWOFS;
|
|
|
|
|
|
|
|
if (old_baseframe != baseframe)
|
|
|
|
SendFlags |= PLAYER_BASEFRAME;
|
|
|
|
|
|
|
|
if (old_frame != frame)
|
|
|
|
SendFlags |= PLAYER_FRAME;
|
|
|
|
|
|
|
|
old_modelindex = modelindex;
|
|
|
|
old_origin = origin;
|
|
|
|
old_angles = angles;
|
|
|
|
old_angles[0] = v_angle[0];
|
|
|
|
old_velocity = velocity;
|
|
|
|
old_flags = flags;
|
|
|
|
old_gflags = gflags;
|
|
|
|
old_activeweapon = activeweapon;
|
|
|
|
old_items = g_items;
|
|
|
|
old_health = health;
|
|
|
|
old_armor = armor;
|
|
|
|
old_movetype = movetype;
|
|
|
|
old_viewofs = view_ofs[2];
|
|
|
|
old_baseframe = baseframe;
|
|
|
|
old_frame = frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
=================
|
|
|
|
player::SendEntity
|
|
|
|
=================
|
|
|
|
*/
|
|
|
|
float
|
|
|
|
player::SendEntity(entity ePEnt, float fChanged)
|
|
|
|
{
|
|
|
|
if (health <= 0 && ePEnt != this) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clienttype(ePEnt) != CLIENTTYPE_REAL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ePEnt != self) {
|
|
|
|
fChanged &= ~PLAYER_ITEMS;
|
|
|
|
fChanged &= ~PLAYER_HEALTH;
|
|
|
|
fChanged &= ~PLAYER_ARMOR;
|
|
|
|
fChanged &= ~PLAYER_VIEWOFS;
|
|
|
|
fChanged &= ~PLAYER_AMMO1;
|
|
|
|
fChanged &= ~PLAYER_AMMO2;
|
|
|
|
fChanged &= ~PLAYER_AMMO3;
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteByte(MSG_ENTITY, ENT_PLAYER);
|
|
|
|
WriteFloat(MSG_ENTITY, fChanged);
|
|
|
|
|
|
|
|
/* really trying to get our moneys worth with 23 bits of mantissa */
|
|
|
|
if (fChanged & PLAYER_MODELINDEX)
|
|
|
|
WriteShort(MSG_ENTITY, modelindex);
|
|
|
|
if (fChanged & PLAYER_ORIGIN) {
|
|
|
|
WriteCoord(MSG_ENTITY, origin[0]);
|
|
|
|
WriteCoord(MSG_ENTITY, origin[1]);
|
|
|
|
}
|
|
|
|
if (fChanged & PLAYER_ORIGIN_Z)
|
|
|
|
WriteCoord(MSG_ENTITY, origin[2]);
|
|
|
|
if (fChanged & PLAYER_ANGLES_X)
|
|
|
|
WriteFloat(MSG_ENTITY, v_angle[0]);
|
|
|
|
if (fChanged & PLAYER_ANGLES_Y)
|
|
|
|
WriteFloat(MSG_ENTITY, angles[1]);
|
|
|
|
if (fChanged & PLAYER_ANGLES_Z)
|
|
|
|
WriteFloat(MSG_ENTITY, angles[2]);
|
|
|
|
if (fChanged & PLAYER_VELOCITY) {
|
|
|
|
WriteCoord(MSG_ENTITY, velocity[0]);
|
|
|
|
WriteCoord(MSG_ENTITY, velocity[1]);
|
|
|
|
}
|
|
|
|
if (fChanged & PLAYER_VELOCITY_Z)
|
|
|
|
WriteCoord(MSG_ENTITY, velocity[2]);
|
|
|
|
if (fChanged & PLAYER_FLAGS) {
|
|
|
|
WriteFloat(MSG_ENTITY, flags);
|
|
|
|
WriteFloat(MSG_ENTITY, gflags);
|
|
|
|
}
|
|
|
|
if (fChanged & PLAYER_WEAPON)
|
|
|
|
WriteByte(MSG_ENTITY, activeweapon);
|
|
|
|
if (fChanged & PLAYER_ITEMS)
|
|
|
|
WriteFloat(MSG_ENTITY, (__variant)g_items);
|
|
|
|
if (fChanged & PLAYER_HEALTH)
|
|
|
|
WriteByte(MSG_ENTITY, bound(0, health, 255));
|
|
|
|
if (fChanged & PLAYER_ARMOR)
|
|
|
|
WriteByte(MSG_ENTITY, armor);
|
|
|
|
if (fChanged & PLAYER_MOVETYPE)
|
|
|
|
WriteByte(MSG_ENTITY, movetype);
|
|
|
|
if (fChanged & PLAYER_VIEWOFS)
|
|
|
|
WriteFloat(MSG_ENTITY, view_ofs[2]);
|
|
|
|
if (fChanged & PLAYER_BASEFRAME)
|
|
|
|
WriteByte(MSG_ENTITY, baseframe);
|
|
|
|
if (fChanged & PLAYER_FRAME)
|
|
|
|
WriteByte(MSG_ENTITY, frame);
|
|
|
|
|
|
|
|
if (fChanged & PLAYER_AMMO1) {
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fChanged & PLAYER_AMMO2) {
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fChanged & PLAYER_AMMO3) {
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
#endif
|