2017-10-12 21:50:56 +07:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the
|
|
|
|
* Free Software Foundation; either version 2 of the License, or (at
|
|
|
|
* your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
* In addition, as a special exception, the author gives permission to
|
|
|
|
* link the code of this program with the Half-Life Game Engine ("HL
|
|
|
|
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
|
|
|
|
* L.L.C ("Valve"). You must obey the GNU General Public License in all
|
|
|
|
* respects for all of the code used other than the HL Engine and MODs
|
|
|
|
* from Valve. If you modify this file, you may extend this exception
|
|
|
|
* to your version of the file, but you are not obligated to do so. If
|
|
|
|
* you do not wish to do so, delete this exception statement from your
|
|
|
|
* version.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
// Safe way to point to CBaseEntities who may die between frames.
|
|
|
|
template <typename T = CBaseEntity>
|
|
|
|
class EntityHandle
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
EntityHandle() : m_edict(nullptr), m_serialnumber(0) {}
|
|
|
|
EntityHandle(const EntityHandle<T> &other);
|
|
|
|
EntityHandle(const T *pEntity);
|
|
|
|
EntityHandle(const edict_t *pEdict);
|
|
|
|
|
|
|
|
// cast to base class
|
|
|
|
// NOTE: this is a unsafe method
|
|
|
|
template <typename R>
|
|
|
|
R *Get() const;
|
|
|
|
|
|
|
|
edict_t *Get() const;
|
|
|
|
edict_t *Set(edict_t *pEdict);
|
|
|
|
|
2017-10-14 13:20:23 +07:00
|
|
|
void Remove();
|
2017-10-12 21:50:56 +07:00
|
|
|
bool IsValid() const;
|
|
|
|
int GetSerialNumber() const;
|
|
|
|
|
|
|
|
bool operator==(T *pEntity) const;
|
|
|
|
operator bool() const;
|
|
|
|
operator T *() const;
|
|
|
|
|
|
|
|
T *operator=(T *pEntity);
|
|
|
|
T *operator->();
|
|
|
|
|
|
|
|
// Copy the ehandle.
|
|
|
|
EntityHandle<T>& operator=(const EntityHandle<T> &other);
|
|
|
|
|
|
|
|
private:
|
|
|
|
edict_t *m_edict;
|
|
|
|
int m_serialnumber;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Short alias
|
|
|
|
using EHandle = EntityHandle<>;
|
|
|
|
using EHANDLE = EHandle;
|
|
|
|
|
|
|
|
// Inlines
|
|
|
|
template <typename T>
|
|
|
|
inline bool FNullEnt(EntityHandle<T> &hent)
|
|
|
|
{
|
|
|
|
return (!hent || FNullEnt(OFFSET(hent.Get())));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy constructor
|
|
|
|
template <typename T>
|
|
|
|
EntityHandle<T>::EntityHandle(const EntityHandle<T> &other)
|
|
|
|
{
|
|
|
|
m_edict = other.m_edict;
|
|
|
|
m_serialnumber = other.m_serialnumber;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
EntityHandle<T>::EntityHandle(const T *pEntity)
|
|
|
|
{
|
|
|
|
if (pEntity)
|
|
|
|
{
|
|
|
|
Set(ENT(pEntity->pev));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_edict = nullptr;
|
|
|
|
m_serialnumber = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
EntityHandle<T>::EntityHandle(const edict_t *pEdict)
|
|
|
|
{
|
|
|
|
Set(const_cast<edict_t *>(pEdict));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
template <typename R>
|
|
|
|
inline R *EntityHandle<T>::Get() const
|
|
|
|
{
|
|
|
|
return GET_PRIVATE<R>(Get());
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline edict_t *EntityHandle<T>::Get() const
|
|
|
|
{
|
|
|
|
if (!m_edict || m_edict->serialnumber != m_serialnumber || m_edict->free)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_edict;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline edict_t *EntityHandle<T>::Set(edict_t *pEdict)
|
|
|
|
{
|
|
|
|
m_edict = pEdict;
|
|
|
|
if (pEdict)
|
|
|
|
{
|
|
|
|
m_serialnumber = pEdict->serialnumber;
|
|
|
|
}
|
2024-12-11 23:30:27 +07:00
|
|
|
else
|
|
|
|
{
|
|
|
|
m_serialnumber = 0;
|
|
|
|
}
|
2017-10-12 21:50:56 +07:00
|
|
|
|
|
|
|
return pEdict;
|
|
|
|
}
|
|
|
|
|
2017-10-14 13:20:23 +07:00
|
|
|
template <typename T>
|
|
|
|
void EntityHandle<T>::Remove()
|
|
|
|
{
|
|
|
|
if (IsValid())
|
|
|
|
{
|
|
|
|
UTIL_Remove(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_edict = nullptr;
|
|
|
|
m_serialnumber = 0;
|
|
|
|
}
|
|
|
|
|
2017-10-12 21:50:56 +07:00
|
|
|
// Returns whether this handle is valid.
|
|
|
|
template <typename T>
|
|
|
|
inline bool EntityHandle<T>::IsValid() const
|
|
|
|
{
|
|
|
|
edict_t *pEdict = Get();
|
|
|
|
if (!pEdict)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
CBaseEntity *pEntity = GET_PRIVATE<CBaseEntity>(pEdict);
|
|
|
|
if (!pEntity)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// CBaseEntity serial number.
|
|
|
|
// Used to determine if the entity is still valid.
|
|
|
|
template <typename T>
|
|
|
|
inline int EntityHandle<T>::GetSerialNumber() const
|
|
|
|
{
|
|
|
|
return m_serialnumber;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline bool EntityHandle<T>::operator==(T *pEntity) const
|
|
|
|
{
|
2023-12-14 03:16:27 +07:00
|
|
|
DbgAssertMsg(pEntity != nullptr, "EntityHandle<T>::operator==: got a nullptr pointer!");
|
2017-10-12 21:50:56 +07:00
|
|
|
|
|
|
|
if (m_serialnumber != pEntity->edict()->serialnumber)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_edict == pEntity->edict();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline EntityHandle<T>::operator bool() const
|
|
|
|
{
|
2017-10-20 00:12:02 +07:00
|
|
|
return IsValid();
|
2017-10-12 21:50:56 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Gets the Entity this handle refers to.
|
|
|
|
// Returns null if invalid.
|
|
|
|
template <typename T>
|
|
|
|
inline EntityHandle<T>::operator T *() const
|
|
|
|
{
|
|
|
|
return GET_PRIVATE<T>(Get());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assigns the given entity to this handle.
|
|
|
|
template <typename T>
|
|
|
|
inline T *EntityHandle<T>::operator=(T *pEntity)
|
|
|
|
{
|
|
|
|
if (pEntity)
|
|
|
|
{
|
|
|
|
Set(ENT(pEntity->pev));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_edict = nullptr;
|
|
|
|
m_serialnumber = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return static_cast<T *>(pEntity);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline T *EntityHandle<T>::operator->()
|
|
|
|
{
|
|
|
|
edict_t *pEdict = Get();
|
2023-12-14 03:16:27 +07:00
|
|
|
DbgAssertMsg(pEdict != nullptr, "EntityHandle<T>::operator->: pointer is nullptr!");
|
2017-10-12 21:50:56 +07:00
|
|
|
|
|
|
|
T *pEntity = GET_PRIVATE<T>(pEdict);
|
2023-12-14 03:16:27 +07:00
|
|
|
DbgAssertMsg(pEntity != nullptr, "EntityHandle<T>::operator->: pvPrivateData is nullptr!");
|
2017-10-12 21:50:56 +07:00
|
|
|
return pEntity;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Makes this handle refer to the same entity as the given handle.
|
|
|
|
template <typename T>
|
|
|
|
inline EntityHandle<T>& EntityHandle<T>::operator=(const EntityHandle<T> &other)
|
|
|
|
{
|
|
|
|
m_edict = other.m_edict;
|
|
|
|
m_serialnumber = other.m_serialnumber;
|
|
|
|
|
|
|
|
return (*this);
|
|
|
|
}
|