2015-10-25 21:07:56 +06:00
|
|
|
/***
|
|
|
|
*
|
|
|
|
* Copyright (c) 1996-2002, Valve LLC. All rights reserved.
|
2015-12-27 14:06:55 +03:00
|
|
|
*
|
|
|
|
* This product contains software technology licensed from Id
|
|
|
|
* Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc.
|
2015-10-25 21:07:56 +06:00
|
|
|
* All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use, distribution, and modification of this source code and/or resulting
|
|
|
|
* object code is restricted to non-commercial enhancements to products from
|
|
|
|
* Valve LLC. All other use, distribution, or modification is prohibited
|
|
|
|
* without written permission from Valve LLC.
|
|
|
|
*
|
|
|
|
****/
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
#include "stdafx.h"
|
2015-10-25 21:07:56 +06:00
|
|
|
#include "cbase.h"
|
|
|
|
#include "animation.h"
|
|
|
|
#include "saverestore.h"
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
TYPEDESCRIPTION CBaseAnimating::m_SaveData[] =
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
DEFINE_FIELD(CBaseMonster, m_flFrameRate, FIELD_FLOAT),
|
|
|
|
DEFINE_FIELD(CBaseMonster, m_flGroundSpeed, FIELD_FLOAT),
|
|
|
|
DEFINE_FIELD(CBaseMonster, m_flLastEventCheck, FIELD_TIME),
|
|
|
|
DEFINE_FIELD(CBaseMonster, m_fSequenceFinished, FIELD_BOOLEAN),
|
|
|
|
DEFINE_FIELD(CBaseMonster, m_fSequenceLoops, FIELD_BOOLEAN),
|
2015-10-25 21:07:56 +06:00
|
|
|
};
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
IMPLEMENT_SAVERESTORE(CBaseAnimating, CBaseDelay);
|
2015-10-25 21:07:56 +06:00
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
float CBaseAnimating::StudioFrameAdvance(float flInterval)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
if (flInterval == 0)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
flInterval = (gpGlobals->time - pev->animtime);
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (flInterval <= 0.001)
|
|
|
|
{
|
|
|
|
pev->animtime = gpGlobals->time;
|
2015-12-27 14:06:55 +03:00
|
|
|
return 0;
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
}
|
2015-12-27 14:06:55 +03:00
|
|
|
|
|
|
|
if (!pev->animtime)
|
|
|
|
flInterval = 0;
|
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
pev->frame += flInterval * m_flFrameRate * pev->framerate;
|
|
|
|
pev->animtime = gpGlobals->time;
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
if (pev->frame < 0 || pev->frame >= 256)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
if (m_fSequenceLoops)
|
|
|
|
pev->frame -= (int)(pev->frame / 256.0) * 256.0;
|
|
|
|
else
|
2015-12-27 14:06:55 +03:00
|
|
|
pev->frame = (pev->frame < 0) ? 0 : 255;
|
|
|
|
|
|
|
|
m_fSequenceFinished = TRUE;
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
return flInterval;
|
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
int CBaseAnimating::LookupActivity(int activity)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
return ::LookupActivity(pmodel, pev, activity);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
int CBaseAnimating::LookupActivityHeaviest(int activity)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
return ::LookupActivityHeaviest(pmodel, pev, activity);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
int CBaseAnimating::LookupSequence(const char *label)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
return ::LookupSequence(pmodel, label);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::ResetSequenceInfo(void)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
2015-10-25 21:07:56 +06:00
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
GetSequenceInfo(pmodel, pev, &m_flFrameRate, &m_flGroundSpeed);
|
2015-10-25 21:07:56 +06:00
|
|
|
m_fSequenceLoops = ((GetSequenceFlags() & STUDIO_LOOPING) != 0);
|
|
|
|
pev->animtime = gpGlobals->time;
|
2015-12-27 14:06:55 +03:00
|
|
|
pev->framerate = 1;
|
2015-10-25 21:07:56 +06:00
|
|
|
m_fSequenceFinished = FALSE;
|
|
|
|
m_flLastEventCheck = gpGlobals->time;
|
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
BOOL CBaseAnimating::GetSequenceFlags(void)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
return ::GetSequenceFlags(pmodel, pev);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::DispatchAnimEvents(float flInterval)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
2015-10-25 21:07:56 +06:00
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
if (!pmodel)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
ALERT(at_aiconsole, "Gibbed monster is thinking!\n");
|
2015-10-25 21:07:56 +06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
flInterval = 0.1;
|
|
|
|
|
|
|
|
float flStart = pev->frame + (m_flLastEventCheck - pev->animtime) * m_flFrameRate * pev->framerate;
|
|
|
|
float flEnd = pev->frame + flInterval * m_flFrameRate * pev->framerate;
|
|
|
|
m_flLastEventCheck = pev->animtime + flInterval;
|
|
|
|
m_fSequenceFinished = FALSE;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
|
|
|
if (flEnd >= 256 || flEnd <= 0)
|
2015-10-25 21:07:56 +06:00
|
|
|
m_fSequenceFinished = TRUE;
|
|
|
|
|
|
|
|
int index = 0;
|
2015-12-27 14:06:55 +03:00
|
|
|
MonsterEvent_t event;
|
2015-10-25 21:07:56 +06:00
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
while ((index = GetAnimationEvent(pmodel, pev, &event, flStart, flEnd, index)) != 0)
|
|
|
|
HandleAnimEvent(&event);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
float CBaseAnimating::SetBoneController(int iController, float flValue)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
return SetController(pmodel, pev, iController, flValue);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::InitBoneControllers(void)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
2015-10-25 21:07:56 +06:00
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
SetController(pmodel, pev, 0, 0);
|
|
|
|
SetController(pmodel, pev, 1, 0);
|
|
|
|
SetController(pmodel, pev, 2, 0);
|
|
|
|
SetController(pmodel, pev, 3, 0);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
float CBaseAnimating::SetBlending(int iBlender, float flValue)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
return ::SetBlending(pmodel, pev, iBlender, flValue);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::GetBonePosition(int iBone, Vector &origin, Vector &angles)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
GET_BONE_POSITION(ENT(pev), iBone, origin, angles);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::GetAttachment(int iAttachment, Vector &origin, Vector &angles)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
GET_ATTACHMENT(ENT(pev), iAttachment, origin, angles);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
int CBaseAnimating::FindTransition(int iEndingSequence, int iGoalSequence, int *piDir)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
void *pmodel = GET_MODEL_PTR(ENT(pev));
|
|
|
|
|
|
|
|
if (!piDir)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
int iDir;
|
2015-12-27 14:06:55 +03:00
|
|
|
int sequence = ::FindTransition(pmodel, iEndingSequence, iGoalSequence, &iDir);
|
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (iDir != 1)
|
|
|
|
return -1;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
|
|
|
return sequence;
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
return ::FindTransition(pmodel, iEndingSequence, iGoalSequence, piDir);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::GetAutomovement(Vector &origin, Vector &angles, float flInterval)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::SetBodygroup(int iGroup, int iValue)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
::SetBodygroup(GET_MODEL_PTR(ENT(pev)), pev, iGroup, iValue);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
int CBaseAnimating::GetBodygroup(int iGroup)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
return ::GetBodygroup(GET_MODEL_PTR(ENT(pev)), pev, iGroup);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
int CBaseAnimating::ExtractBbox(int sequence, float *mins, float *maxs)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
return ::ExtractBbox(GET_MODEL_PTR(ENT(pev)), sequence, mins, maxs);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
void CBaseAnimating::SetSequenceBox(void)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
Vector mins, maxs;
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
if (ExtractBbox(pev->sequence, mins, maxs))
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
2015-12-27 14:06:55 +03:00
|
|
|
float yaw = pev->angles.y * (M_PI / 180);
|
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
Vector xvector, yvector;
|
|
|
|
xvector.x = cos(yaw);
|
|
|
|
xvector.y = sin(yaw);
|
|
|
|
yvector.x = -sin(yaw);
|
|
|
|
yvector.y = cos(yaw);
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
Vector bounds[2] = { mins, maxs };
|
|
|
|
Vector rmin(9999, 9999, 9999);
|
|
|
|
Vector rmax(-9999, -9999, -9999);
|
2015-10-25 21:07:56 +06:00
|
|
|
Vector base, transformed;
|
|
|
|
|
2015-12-27 14:06:55 +03:00
|
|
|
for (int i = 0; i <= 1; i++)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
base.x = bounds[i].x;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
|
|
|
for (int j = 0; j <= 1; j++)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
base.y = bounds[j].y;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
|
|
|
for (int k = 0; k <= 1; k++)
|
2015-10-25 21:07:56 +06:00
|
|
|
{
|
|
|
|
base.z = bounds[k].z;
|
2015-12-27 14:06:55 +03:00
|
|
|
transformed.x = xvector.x * base.x + yvector.x * base.y;
|
|
|
|
transformed.y = xvector.y * base.x + yvector.y * base.y;
|
2015-10-25 21:07:56 +06:00
|
|
|
transformed.z = base.z;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (transformed.x < rmin.x)
|
|
|
|
rmin.x = transformed.x;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (transformed.x > rmax.x)
|
|
|
|
rmax.x = transformed.x;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (transformed.y < rmin.y)
|
|
|
|
rmin.y = transformed.y;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (transformed.y > rmax.y)
|
|
|
|
rmax.y = transformed.y;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (transformed.z < rmin.z)
|
|
|
|
rmin.z = transformed.z;
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
if (transformed.z > rmax.z)
|
|
|
|
rmax.z = transformed.z;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-12-27 14:06:55 +03:00
|
|
|
|
2015-10-25 21:07:56 +06:00
|
|
|
rmin.z = 0;
|
|
|
|
rmax.z = rmin.z + 1;
|
2015-12-27 14:06:55 +03:00
|
|
|
UTIL_SetSize(pev, rmin, rmax);
|
2015-10-25 21:07:56 +06:00
|
|
|
}
|
2015-12-27 14:06:55 +03:00
|
|
|
}
|