1
0
mirror of https://github.com/alliedmodders/hl2sdk.git synced 2025-01-07 09:43:40 +08:00
hl2sdk/public/variant.h

978 lines
33 KiB
C
Raw Normal View History

2023-07-26 14:01:46 +03:00
#ifndef CVARIANT_H
#define CVARIANT_H
#if _WIN32
#pragma once
#endif
2023-12-23 20:54:00 +03:00
#include "string_t.h"
2023-07-26 14:01:46 +03:00
#include "datamap.h"
#include "vector.h"
#include "vector2d.h"
#include "vector4d.h"
#include "Color.h"
#include "entity2/entityidentity.h"
#include "entityhandle.h"
2023-09-29 13:13:00 +03:00
#include "tier1/bufferstring.h"
#include "tier1/utlscratchmemory.h"
#include "resourcefile/resourcetype.h"
2023-07-26 14:01:46 +03:00
2023-12-23 20:54:00 +03:00
FORWARD_DECLARE_HANDLE( HSCRIPT );
2023-07-26 14:01:46 +03:00
// ========
class CVariantDefaultAllocator
{
public:
static void Free(void *pMemory)
{
free(pMemory);
}
static void *Allocate(int nSize)
{
return malloc(nSize);
}
};
class CEntityVariantAllocator
{
public:
static void Free(void *pMemory) { /* Skipped intentionally */ }
static void *Allocate(int nSize)
{
2024-05-06 09:08:31 +03:00
return sm_pMemoryPool->AllocAligned(nSize, 8 * (nSize >= 16) + 8);
2023-07-26 14:01:46 +03:00
}
static void Activate(CUtlScratchMemoryPool *pMemoryPool, bool bEnable)
{
sm_pMemoryPool = bEnable ? pMemoryPool : NULL;
}
private:
static CUtlScratchMemoryPool *sm_pMemoryPool;
};
inline const char *VariantFieldTypeName(fieldtype_t eType)
2023-07-26 14:01:46 +03:00
{
switch(eType)
{
case FIELD_VOID: return "void";
case FIELD_FLOAT: return "float";
case FIELD_STRING: return "string_t";
case FIELD_VECTOR: return "vector";
case FIELD_QUATERNION: return "quaternion";
case FIELD_INTEGER: return "integer";
case FIELD_BOOLEAN: return "boolean";
case FIELD_CHARACTER: return "character";
case FIELD_COLOR32: return "color";
case FIELD_EHANDLE: return "ehandle";
case FIELD_VECTOR2D: return "vector2d";
case FIELD_VECTOR4D: return "vector4d";
case FIELD_INTEGER64: return "int64";
case FIELD_RESOURCE: return "resourcehandle";
case FIELD_CSTRING: return "cstring";
case FIELD_HSCRIPT: return "hscript";
case FIELD_VARIANT: return "variant";
case FIELD_UINT64: return "uint64";
case FIELD_FLOAT64: return "float64";
case FIELD_UINT: return "unsigned";
case FIELD_UTLSTRINGTOKEN: return "utlstringtoken";
case FIELD_QANGLE: return "qangle";
case FIELD_HSCRIPT_LIGHTBINDING: return "hscript_lightbinding";
case FIELD_V8_VALUE: return "js_value";
case FIELD_V8_OBJECT: return "js_object";
case FIELD_V8_ARRAY: return "js_array";
case FIELD_V8_CALLBACK_INFO: return "js_raw_args";
default: return "unknown_variant_type";
}
}
template <typename T> struct VariantTypeDeducer { };
#define DECLARE_DEDUCE_VARIANT_FIELDTYPE( fieldType, type ) template<> struct VariantTypeDeducer<type> { enum { FIELD_TYPE = fieldType }; };
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VOID, void);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_CSTRING, const char *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_CSTRING, char *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR, Vector);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR, Vector *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR, const Vector &);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR2D, Vector2D);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR2D, Vector2D *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR2D, const Vector2D &);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR4D, Vector4D);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR4D, Vector4D *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_VECTOR4D, const Vector4D &);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_COLOR32, Color);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_COLOR32, Color *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_COLOR32, const Color &);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_QANGLE, QAngle);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_QANGLE, QAngle *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_QANGLE, const QAngle &);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_QUATERNION, Quaternion);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_QUATERNION, Quaternion *);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_QUATERNION, const Quaternion &);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_STRING, string_t);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_BOOLEAN, bool);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_CHARACTER, char);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_INTEGER64, int64);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_UINT64, uint64);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_FLOAT, float);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_FLOAT64, float64);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_INTEGER, int);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_UINT, uint);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_HSCRIPT, HSCRIPT);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_EHANDLE, CEntityHandle);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_RESOURCE, ResourceHandle_t);
DECLARE_DEDUCE_VARIANT_FIELDTYPE(FIELD_UTLSTRINGTOKEN, CUtlStringToken);
#define VariantDeduceType( T ) (fieldtype_t)VariantTypeDeducer<T>::FIELD_TYPE
2023-07-26 14:01:46 +03:00
#undef DECLARE_DEDUCE_VARIANT_FIELDTYPE
template <typename T>
inline const char *VariantFieldTypeName()
{
return T::using_unknown_variant_type();
}
#define DECLARE_NAMED_VARIANT_FIELDTYPE( fieldType, strName ) template <> inline const char * VariantFieldTypeName<fieldType>() { return strName; }
DECLARE_NAMED_VARIANT_FIELDTYPE(void, "void");
DECLARE_NAMED_VARIANT_FIELDTYPE(const char *, "cstring");
DECLARE_NAMED_VARIANT_FIELDTYPE(char *, "cstring");
DECLARE_NAMED_VARIANT_FIELDTYPE(bool, "boolean");
DECLARE_NAMED_VARIANT_FIELDTYPE(char, "character");
DECLARE_NAMED_VARIANT_FIELDTYPE(int, "integer");
DECLARE_NAMED_VARIANT_FIELDTYPE(uint, "unsigned integer");
DECLARE_NAMED_VARIANT_FIELDTYPE(int64, "int64");
DECLARE_NAMED_VARIANT_FIELDTYPE(uint64, "uint64");
DECLARE_NAMED_VARIANT_FIELDTYPE(float, "float");
DECLARE_NAMED_VARIANT_FIELDTYPE(float64, "float64");
#undef DECLARE_NAMED_VARIANT_FIELDTYPE
enum CVFlags_t
{
// Indicates that variant has the memory allocated in place of a primitive types and it would be freed when needed
CV_FREE = 0x01,
};
template <typename A>
class CVariantBase
{
public:
typedef A Allocator;
CVariantBase() : m_type( FIELD_VOID ), m_flags( 0 ) { m_pData = NULL; }
CVariantBase( int val ) : m_type( FIELD_INTEGER ), m_flags( 0 ) { m_int = val;}
CVariantBase( uint val) : m_type( FIELD_UINT ), m_flags( 0 ) { m_uint = val; }
CVariantBase( float val ) : m_type( FIELD_FLOAT ), m_flags( 0 ) { m_float = val; }
CVariantBase( float64 val ) : m_type( FIELD_FLOAT64 ), m_flags( 0 ) { m_float64 = val; }
CVariantBase( char val ) : m_type( FIELD_CHARACTER ), m_flags( 0 ) { m_char = val; }
CVariantBase( bool val ) : m_type( FIELD_BOOLEAN ), m_flags( 0 ) { m_bool = val; }
CVariantBase( HSCRIPT val ) : m_type( FIELD_HSCRIPT ), m_flags( 0 ) { m_hScript = val; }
CVariantBase( CEntityHandle val ) : m_type( FIELD_EHANDLE ), m_flags( 0 ) { m_hEntity = val; }
CVariantBase( CUtlStringToken val ) : m_type( FIELD_UTLSTRINGTOKEN ), m_flags( 0 ){ m_utlStringToken = val; }
CVariantBase( ResourceHandle_t val ) : m_type( FIELD_RESOURCE ), m_flags( 0 ) { m_hResource = val; }
CVariantBase( string_t val ) : m_type( FIELD_STRING ), m_flags( 0 ) { m_stringt = val; }
CVariantBase( int64 val ) : m_type( FIELD_INTEGER64 ), m_flags( 0 ) { m_int64 = val; }
CVariantBase( uint64 val ) : m_type( FIELD_UINT64), m_flags( 0 ) { m_uint64 = val; }
CVariantBase( const Vector &val, bool bCopy = true) : m_type( FIELD_VECTOR ), m_flags( 0 ) { CopyData(val, bCopy); }
CVariantBase( const Vector *val, bool bCopy = true) : m_type( FIELD_VECTOR ), m_flags( 0 ) { CopyData(*val, bCopy); }
CVariantBase( const QAngle &val, bool bCopy = true) : m_type( FIELD_QANGLE ), m_flags( 0 ) { CopyData(val, bCopy); }
CVariantBase( const QAngle *val, bool bCopy = true) : m_type( FIELD_QANGLE ), m_flags( 0 ) { CopyData(*val, bCopy); }
CVariantBase( const Vector2D &val, bool bCopy = true ) : m_type( FIELD_VECTOR2D ), m_flags( 0 ) { CopyData(val, bCopy); }
CVariantBase( const Vector2D *val, bool bCopy = true ) : m_type( FIELD_VECTOR2D ), m_flags( 0 ) { CopyData(*val, bCopy); }
CVariantBase( const Vector4D &val, bool bCopy = true ) : m_type( FIELD_VECTOR4D ), m_flags( 0 ) { CopyData(val, bCopy); }
CVariantBase( const Vector4D *val, bool bCopy = true ) : m_type( FIELD_VECTOR4D ), m_flags( 0 ) { CopyData(*val, bCopy); }
CVariantBase( const Quaternion &val, bool bCopy = true ) : m_type( FIELD_QUATERNION ), m_flags( 0 ) { CopyData(val, bCopy); }
CVariantBase( const Quaternion *val, bool bCopy = true ) : m_type( FIELD_QUATERNION ), m_flags( 0 ) { CopyData(*val, bCopy); }
CVariantBase( const Color &val, bool bCopy = true ) : m_type( FIELD_COLOR32 ), m_flags( 0 ) { CopyData(val, bCopy); }
CVariantBase( const Color *val, bool bCopy = true ) : m_type( FIELD_COLOR32 ), m_flags( 0 ) { CopyData(*val, bCopy); }
CVariantBase( const char *val, bool bCopy = true ) : m_type( FIELD_CSTRING ), m_flags( 0 ) { CopyData(val, bCopy); }
// Checks if the stored value is of type FIELD_VOID
bool IsNull() const { return (m_type == FIELD_VOID ); }
// Copies the src data into an internal value, setting bForceCopy would allocate its own memory to store the contents
void CopyData(const char *src, bool bForceCopy)
{
Free();
m_type = FIELD_CSTRING;
if(src && bForceCopy)
{
int len = strlen(src) + 1;
m_pszString = (char *)Allocator::Allocate(len);
memcpy((void *)m_pszString, src, len);
m_flags |= CV_FREE;
}
else
{
m_pszString = src;
}
}
// Copies the src data into an internal value, setting bForceCopy would allocate its own memory to store the contents
template <typename T>
void CopyData(const T &src, bool bForceCopy)
{
Free();
m_type = VariantDeduceType(T);
if(bForceCopy)
{
m_pData = Allocator::Allocate(sizeof(T));
*this = src;
m_flags |= CV_FREE;
}
else
{
m_pData = *(void **)&src;
}
}
operator int() const { Assert( m_type == FIELD_INTEGER ); return m_int; }
operator uint() const { Assert( m_type == FIELD_UINT ); return m_uint; }
operator int64() const { Assert( m_type == FIELD_INTEGER64); return m_int64; }
operator uint64() const { Assert( m_type == FIELD_UINT64); return m_uint64; }
operator float() const { Assert( m_type == FIELD_FLOAT ); return m_float; }
operator float64() const { Assert( m_type == FIELD_FLOAT64 ); return m_float64; }
operator const string_t() const { Assert( m_type == FIELD_STRING ); return m_stringt; }
operator const char *() const { Assert( m_type == FIELD_CSTRING ); return ( m_pszString ) ? m_pszString : ""; }
operator const Vector &() const { Assert( m_type == FIELD_VECTOR ); static Vector vecNull(0, 0, 0); return (m_pVector) ? *m_pVector : vecNull; }
operator const Vector2D &() const { Assert( m_type == FIELD_VECTOR2D ); static Vector2D vecNull(0, 0); return (m_pVector2D) ? *m_pVector2D : vecNull; }
operator const Vector4D &() const { Assert( m_type == FIELD_VECTOR4D ); static Vector4D vecNull(0, 0, 0, 0); return (m_pVector4D) ? *m_pVector4D : vecNull; }
operator const QAngle &() const { Assert( m_type == FIELD_QANGLE); static QAngle angNull(0, 0, 0); return (m_pQAngle) ? *m_pQAngle : angNull; }
operator const Quaternion &() const { Assert( m_type == FIELD_QUATERNION); static Quaternion quatNull(0, 0, 0, 0); return (m_pQuaternion) ? *m_pQuaternion : quatNull; }
operator const Color &() const { Assert( m_type == FIELD_COLOR32); static Color colorNull(0, 0, 0); return (m_pColor) ? *m_pColor : colorNull; }
operator char() const { Assert( m_type == FIELD_CHARACTER ); return m_char; }
operator bool() const { Assert( m_type == FIELD_BOOLEAN ); return m_bool; }
operator HSCRIPT() const { Assert( m_type == FIELD_HSCRIPT ); return m_hScript; }
operator CEntityHandle() const { Assert( m_type == FIELD_EHANDLE); return m_hEntity; }
operator CUtlStringToken() const { Assert( m_type == FIELD_UTLSTRINGTOKEN); return m_utlStringToken; }
operator ResourceHandle_t() const { Assert( m_type == FIELD_RESOURCE); return m_hResource; }
void operator=( int i ) { m_type = FIELD_INTEGER; m_int = i; }
void operator=( uint u ) { m_type = FIELD_UINT; m_uint = u; }
void operator=( int64 i ) { m_type = FIELD_INTEGER64; m_int64 = i; }
void operator=( uint64 u ) { m_type = FIELD_UINT64; m_uint64 = u; }
void operator=( float f ) { m_type = FIELD_FLOAT; m_float = f; }
void operator=( float64 d ) { m_type = FIELD_FLOAT64; m_float64 = d; }
void operator=( const Vector &vec ) { m_type = FIELD_VECTOR; *(Vector *)m_pVector = vec; }
void operator=( const Vector *vec ) { m_type = FIELD_VECTOR; m_pVector = vec; }
void operator=( const Vector2D &vec ) { m_type = FIELD_VECTOR2D; *(Vector2D *)m_pVector2D = vec; }
void operator=( const Vector2D *vec ) { m_type = FIELD_VECTOR2D; m_pVector2D = vec; }
void operator=( const Vector4D &vec ) { m_type = FIELD_VECTOR4D; *(Vector4D *)m_pVector4D = vec; }
void operator=( const Vector4D *vec ) { m_type = FIELD_VECTOR4D; m_pVector4D = vec; }
void operator=( const QAngle &ang ) { m_type = FIELD_QANGLE; *(QAngle *)m_pQAngle = ang; }
void operator=( const QAngle *ang ) { m_type = FIELD_QANGLE; m_pQAngle = ang; }
void operator=( const Quaternion &quat ){ m_type = FIELD_QUATERNION; *(Quaternion *)m_pQuaternion = quat; }
void operator=( const Quaternion *quat ){ m_type = FIELD_QUATERNION; m_pQuaternion = quat; }
void operator=( const Color &color ) { m_type = FIELD_COLOR32; *(Color *)m_pColor = color; }
void operator=( const Color *color ) { m_type = FIELD_COLOR32; m_pColor = color; }
void operator=( string_t psz ) { m_type = FIELD_STRING; m_stringt = psz; }
void operator=( const char *psz ) { m_type = FIELD_CSTRING; m_pszString = psz; }
void operator=( char c ) { m_type = FIELD_CHARACTER; m_char = c; }
void operator=( bool b ) { m_type = FIELD_BOOLEAN; m_bool = b; }
void operator=( HSCRIPT h ) { m_type = FIELD_HSCRIPT; m_hScript = h; }
void operator=( CEntityHandle eh) { m_type = FIELD_EHANDLE; m_hEntity = eh; }
void operator=( CUtlStringToken tok ) { m_type = FIELD_UTLSTRINGTOKEN; m_utlStringToken = tok; }
void operator=( ResourceHandle_t r ) { m_type = FIELD_RESOURCE; m_hResource = r; }
~CVariantBase()
{
Free();
}
// Frees the internal buffer and resets the value to be FIELD_VOID
void Free()
{
if(m_flags & CV_FREE)
{
Allocator::Free(m_pData);
m_flags &= ~CV_FREE;
}
m_pData = NULL;
m_type = FIELD_VOID;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(CBufferString &buf)
{
buf.Purge(0);
switch(m_type)
{
case FIELD_VOID: return false;
case FIELD_FLOAT: buf.Format("%g", m_float); return true;
case FIELD_FLOAT64: buf.Format("%g", m_float64); return true;
case FIELD_INTEGER: buf.Format("%d", m_int); return true;
case FIELD_UINT: buf.Format("%u", m_uint); return true;
case FIELD_INTEGER64: buf.Format("%lld", m_int64); return true;
case FIELD_UINT64: buf.Format("%llu", m_uint64); return true;
case FIELD_BOOLEAN: buf.Insert(0, m_bool ? "true" : "false"); return true;
case FIELD_STRING: buf.Insert(0, m_stringt.ToCStr()); return true;
case FIELD_CSTRING: buf.Insert(0, m_pszString ? m_pszString : "(null)"); return true;
case FIELD_CHARACTER: buf.Format("%c", m_char); return true;
case FIELD_VECTOR2D: buf.Format("%g %g", m_pVector2D->x, m_pVector2D->y); return true;
case FIELD_COLOR32: buf.Format("%d %d %d %d", m_pColor->r(), m_pColor->g(), m_pColor->b(), m_pColor->a()); return true;
case FIELD_VECTOR:
case FIELD_QANGLE:
{
buf.Format("%g %g %g", m_pVector->x, m_pVector->y, m_pVector->z);
return true;
}
case FIELD_QUATERNION:
case FIELD_VECTOR4D:
{
buf.Format("%g %g %g %g", m_pVector4D->x, m_pVector4D->y, m_pVector4D->z, m_pVector4D->w);
return true;
}
default: Warning("No conversion from %s to string at the moment!\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(float *pDest)
{
switch(m_type)
{
case FIELD_VOID: *pDest = 0.0; return false;
case FIELD_INTEGER: *pDest = m_int; return true;
case FIELD_INTEGER64: *pDest = m_int64; return true;
case FIELD_UINT: *pDest = m_uint; return true;
case FIELD_UINT64: *pDest = m_uint64; return true;
case FIELD_FLOAT: *pDest = m_float; return true;
case FIELD_FLOAT64: *pDest = m_float64; return true;
case FIELD_BOOLEAN: *pDest = m_bool; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0])
{
*pDest = V_atof(m_pszString);
return true;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
*pDest = V_atof(m_stringt.ToCStr());
return true;
}
}
default: Warning("No conversion from %s to float right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(int *pDest)
{
switch(m_type)
{
case FIELD_VOID: *pDest = 0; return false;
case FIELD_INTEGER: *pDest = m_int; return true;
case FIELD_INTEGER64: *pDest = m_int64; return true;
case FIELD_UINT: *pDest = m_uint; return true;
case FIELD_UINT64: *pDest = m_uint64; return true;
case FIELD_FLOAT: *pDest = m_float; return true;
case FIELD_FLOAT64: *pDest = m_float64; return true;
case FIELD_BOOLEAN: *pDest = m_bool; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0])
{
*pDest = V_atoi(m_pszString);
return true;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
*pDest = V_atoi(m_stringt.ToCStr());
return true;
}
}
default: Warning("No conversion from %s to int right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(bool *pDest)
{
switch(m_type)
{
case FIELD_VOID: *pDest = 0; return false;
case FIELD_INTEGER: *pDest = m_int != 0; return true;
case FIELD_INTEGER64: *pDest = m_int64 != 0; return true;
case FIELD_UINT: *pDest = m_uint != 0; return true;
case FIELD_UINT64: *pDest = m_uint64 != 0; return true;
case FIELD_FLOAT: *pDest = m_float != 0.0; return true;
case FIELD_FLOAT64: *pDest = m_float64 != 0.0; return true;
case FIELD_BOOLEAN: *pDest = m_bool; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0])
{
bool successful = false;
*pDest = V_StringToBool(m_pszString, false, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
*pDest = V_StringToBool(m_stringt.ToCStr(), false, &successful);
return successful;
}
}
default: Warning("No conversion from %s to bool right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(string_t *pDest)
{
if(m_type == FIELD_STRING)
{
*pDest = *this;
return true;
}
else if(m_type == FIELD_CSTRING)
{
if(m_pszString)
{
*pDest = MAKE_STRING(m_pszString);
return true;
}
}
else
{
Warning("No conversion from %s to string_t right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(Vector *pDest)
{
switch(m_type)
{
case FIELD_VOID: return false;
case FIELD_VECTOR: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0] != '\0')
{
bool successful = false;
V_StringToVector(m_pszString, *pDest, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
V_StringToVector(m_stringt.ToCStr(), *pDest, &successful);
return successful;
}
}
default: Warning("No conversion from %s to Vector right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(Vector2D *pDest)
{
switch(m_type)
{
case FIELD_VOID: return false;
case FIELD_VECTOR2D: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0] != '\0')
{
bool successful = false;
V_StringToVector2D(m_pszString, *pDest, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
V_StringToVector2D(m_stringt.ToCStr(), *pDest, &successful);
return successful;
}
}
default: Warning("No conversion from %s to Vector2D right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(Vector4D *pDest)
{
switch(m_type)
{
case FIELD_VOID: return false;
case FIELD_VECTOR4D: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0] != '\0')
{
bool successful = false;
V_StringToVector4D(m_pszString, *pDest, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
V_StringToVector4D(m_stringt.ToCStr(), *pDest, &successful);
return successful;
}
}
default: Warning("No conversion from %s to Vector4D right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(Quaternion *pDest)
{
switch(m_type)
{
case FIELD_VOID: return false;
case FIELD_QUATERNION: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0] != '\0')
{
bool successful = false;
V_StringToQuaternion(m_pszString, *pDest, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
V_StringToQuaternion(m_stringt.ToCStr(), *pDest, &successful);
return successful;
}
}
default: Warning("No conversion from %s to Quaternion right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(QAngle *pDest)
{
switch(m_type)
{
case FIELD_VOID: return false;
case FIELD_QANGLE: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0] != '\0')
{
bool successful = false;
V_StringToQAngle(m_pszString, *pDest, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
V_StringToQAngle(m_stringt.ToCStr(), *pDest, &successful);
return successful;
}
}
default: Warning("No conversion from %s to QAngle right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(Color *pDest)
{
switch(m_type)
{
case FIELD_COLOR32: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0] != '\0')
{
bool successful = false;
V_StringToColor(m_pszString, *pDest, &successful);
return successful;
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
bool successful = false;
V_StringToColor(m_stringt.ToCStr(), *pDest, &successful);
return successful;
}
}
default: Warning("No conversion from %s to Color right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(ResourceHandle_t *pDest)
{
if(m_type == FIELD_RESOURCE)
{
*pDest = *this;
return true;
}
else
{
Warning("No conversion from %s to ResourceHandle_t right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(HSCRIPT *pDest)
{
if(m_type == FIELD_HSCRIPT)
{
*pDest = *this;
return true;
}
else
{
Warning("No conversion from %s to HSCRIPT right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(CUtlStringToken *pDest)
{
if(m_type == FIELD_UTLSTRINGTOKEN)
{
*pDest = *this;
return true;
}
else
{
Warning("No conversion from %s to CUtlStringToken right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(CEntityHandle *pDest)
{
switch(m_type)
{
case FIELD_EHANDLE: *pDest = *this; return true;
case FIELD_CSTRING:
{
if(m_pszString && m_pszString[0])
{
// TODO: Perform actual entity handle lookup via CEntitySystem::FindFirstEntityHandleByName()
Assert(false);
}
}
case FIELD_STRING:
{
if(m_stringt.ToCStr()[0])
{
// TODO: Perform actual entity handle lookup via CEntitySystem::FindFirstEntityHandleByName()
Assert(false);
}
}
default: Warning("No conversion from %s to EHANDLE2 right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
bool AssignTo(CEntityInstance *pDest)
{
if(m_type == FIELD_EHANDLE)
{
// TODO: Perform actual CEntityInstance lookup via CEntityHandle
Assert(false);
return true;
}
else
{
Warning("No conversion from %s to CEntityInstance right now\n", VariantFieldTypeName(m_type));
}
return false;
}
// Copies the contents of the value into a pDest, also converts the content when possible
template <typename T>
bool AssignTo(CVariantBase<T> *pDest)
{
switch(m_type)
{
case FIELD_VECTOR: pDest->CopyData(*m_pVector, true); return true;
case FIELD_VECTOR2D: pDest->CopyData(*m_pVector2D, true); return true;
case FIELD_VECTOR4D: pDest->CopyData(*m_pVector4D, true); return true;
case FIELD_QUATERNION: pDest->CopyData(*m_pQuaternion, true); return true;
case FIELD_QANGLE: pDest->CopyData(*m_pQAngle, true); return true;
case FIELD_COLOR32: pDest->CopyData(*m_pColor, true); return true;
case FIELD_CSTRING: pDest->CopyData(m_pszString, true); return true;
default:
{
pDest->Free();
pDest->m_type = m_type;
pDest->m_pData = m_pData;
return true;
}
}
}
template <typename T>
bool AssignTo(T *pDest)
{
fieldtype_t destType = VariantDeduceType(T);
2023-07-26 14:01:46 +03:00
if(destType == FIELD_VOID)
{
return false;
}
if(destType == m_type)
{
*pDest = *this;
return true;
}
if(destType != FIELD_VECTOR && destType != FIELD_QUATERNION && destType != FIELD_VECTOR2D &&
destType != FIELD_VECTOR4D && destType != FIELD_CSTRING && destType != FIELD_QANGLE)
{
switch(m_type)
{
case FIELD_VOID: *pDest = 0; return false;
case FIELD_INTEGER: *pDest = m_int; return true;
case FIELD_INTEGER64: *pDest = m_int64; return true;
case FIELD_UINT: *pDest = m_uint; return true;
case FIELD_UINT64: *pDest = m_uint64; return true;
case FIELD_FLOAT: *pDest = m_float; return true;
case FIELD_FLOAT64: *pDest = m_float64; return true;
case FIELD_CHARACTER: *pDest = m_char; return true;
case FIELD_BOOLEAN: *pDest = m_bool; return true;
}
}
else
{
Warning("No free conversion of %s variant to %s right now\n", VariantFieldTypeName(m_type), VariantFieldTypeName<T>());
*pDest = 0;
}
return false;
}
// Allocates own buffers and copies the internal value when needed, if silent = false, emits a global warning
void ConvertToCopiedData(bool silent = true)
{
switch(m_type)
{
case FIELD_VECTOR: CopyData(*m_pVector, true); break;
case FIELD_VECTOR2D: CopyData(*m_pVector2D, true); break;
case FIELD_VECTOR4D: CopyData(*m_pVector4D, true); break;
case FIELD_QUATERNION: CopyData(*m_pQuaternion, true); break;
case FIELD_QANGLE: CopyData(*m_pQAngle, true); break;
case FIELD_COLOR32: CopyData(*m_pColor, true); break;
case FIELD_CSTRING: CopyData(m_pszString, true); break;
default:
{
if(!silent)
{
Warning("Attempted to ConvertToCopiedData for unsupported type (%d)\n", m_type);
}
}
}
}
template <typename T>
T Get()
{
T value = {};
AssignTo(&value);
return value;
}
// Sets the internal value to the pData content, doesn't allocate memory nor copies the content.
// Be sure to call ConvertToCopiedData() when required
void Set(fieldtype_t ftype, void *pData)
{
switch(ftype)
{
case FIELD_VOID: Free(); m_type = FIELD_VOID; m_pData = NULL; return;
case FIELD_FLOAT: CopyData(*(float *)pData, false); return;
case FIELD_FLOAT64: CopyData(*(float64 *)pData, false); return;
case FIELD_INTEGER: CopyData(*(int *)pData, false); return;
case FIELD_UINT: CopyData(*(uint *)pData, false); return;
case FIELD_INTEGER64: CopyData(*(int64 *)pData, false); return;
case FIELD_UINT64: CopyData(*(uint64 *)pData, false); return;
case FIELD_BOOLEAN: CopyData(*(bool *)pData, false); return;
case FIELD_CHARACTER: CopyData(*(char *)pData, false); return;
case FIELD_STRING: CopyData(*(string_t *)pData, false); return;
case FIELD_CSTRING: CopyData(*(const char **)pData, false); return;
case FIELD_VECTOR: CopyData((Vector *)pData, false); return;
case FIELD_VECTOR2D: CopyData((Vector2D *)pData, false); return;
case FIELD_VECTOR4D: CopyData((Vector4D *)pData, false); return;
case FIELD_COLOR32: CopyData((Color *)pData, false); return;
case FIELD_QANGLE: CopyData((QAngle *)pData, false); return;
case FIELD_QUATERNION: CopyData((Quaternion *)pData, false); return;
case FIELD_HSCRIPT: CopyData(*(HSCRIPT *)pData, false); return;
case FIELD_EHANDLE: CopyData(*(CEntityHandle *)pData, false); return;
case FIELD_RESOURCE: CopyData(*(ResourceHandle_t *)pData, false); return;
case FIELD_UTLSTRINGTOKEN: CopyData(*(CUtlStringToken *)pData, false); return;
}
}
// Converts underlying value into a different type
bool Convert(fieldtype_t newType)
{
if(newType == m_type)
{
return true;
}
bool successful = false;
void *pData = NULL;
switch(newType)
{
case FIELD_VOID: successful = true; Free(); m_type = FIELD_VOID; m_pData = NULL; break;
2024-03-04 07:02:02 +03:00
case FIELD_FLOAT: if((successful = AssignTo((float *)&pData))) { Set(newType, &pData); } break;
case FIELD_FLOAT64: if((successful = AssignTo((float64 *)&pData))) { Set(newType, &pData); } break;
case FIELD_INTEGER: if((successful = AssignTo((int *)&pData))) { Set(newType, &pData); } break;
case FIELD_UINT: if((successful = AssignTo((uint *)&pData))) { Set(newType, &pData); } break;
case FIELD_INTEGER64: if((successful = AssignTo((int64 *)&pData))) { Set(newType, &pData); } break;
case FIELD_UINT64: if((successful = AssignTo((uint64 *)&pData))) { Set(newType, &pData); } break;
case FIELD_BOOLEAN: if((successful = AssignTo((bool *)&pData))) { Set(newType, &pData); } break;
case FIELD_CHARACTER: if((successful = AssignTo((char *)&pData))) { Set(newType, &pData); } break;
case FIELD_STRING: if((successful = AssignTo((string_t *)&pData))) { Set(newType, &pData); } break;
2023-07-26 14:01:46 +03:00
case FIELD_CSTRING: successful = true; CopyData(ToString(), true); break;
2024-03-04 07:02:02 +03:00
case FIELD_HSCRIPT: if((successful = AssignTo((HSCRIPT *)&pData))) { Set(newType, &pData); } break;
case FIELD_EHANDLE: if((successful = AssignTo((CEntityHandle *)&pData))) { Set(newType, &pData); } break;
case FIELD_RESOURCE: if((successful = AssignTo((ResourceHandle_t *)&pData))) { Set(newType, &pData); } break;
case FIELD_UTLSTRINGTOKEN: if((successful = AssignTo((CUtlStringToken *)&pData))) { Set(newType, &pData); } break;
case FIELD_VECTOR: { Vector vec; if((successful = AssignTo(&vec))) { CopyData(vec, true); } break; }
case FIELD_VECTOR2D: { Vector2D vec; if((successful = AssignTo(&vec))) { CopyData(vec, true); } break; }
case FIELD_VECTOR4D: { Vector4D vec; if((successful = AssignTo(&vec))) { CopyData(vec, true); } break; }
case FIELD_COLOR32: { Color clr; if((successful = AssignTo(&clr))) { CopyData(clr, true); } break; }
case FIELD_QANGLE: { QAngle ang; if((successful = AssignTo(&ang))) { CopyData(ang, true); } break; }
case FIELD_QUATERNION: { Quaternion quat; if((successful = AssignTo(&quat))) { CopyData(quat, true); } break; }
2023-07-26 14:01:46 +03:00
}
if(successful)
{
m_type = newType;
}
return successful;
}
const char *ToString()
{
static CBufferStringGrowable<200> szBuf;
AssignTo(szBuf);
return szBuf.Get();
}
public:
union
{
int m_int;
uint m_uint;
float m_float;
const char *m_pszString;
const Vector *m_pVector;
const QAngle *m_pQAngle;
const Vector2D *m_pVector2D;
const Vector4D *m_pVector4D;
const Quaternion *m_pQuaternion;
const Color *m_pColor;
void *m_pData;
char m_char;
bool m_bool;
HSCRIPT m_hScript;
CEntityHandle m_hEntity;
uint64 m_uint64;
int64 m_int64;
float64 m_float64;
string_t m_stringt;
CUtlStringToken m_utlStringToken;
ResourceHandle_t m_hResource;
};
fieldtype_t m_type;
2023-07-26 14:01:46 +03:00
// CVFlags_t flags
uint16 m_flags;
};
typedef CVariantBase<CVariantDefaultAllocator> CVariant;
typedef CVariantBase<CEntityVariantAllocator> CEntityVariant;
2023-09-22 23:23:21 +03:00
typedef CVariant variant_t;
2023-07-26 14:01:46 +03:00
#endif // CVARIANT_H