//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============// // // Purpose: // // $NoKeywords: $ //=============================================================================// #ifndef EHANDLE_H #define EHANDLE_H #ifdef _WIN32 #pragma once #endif #if defined( _DEBUG ) && defined( GAME_DLL ) #include "tier0/dbg.h" #include "cbase.h" #endif #include "const.h" #include "basehandle.h" #include "entitylist_base.h" class IHandleEntity; // -------------------------------------------------------------------------------------------------- // // Game-code CBaseHandle implementation. // -------------------------------------------------------------------------------------------------- // inline IHandleEntity* CBaseHandle::Get() const { extern CBaseEntityList *g_pEntityList; return g_pEntityList->LookupEntity( *this ); } // -------------------------------------------------------------------------------------------------- // // CHandle. // // Only safe to use in cases where you can statically verify that T* can safely be reinterpret-casted // to IHandleEntity*; that is, that it's derived from IHandleEntity and IHandleEntity is the // first base class. // // Unfortunately some classes are forward-declared and the compiler can't determine at compile time // how to static_cast<> them to IHandleEntity. // -------------------------------------------------------------------------------------------------- // template< class T > class CHandle : public CBaseHandle { public: CHandle(); CHandle( int iEntry, int iSerialNumber ); /*implicit*/ CHandle( T *pVal ); /*implicit*/ CHandle( INVALID_EHANDLE_tag ); // NOTE: The following two constructor functions are not type-safe, and can allow creating a // CHandle that doesn't actually point to an object of type T. // // It is your responsibility to ensure that the target of the handle actually points to the // correct type of object before calling these functions. static CHandle UnsafeFromBaseHandle( const CBaseHandle& handle ); // The index should have come from a call to CBaseHandle::ToInt(). If it hasn't, you're in trouble. static CHandle UnsafeFromIndex( int index ); T* Get() const; void Set( const T* pVal ); /*implicit*/ operator T*(); /*implicit*/ operator T*() const; bool operator !() const; bool operator==( T *val ) const; bool operator!=( T *val ) const; CHandle& operator=( const T *val ); T* operator->() const; }; // ----------------------------------------------------------------------- // // Inlines. // ----------------------------------------------------------------------- // template inline CHandle::CHandle() { } template inline CHandle::CHandle( INVALID_EHANDLE_tag ) : CBaseHandle( INVALID_EHANDLE ) { } template inline CHandle::CHandle( int iEntry, int iSerialNumber ) { Init( iEntry, iSerialNumber ); } template inline CHandle::CHandle( T *pObj ) : CBaseHandle( INVALID_EHANDLE ) { Set( pObj ); } template inline CHandle CHandle::UnsafeFromBaseHandle( const CBaseHandle &handle ) { CHandle ret; ret.m_Index = handle.m_Index; return ret; } template inline CHandle CHandle::UnsafeFromIndex( int index ) { CHandle ret; ret.m_Index = index; return ret; } template inline T* CHandle::Get() const { return (T*)CBaseHandle::Get(); } template inline CHandle::operator T *() { return Get( ); } template inline CHandle::operator T *() const { return Get( ); } template inline bool CHandle::operator !() const { return !Get(); } template inline bool CHandle::operator==( T *val ) const { return Get() == val; } template inline bool CHandle::operator!=( T *val ) const { return Get() != val; } template void CHandle::Set( const T* pVal ) { // We can't even verify that the class can successfully reinterpret-cast to IHandleEntity* // because that will cause this function to fail to compile in Debug in the case of forward-declared // pointer types. //Assert( reinterpret_cast< const IHandleEntity* >( pVal ) == static_cast< IHandleEntity* >( pVal ) ); const IHandleEntity* pValInterface = reinterpret_cast( pVal ); CBaseHandle::Set( pValInterface ); } template inline CHandle& CHandle::operator=( const T *val ) { Set( val ); return *this; } template T* CHandle::operator -> () const { return Get(); } // specialization of EnsureValidValue for CHandle template FORCEINLINE void EnsureValidValue( CHandle &x ) { x = INVALID_EHANDLE; } #endif // EHANDLE_H