290 lines
7.5 KiB
C++
290 lines
7.5 KiB
C++
#include <std_include.hpp>
|
|
#include "dvars.hpp"
|
|
#include "scheduler.hpp"
|
|
#include "spoofer/spoofcall.hpp"
|
|
#include "definitions/variables.hpp"
|
|
#include "loader/component_loader.hpp"
|
|
|
|
#include <utilities/string.hpp>
|
|
|
|
namespace dvars
|
|
{
|
|
namespace
|
|
{
|
|
void fetch_dvar_pointers()
|
|
{
|
|
for (auto& dvar : variables::dvars_record)
|
|
{
|
|
dvar.pointer = spoofcall::invoke<uintptr_t>(game::Dvar_FindVar, dvar.name.data());
|
|
}
|
|
}
|
|
|
|
std::string get_vector_string(const int components, const game::DvarLimits& domain)
|
|
{
|
|
if (domain.vector.min == -FLT_MAX)
|
|
{
|
|
if (domain.vector.max == FLT_MAX)
|
|
{
|
|
return utilities::string::va("Domain is any %iD vector", components);
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any %iD vector with components %g or smaller", components,
|
|
domain.vector.max);
|
|
}
|
|
}
|
|
else if (domain.vector.max == FLT_MAX)
|
|
{
|
|
return utilities::string::va("Domain is any %iD vector with components %g or bigger", components,
|
|
domain.vector.min);
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any %iD vector with components from %g to %g", components,
|
|
domain.vector.min, domain.vector.max);
|
|
}
|
|
}
|
|
}
|
|
|
|
std::string get_domain_string(const game::dvarType_t type, const game::DvarLimits& domain)
|
|
{
|
|
std::string str;
|
|
|
|
switch (type)
|
|
{
|
|
case game::DVAR_TYPE_BOOL:
|
|
return "Domain is 0 or 1"s;
|
|
|
|
case game::DVAR_TYPE_FLOAT:
|
|
if (domain.value.min == -FLT_MAX)
|
|
{
|
|
if (domain.value.max == FLT_MAX)
|
|
{
|
|
return "Domain is any number"s;
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any number %g or smaller", domain.value.max);
|
|
}
|
|
}
|
|
else if (domain.value.max == FLT_MAX)
|
|
{
|
|
return utilities::string::va("Domain is any number %g or bigger", domain.value.min);
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any number from %g to %g", domain.value.min, domain.value.max);
|
|
}
|
|
|
|
case game::DVAR_TYPE_FLOAT_2:
|
|
return get_vector_string(2, domain);
|
|
|
|
case game::DVAR_TYPE_FLOAT_3:
|
|
case game::DVAR_TYPE_LINEAR_COLOR_RGB:
|
|
case game::DVAR_TYPE_COLOR_XYZ:
|
|
case game::DVAR_TYPE_COLOR_LAB:
|
|
return get_vector_string(3, domain);
|
|
|
|
case game::DVAR_TYPE_FLOAT_4:
|
|
return get_vector_string(4, domain);
|
|
|
|
case game::DVAR_TYPE_INT:
|
|
if (domain.integer.min == INT_MIN)
|
|
{
|
|
if (domain.integer.max == INT_MAX)
|
|
{
|
|
return "Domain is any integer"s;
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any integer %i or smaller", domain.integer.max);
|
|
}
|
|
}
|
|
else if (domain.integer.max == INT_MAX)
|
|
{
|
|
return utilities::string::va("Domain is any integer %i or bigger", domain.integer.min);
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any integer from %i to %i", domain.integer.min, domain.integer.max);
|
|
}
|
|
|
|
case game::DVAR_TYPE_ENUM:
|
|
str = "Domain is one of the following:"s;
|
|
|
|
for (auto string_index = 0; string_index < domain.enumeration.stringCount; ++string_index)
|
|
{
|
|
str += utilities::string::va("\n %2i: %s", string_index, domain.enumeration.strings[string_index]);
|
|
}
|
|
|
|
return str;
|
|
|
|
case game::DVAR_TYPE_STRING:
|
|
return "Domain is any text"s;
|
|
|
|
case game::DVAR_TYPE_COLOR:
|
|
return "Domain is any 4-component color, in RGBA format"s;
|
|
|
|
case game::DVAR_TYPE_INT64:
|
|
if (domain.integer64.min == _I64_MIN)
|
|
{
|
|
if (domain.integer64.max == _I64_MAX)
|
|
{
|
|
return "Domain is any integer"s;
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any integer %lli or smaller", domain.integer64.max);
|
|
}
|
|
}
|
|
else if (domain.integer64.max == _I64_MAX)
|
|
{
|
|
return utilities::string::va("Domain is any integer %lli or bigger", domain.integer64.min);
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any integer from %lli to %lli", domain.integer64.min, domain.integer64.max);
|
|
}
|
|
|
|
case game::DVAR_TYPE_UINT64:
|
|
if (domain.unsignedInt64.min)
|
|
{
|
|
if (domain.unsignedInt64.max == _UI64_MAX)
|
|
{
|
|
return utilities::string::va("Domain is any unsigned integer %zu or bigger", domain.unsignedInt64.min);
|
|
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any unsigned integer from %zu to %zu", domain.unsignedInt64.min, domain.unsignedInt64.max);
|
|
}
|
|
}
|
|
else if (domain.unsignedInt64.max == _UI64_MAX)
|
|
{
|
|
return "Domain is any integer"s;
|
|
}
|
|
else
|
|
{
|
|
return utilities::string::va("Domain is any integer %zu or smaller", domain.unsignedInt64.max);
|
|
}
|
|
|
|
default:
|
|
return utilities::string::va("unhandled dvar type '%i'", type);
|
|
}
|
|
}
|
|
|
|
std::string get_value_string(const game::dvar_t * dvar, game::DvarValue * value)
|
|
{
|
|
std::string result = "N/A";
|
|
|
|
switch (dvar->type)
|
|
{
|
|
case game::DVAR_TYPE_BOOL:
|
|
if (value->naked.enabled)
|
|
{
|
|
result = "1"s;
|
|
}
|
|
else
|
|
{
|
|
result = "0"s;
|
|
}
|
|
break;
|
|
|
|
case game::DVAR_TYPE_FLOAT:
|
|
result = std::format("{:.2f}", value->naked.value);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_FLOAT_2:
|
|
result = std::format("{:.2f} {:.2f}", value->naked.vector[0], value->naked.vector[1]);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_FLOAT_3:
|
|
case game::DVAR_TYPE_LINEAR_COLOR_RGB:
|
|
case game::DVAR_TYPE_COLOR_XYZ:
|
|
case game::DVAR_TYPE_COLOR_LAB:
|
|
result = std::format("{:.2f} {:.2f} {:.2f}", value->naked.vector[0], value->naked.vector[1], value->naked.vector[2]);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_FLOAT_4:
|
|
result = std::format("{:.2f} {:.2f} {:.2f} {:.2f}", value->naked.vector[0], value->naked.vector[1], value->naked.vector[2], value->naked.vector[3]);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_INT:
|
|
result = std::format("{}", value->naked.integer);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_ENUM:
|
|
if (dvar->domain.enumeration.stringCount)
|
|
{
|
|
result = std::string(dvar->domain.enumeration.strings[value->naked.integer]);
|
|
}
|
|
break;
|
|
|
|
case game::DVAR_TYPE_STRING:
|
|
result = std::string(value->naked.string);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_COLOR:
|
|
result = std::format("{:.2f} {:.2f} {:.2f} {:.2f}", (float)value->naked.color[0] * 0.0039215689, (float)value->naked.color[1] * 0.0039215689, (float)value->naked.color[2] * 0.0039215689, (float)value->naked.color[3] * 0.0039215689);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_INT64:
|
|
result = std::format("{}", value->naked.integer64);
|
|
break;
|
|
|
|
case game::DVAR_TYPE_UINT64:
|
|
result = std::format("{}", value->naked.unsignedInt64);
|
|
break;
|
|
|
|
default:
|
|
result = "ERROR:DVAR_TYPE_UNKNOWN"s;
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
game::dvar_t* find_dvar(uint64_t hashRef)
|
|
{
|
|
if (hashRef == 0) return NULL;
|
|
|
|
auto it = std::find_if(variables::dvars_record.begin(), variables::dvars_record.end(), [&hashRef](variables::varEntry& i) { return i.fnv1a == hashRef; });
|
|
|
|
if (it != variables::dvars_record.end() && it->pointer)
|
|
{
|
|
return reinterpret_cast<game::dvar_t*>(it->pointer);
|
|
}
|
|
|
|
return spoofcall::invoke<game::dvar_t*>(game::Dvar_FindVar_Hash, game::AssetRef(hashRef));
|
|
}
|
|
|
|
game::dvar_t* find_dvar(const char* nameRef)
|
|
{
|
|
return spoofcall::invoke<game::dvar_t*>(game::Dvar_FindVar, nameRef);
|
|
}
|
|
|
|
game::dvar_t* find_dvar(const std::string& nameRef)
|
|
{
|
|
auto it = std::find_if(variables::dvars_record.begin(), variables::dvars_record.end(), [&nameRef](variables::varEntry& i) { return utilities::string::compare(i.name, nameRef); });
|
|
|
|
if (it != variables::dvars_record.end() && it->pointer)
|
|
{
|
|
return reinterpret_cast<game::dvar_t*>(it->pointer);
|
|
}
|
|
|
|
return spoofcall::invoke<game::dvar_t*>(game::Dvar_FindVar, nameRef.data());
|
|
}
|
|
|
|
|
|
class component final : public component_interface
|
|
{
|
|
public:
|
|
void post_unpack() override
|
|
{
|
|
scheduler::once(fetch_dvar_pointers, scheduler::pipeline::main);
|
|
}
|
|
};
|
|
}
|
|
|
|
REGISTER_COMPONENT(dvars::component)
|