2022-08-16 18:23:36 -04:00
|
|
|
#include <std_include.hpp>
|
|
|
|
#include "loader/component_loader.hpp"
|
|
|
|
|
|
|
|
#include "command.hpp"
|
2022-11-01 01:09:02 -04:00
|
|
|
#include "console.hpp"
|
2022-08-16 18:23:36 -04:00
|
|
|
#include "fastfiles.hpp"
|
|
|
|
|
2022-11-01 01:09:02 -04:00
|
|
|
#include "game/game.hpp"
|
|
|
|
|
|
|
|
#include <utils/hook.hpp>
|
2022-08-16 18:23:36 -04:00
|
|
|
|
|
|
|
namespace weapon
|
|
|
|
{
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
utils::hook::detour g_setup_level_weapon_def_hook;
|
|
|
|
void g_setup_level_weapon_def_stub()
|
|
|
|
{
|
|
|
|
// precache level weapons first
|
|
|
|
g_setup_level_weapon_def_hook.invoke<void>();
|
|
|
|
|
|
|
|
std::vector<game::WeaponDef*> weapons;
|
|
|
|
|
|
|
|
// find all weapons in asset pools
|
|
|
|
fastfiles::enum_assets(game::ASSET_TYPE_WEAPON, [&weapons](game::XAssetHeader header)
|
|
|
|
{
|
|
|
|
weapons.push_back(header.weapon);
|
|
|
|
}, false);
|
|
|
|
|
|
|
|
// sort weapons
|
|
|
|
std::sort(weapons.begin(), weapons.end(), [](game::WeaponDef* weapon1, game::WeaponDef* weapon2)
|
|
|
|
{
|
|
|
|
return std::string_view(weapon1->name) <
|
|
|
|
std::string_view(weapon2->name);
|
|
|
|
});
|
|
|
|
|
|
|
|
// precache items
|
|
|
|
for (std::size_t i = 0; i < weapons.size(); i++)
|
|
|
|
{
|
|
|
|
console::debug("precaching weapon \"%s\"\n", weapons[i]->name);
|
|
|
|
game::G_GetWeaponForName(weapons[i]->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-30 12:23:41 -04:00
|
|
|
utils::hook::detour xmodel_get_bone_index_hook;
|
2022-09-08 00:32:29 -04:00
|
|
|
__int64 xmodel_get_bone_index_stub(game::XModel* model, game::scr_string_t name, unsigned int offset, char* index)
|
2022-08-30 12:23:41 -04:00
|
|
|
{
|
2022-09-08 00:32:29 -04:00
|
|
|
auto result = xmodel_get_bone_index_hook.invoke<__int64>(model, name, offset, index);
|
2022-08-30 12:23:41 -04:00
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
if (name == game::SL_GetString("tag_weapon_right", 0) ||
|
|
|
|
name == game::SL_GetString("tag_knife_attach", 0))
|
|
|
|
{
|
2022-09-08 00:32:29 -04:00
|
|
|
result = xmodel_get_bone_index_hook.invoke<__int64>(model, game::SL_GetString("tag_weapon", 0), offset, index);
|
2022-08-30 12:23:41 -04:00
|
|
|
if (result)
|
|
|
|
{
|
|
|
|
console::debug("using tag_weapon instead of %s (%s, %d)\n", game::SL_ConvertToString(name), model->name, offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-09-11 21:33:58 -04:00
|
|
|
void cw_mismatch_error_stub(int, const char* msg, ...)
|
|
|
|
{
|
|
|
|
char buffer[0x100];
|
|
|
|
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, msg);
|
|
|
|
|
|
|
|
vsnprintf_s(buffer, sizeof(buffer), _TRUNCATE, msg, ap);
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
console::error(buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
int g_find_config_string_index_stub(const char* string, int start, int max, int create, const char* errormsg)
|
|
|
|
{
|
|
|
|
create = 1;
|
|
|
|
return utils::hook::invoke<int>(0x8B530_b, string, start, max, create, errormsg); // G_FindConfigstringIndex
|
|
|
|
}
|
|
|
|
|
2022-08-16 18:23:36 -04:00
|
|
|
template <typename T>
|
|
|
|
void set_weapon_field(const std::string& weapon_name, unsigned int field, T value)
|
|
|
|
{
|
|
|
|
auto weapon = game::DB_FindXAssetHeader(game::ASSET_TYPE_WEAPON, weapon_name.data(), false).data;
|
|
|
|
if (weapon)
|
|
|
|
{
|
|
|
|
if (field && field < (0xE20 + sizeof(T)))
|
|
|
|
{
|
|
|
|
*reinterpret_cast<T*>(reinterpret_cast<std::uintptr_t>(weapon) + field) = value;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
console::warn("weapon field: %d is higher than the size of weapon struct!\n", field);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
console::warn("weapon %s not found!\n", weapon_name.data());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_weapon_field_float(const std::string& weapon_name, unsigned int field, float value)
|
|
|
|
{
|
|
|
|
set_weapon_field<float>(weapon_name, field, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_weapon_field_int(const std::string& weapon_name, unsigned int field, int value)
|
|
|
|
{
|
|
|
|
set_weapon_field<int>(weapon_name, field, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_weapon_field_bool(const std::string& weapon_name, unsigned int field, bool value)
|
|
|
|
{
|
|
|
|
set_weapon_field<bool>(weapon_name, field, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class component final : public component_interface
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void post_unpack() override
|
|
|
|
{
|
2022-08-20 09:27:41 -04:00
|
|
|
if (!game::environment::is_sp())
|
|
|
|
{
|
2022-08-30 12:23:41 -04:00
|
|
|
// precache all weapons that are loaded in zones
|
2022-08-20 09:27:41 -04:00
|
|
|
g_setup_level_weapon_def_hook.create(0x462630_b, g_setup_level_weapon_def_stub);
|
|
|
|
|
2022-08-30 12:23:41 -04:00
|
|
|
// use tag_weapon if tag_weapon_right or tag_knife_attach are not found on model
|
|
|
|
xmodel_get_bone_index_hook.create(0x5C82B0_b, xmodel_get_bone_index_stub);
|
|
|
|
|
2022-09-11 21:33:58 -04:00
|
|
|
// make custom weapon index mismatch not drop in CG_SetupCustomWeapon
|
|
|
|
utils::hook::call(0x11B9AF_b, cw_mismatch_error_stub);
|
|
|
|
|
|
|
|
// patch attachment configstring so it will create if not found
|
|
|
|
utils::hook::call(0x41C595_b, g_find_config_string_index_stub);
|
2022-08-20 09:27:41 -04:00
|
|
|
}
|
2022-08-16 18:23:36 -04:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
command::add("setWeaponFieldFloat", [](const command::params& params)
|
|
|
|
{
|
|
|
|
if (params.size() <= 3)
|
|
|
|
{
|
|
|
|
console::info("usage: setWeaponFieldInt <weapon> <field> <value>\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
set_weapon_field_float(params.get(1), atoi(params.get(2)), static_cast<float>(atof(params.get(3))));
|
|
|
|
});
|
|
|
|
|
|
|
|
command::add("setWeaponFieldInt", [](const command::params& params)
|
|
|
|
{
|
|
|
|
if (params.size() <= 3)
|
|
|
|
{
|
|
|
|
console::info("usage: setWeaponFieldInt <weapon> <field> <value>\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
set_weapon_field_int(params.get(1), atoi(params.get(2)), static_cast<int>(atoi(params.get(3))));
|
|
|
|
});
|
|
|
|
|
|
|
|
command::add("setWeaponFieldBool", [](const command::params& params)
|
|
|
|
{
|
|
|
|
if (params.size() <= 3)
|
|
|
|
{
|
|
|
|
console::info("usage: setWeaponFieldBool <weapon> <field> <value>\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
set_weapon_field_bool(params.get(1), atoi(params.get(2)), static_cast<bool>(atoi(params.get(3))));
|
|
|
|
});
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
REGISTER_COMPONENT(weapon::component)
|