//
// AMX Mod X, based on AMX Mod by Aleksander Naszko ("OLO").
// Copyright (C) The AMX Mod X Development Team.
//
// This software is licensed under the GNU General Public License, version 3 or higher.
// Additional exceptions apply. For full license details, see LICENSE.txt or visit:
//     https://alliedmods.net/amxmodx-license

#ifndef _PDATA_SHARED_H_
#define _PDATA_SHARED_H_

#include <amxxmodule.h>
#include <IGameConfigs.h>
#include <HLTypeConversion.h>
#include <amtl/am-algorithm.h>

extern HLTypeConversion TypeConversion;

enum class BaseFieldType
{
	None,
	Integer,
	Float,
	Vector,
	Entity,
	String,
};

#define GET_TYPE_DESCRIPTION(position, data, conf)                                         \
	int classLength, memberLength;                                                         \
	char const *className = MF_GetAmxString(amx, params[position], 0, &classLength);       \
	char const *memberName = MF_GetAmxString(amx, params[position + 1], 1, &memberLength); \
	if (!classLength || !memberLength)                                                     \
	{                                                                                      \
		MF_LogError(amx, AMX_ERR_NATIVE, "Either class (\"%s\") or member (\"%s\") is empty", className, memberName); \
		return 0;                                                                          \
	}                                                                                      \
	else if (!conf->GetOffsetByClass(className, memberName, &data))                        \
	{                                                                                      \
		MF_LogError(amx, AMX_ERR_NATIVE, "Could not find class \"%s\" and/or member \"%s\" in gamedata", className, memberName); \
		return 0;                                                                          \
	}                                                                                      \
	else if (data.fieldOffset < 0)                                                         \
	{                                                                                      \
			MF_LogError(amx, AMX_ERR_NATIVE, "Invalid offset %d retrieved from \"%s\" member", data.fieldOffset, memberName); \
			return 0;                                                                      \
	}

#define CHECK_DATA(data, element, baseType)                                                \
	if (baseType > BaseFieldType::None && baseType != PvData::GetBaseDataType(data))       \
	{                                                                                      \
			MF_LogError(amx, AMX_ERR_NATIVE, "Data field is not %s-based", PvData::GetBaseTypeName(baseType)); \
			return 0;                                                                      \
	}                                                                                      \
	else if (element < 0 || (element > 0 && element >= data.fieldSize))                    \
	{                                                                                      \
			MF_LogError(amx, AMX_ERR_NATIVE, "Invalid element index %d, value must be between 0 and %d", element, data.fieldSize); \
			return 0;                                                                      \
	}                                                                                      \
	else if (element > 0 && !data.fieldSize)                                               \
	{                                                                                      \
			MF_LogError(amx, AMX_ERR_NATIVE, "Member \"%s\" is not an array. Element %d is invalid.", memberName, element); \
			return 0;                                                                      \
	}

#define CHECK_GAMERULES()                                                                               \
	if ((HasRegameDll && !GameRulesRH) || (!HasRegameDll && (!GameRulesAddress || !*GameRulesAddress))) \
	{                                                                                                   \
		MF_LogError(amx, AMX_ERR_NATIVE, "%s is disabled. Check your AMXX log.", __FUNCTION__);         \
		return 0;                                                                                       \
	}

class PvData
{
public:

	static cell GetInt(int index, TypeDescription &data, int element)
	{
		return GetInt(TypeConversion.id_to_edict(index)->pvPrivateData, data, element);
	}

	static cell GetInt(void *pObject, TypeDescription &data, int element)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_INTEGER:
			case FieldType::FIELD_STRINGINT:
			{
				return get_pdata<int32>(pObject, data.fieldOffset, element);
			}
			case FieldType::FIELD_CLASS:
			case FieldType::FIELD_STRUCTURE:
			{
				return get_pdata_direct<cell>(pObject, data.fieldOffset);
			}
			case FieldType::FIELD_POINTER:
			case FieldType::FIELD_FUNCTION:
			{
				return reinterpret_cast<cell>(get_pdata<void*>(pObject, data.fieldOffset, element));
			}
			case FieldType::FIELD_SHORT:
			{
				if (data.fieldUnsigned)
				{
					return get_pdata<uint16>(pObject, data.fieldOffset, element);
				}
				else
				{
					return get_pdata<int16>(pObject, data.fieldOffset, element);
				}
			}
			case FieldType::FIELD_CHARACTER:
			{
				if (data.fieldUnsigned)
				{
					return get_pdata<uint8>(pObject, data.fieldOffset, element);
				}
				else
				{
					return get_pdata<int8>(pObject, data.fieldOffset, element);
				}
			}
			case FieldType::FIELD_BOOLEAN:
			{
				return get_pdata<bool>(pObject, data.fieldOffset, element) ? 1 : 0;
			}
		}

		return 0;
	}


	static void SetInt(int index, TypeDescription &data, cell value, int element)
	{
		SetInt(TypeConversion.id_to_edict(index)->pvPrivateData, data, value, element);
	}

	static void SetInt(void *pObject, TypeDescription &data, cell value, int element)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_INTEGER:
			case FieldType::FIELD_STRINGINT:
			{
				set_pdata<int32>(pObject, data.fieldOffset, static_cast<int32>(value), element);
				break;
			}
			case FieldType::FIELD_POINTER:
			case FieldType::FIELD_FUNCTION:
			{
				set_pdata<void*>(pObject, data.fieldOffset, reinterpret_cast<void*>(value), element);
				break;
			}
			case FieldType::FIELD_SHORT:
			{
				if (data.fieldUnsigned)
				{
					set_pdata<uint16>(pObject, data.fieldOffset, static_cast<uint16>(value), element);
				}
				else
				{
					set_pdata<int16>(pObject, data.fieldOffset, static_cast<int16>(value), element);
				}
				break;
			}
			case FieldType::FIELD_CHARACTER:
			{
				if (data.fieldUnsigned)
				{
					set_pdata<uint8>(pObject, data.fieldOffset, static_cast<uint8>(value), element);
				}
				else
				{
					set_pdata<int8>(pObject, data.fieldOffset, static_cast<int8>(value), element);
				}
				break;
			}
			case FieldType::FIELD_BOOLEAN:
			{
				set_pdata<bool>(pObject, data.fieldOffset, value != 0, element);
				break;
			}
		}
	}


	static cell GetFloat(int index, TypeDescription &data, int element)
	{
		return GetFloat(TypeConversion.id_to_edict(index)->pvPrivateData, data, element);
	}

	static cell GetFloat(void *pObject, TypeDescription &data, int element)
	{
		return amx_ftoc(get_pdata<float>(pObject, data.fieldOffset, element));
	}


	static void SetFloat(int index, TypeDescription &data, float value, int element)
	{
		SetFloat(TypeConversion.id_to_edict(index)->pvPrivateData, data, value, element);
	}

	static void SetFloat(void *pObject, TypeDescription &data, float value, int element)
	{
		set_pdata<float>(pObject, data.fieldOffset, value, element);
	}


	static void GetVector(int index, TypeDescription &data, cell *pVector, int element)
	{
		return GetVector(TypeConversion.id_to_edict(index)->pvPrivateData, data, pVector, element);
	}

	static void GetVector(void *pObject, TypeDescription &data, cell *pVector, int element)
	{
		auto vector = get_pdata<Vector>(pObject, data.fieldOffset, element);

		pVector[0] = amx_ftoc(vector.x);
		pVector[1] = amx_ftoc(vector.y);
		pVector[2] = amx_ftoc(vector.z);
	}


	static void SetVector(int index, TypeDescription &data, cell *pVector, int element)
	{
		SetVector(TypeConversion.id_to_edict(index)->pvPrivateData, data, pVector, element);
	}

	static void SetVector(void *pObject, TypeDescription &data, cell *pVector, int element)
	{
		Vector vector(amx_ctof(pVector[0]), amx_ctof(pVector[1]), amx_ctof(pVector[2]));

		set_pdata<Vector>(pObject, data.fieldOffset, vector, element);
	}


	static cell GetEntity(int index, TypeDescription &data, int element)
	{
		return GetEntity(TypeConversion.id_to_edict(index)->pvPrivateData, data, element);
	}

	static cell GetEntity(void *pObject, TypeDescription &data, int element)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_CLASSPTR:
			{
				return TypeConversion.cbase_to_id(get_pdata<void*>(pObject, data.fieldOffset, element));
			}
			case FieldType::FIELD_ENTVARS:
			{
				return TypeConversion.entvars_to_id(get_pdata<entvars_t*>(pObject, data.fieldOffset, element));
			}
			case FieldType::FIELD_EDICT:
			{
				return TypeConversion.edict_to_id(get_pdata<edict_t*>(pObject, data.fieldOffset, element));
			}
			case FieldType::FIELD_EHANDLE:
			{
				return TypeConversion.edict_to_id(get_pdata<EHANDLE>(pObject, data.fieldOffset, element).Get());
			}
		}

		return 0;
	}


	static void SetEntity(int index, TypeDescription &data, int value, int element)
	{
		SetEntity(TypeConversion.id_to_edict(index)->pvPrivateData, data, value, element);
	}

	static void SetEntity(void *pObject, TypeDescription &data, int value, int element)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_CLASSPTR:
			{
				set_pdata<void*>(pObject, data.fieldOffset, value != -1 ? TypeConversion.id_to_cbase(value) : nullptr, element);
				break;
			}
			case FieldType::FIELD_ENTVARS:
			{
				set_pdata<entvars_t*>(pObject, data.fieldOffset, value != -1 ? TypeConversion.id_to_entvars(value) : nullptr, element);
				break;
			}
			case FieldType::FIELD_EDICT:
			{
				set_pdata<edict_t*>(pObject, data.fieldOffset, value != -1 ? TypeConversion.id_to_edict(value) : nullptr, element);
				break;
			}
			case FieldType::FIELD_EHANDLE:
			{
				get_pdata<EHANDLE>(pObject, data.fieldOffset, element).Set(value != -1 ? TypeConversion.id_to_edict(value) : nullptr);
				break;
			}
		}
	}


	static char* GetString(int index, TypeDescription &data, int element)
	{
		return GetString(TypeConversion.id_to_edict(index)->pvPrivateData, data, element);
	}

	static char* GetString(void *pObject, TypeDescription &data, int element)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_STRING:
			{
				return get_pdata_direct<char*>(pObject, data.fieldOffset, element, data.fieldSize);
			}
			case FieldType::FIELD_STRINGPTR:
			{
				return get_pdata<char*>(pObject, data.fieldOffset, element);
			}
		}

		return nullptr;
	}


	static cell SetString(int index, TypeDescription &data, const char *value, int maxlen, int element)
	{
		return SetString(TypeConversion.id_to_edict(index)->pvPrivateData, data, value, maxlen, element);
	}

	static cell SetString(void *pObject, TypeDescription &data, const char *value, int maxlen, int element)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_STRING:
			{
				auto buffer = get_pdata_direct<char*>(pObject, data.fieldOffset);
				return strncopy(buffer, value, ke::Min<int>(maxlen + 1, data.fieldSize));
			}
			case FieldType::FIELD_STRINGPTR:
			{
				auto buffer = get_pdata<char*>(pObject, data.fieldOffset, element);

				if (!buffer || maxlen > static_cast<int>(strlen(buffer)))
				{
					if (buffer)
					{
						free(buffer);
					}

					buffer = reinterpret_cast<char*>(malloc(maxlen + 1));
					set_pdata<char*>(pObject, data.fieldOffset, buffer, element);
				}

				return strncopy(buffer, value, maxlen + 1);
			}
		}

		return 0;
	}

public:

	static BaseFieldType GetBaseDataType(TypeDescription &data)
	{
		switch (data.fieldType)
		{
			case FieldType::FIELD_INTEGER:
			case FieldType::FIELD_STRINGINT:
			case FieldType::FIELD_SHORT:
			case FieldType::FIELD_CHARACTER:
			case FieldType::FIELD_CLASS:
			case FieldType::FIELD_STRUCTURE:
			case FieldType::FIELD_POINTER:
			case FieldType::FIELD_FUNCTION:
			case FieldType::FIELD_BOOLEAN:
			{
				return BaseFieldType::Integer;
			}
			case FieldType::FIELD_FLOAT:
			{
				return BaseFieldType::Float;
			}
			case FieldType::FIELD_VECTOR:
			{
				return BaseFieldType::Vector;
			}
			case FieldType::FIELD_CLASSPTR:
			case FieldType::FIELD_ENTVARS:
			case FieldType::FIELD_EDICT:
			case FieldType::FIELD_EHANDLE:
			{
				return BaseFieldType::Entity;
			}
			case FieldType::FIELD_STRINGPTR:
			case FieldType::FIELD_STRING:
			{
				return BaseFieldType::String;
			}
		}

		return BaseFieldType::None;
	}

	static const char* GetBaseTypeName(BaseFieldType baseType)
	{
		static const char *BaseFieldTypeName[] =
		{
			"none",
			"integer",
			"float",
			"vector",
			"entity",
			"string",
		};

		return BaseFieldTypeName[static_cast<size_t>(baseType)];
	}
};

#endif // _PDATA_SHARED_H_