Images component

This commit is contained in:
Federico Cecchetto 2022-06-08 00:39:48 +02:00
parent 733be076fa
commit 2fcd184833
3 changed files with 138 additions and 7 deletions

View File

@ -0,0 +1,123 @@
#include <std_include.hpp>
#include "loader/component_loader.hpp"
#include "images.hpp"
#include "console.hpp"
#include "filesystem.hpp"
#include "game/game.hpp"
#include <utils/hook.hpp>
#include <utils/string.hpp>
#include <utils/image.hpp>
#include <utils/io.hpp>
#include <utils/concurrency.hpp>
namespace images
{
namespace
{
utils::hook::detour load_texture_hook;
utils::hook::detour setup_texture_hook;
utils::concurrency::container<std::unordered_map<std::string, std::string>> overriden_textures;
std::optional<std::string> load_image(game::GfxImage* image)
{
std::string data{};
overriden_textures.access([&](const std::unordered_map<std::string, std::string>& textures)
{
if (const auto i = textures.find(image->name); i != textures.end())
{
data = i->second;
}
});
if (data.empty() && !filesystem::read_file(utils::string::va("images/%s.png", image->name), &data))
{
return {};
}
return {std::move(data)};
}
std::optional<utils::image> load_raw_image_from_file(game::GfxImage* image)
{
const auto image_file = load_image(image);
if (!image_file)
{
return {};
}
return utils::image(*image_file);
}
bool load_custom_texture(game::GfxImage* image)
{
auto raw_image = load_raw_image_from_file(image);
if (!raw_image)
{
return false;
}
image->imageFormat = 0x1000003;
image->resourceSize = -1;
D3D11_SUBRESOURCE_DATA data{};
data.SysMemPitch = raw_image->get_width() * 4;
data.SysMemSlicePitch = data.SysMemPitch * raw_image->get_height();
data.pSysMem = raw_image->get_buffer();
game::Image_Setup(image, raw_image->get_width(), raw_image->get_height(), image->depth, image->numElements,
image->imageFormat, DXGI_FORMAT_R8G8B8A8_UNORM, image->name, &data);
return true;
}
void load_texture_stub(game::GfxImage* image, void* a2, int* a3)
{
try
{
if (load_custom_texture(image))
{
return;
}
}
catch (std::exception& e)
{
console::error("Failed to load image %s: %s\n", image->name, e.what());
}
load_texture_hook.invoke<void>(image, a2, a3);
}
int setup_texture_stub(game::GfxImage* image, void* a2, void* a3)
{
if (image->resourceSize == -1)
{
return 0;
}
return setup_texture_hook.invoke<bool>(image, a2, a3);
}
}
void override_texture(std::string name, std::string data)
{
overriden_textures.access([&](std::unordered_map<std::string, std::string>& textures)
{
textures[std::move(name)] = std::move(data);
});
}
class component final : public component_interface
{
public:
void post_unpack() override
{
setup_texture_hook.create(SELECT_VALUE(0x55F870_b, 0x6829C0_b), setup_texture_stub);
load_texture_hook.create(SELECT_VALUE(0x83300_b, 0xA4AA0_b), load_texture_stub);
}
};
}
REGISTER_COMPONENT(images::component)

View File

@ -0,0 +1,6 @@
#pragma once
namespace images
{
void override_texture(std::string name, std::string data);
}

View File

@ -192,6 +192,8 @@ namespace ui_scripting
return fps::get_fps();
};
if (::game::environment::is_mp())
{
game_type["getping"] = [](const game&)
{
if ((*::game::mp::client_state) == nullptr)
@ -201,6 +203,7 @@ namespace ui_scripting
return (*::game::mp::client_state)->ping;
};
}
game_type["issingleplayer"] = [](const game&)
{
@ -537,7 +540,6 @@ namespace ui_scripting
command::add("lui_restart", []()
{
utils::hook::invoke<void>(SELECT_VALUE(0xFB370_b, 0x2707C0_b));
utils::hook::invoke<void>(SELECT_VALUE(0x1052C0_b, 0x27BEC0_b));
});
}