From 8e583c35fbd00d4ba179387e0c87db2bce73d49d Mon Sep 17 00:00:00 2001 From: Emmanuel AYME Date: Mon, 21 Jul 2025 14:42:51 +0200 Subject: [PATCH] Adding Robocop Ufinished Business --- Reshade Plugins Addons.sln | 10 ++ Robocop UB/RobocopUB.vcxproj | 156 +++++++++++++++++++++++ Robocop UB/dllmain.cpp | 233 +++++++++++++++++++++++++++++++++++ 3 files changed, 399 insertions(+) create mode 100644 Robocop UB/RobocopUB.vcxproj create mode 100644 Robocop UB/dllmain.cpp diff --git a/Reshade Plugins Addons.sln b/Reshade Plugins Addons.sln index e533383..f695cf3 100644 --- a/Reshade Plugins Addons.sln +++ b/Reshade Plugins Addons.sln @@ -25,6 +25,8 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Mindseye", "Mindseye\Mindse EndProject Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Metro Exodus", "Metro Exodus\Metro Exodus.vcxproj", "{ABD1ABB5-BB10-4427-AD26-233702D870BD}" EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Robocop UB", "Robocop UB\RobocopUB.vcxproj", "{B6C09563-1AF1-4226-A6B6-76802399CCDB}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|x64 = Debug|x64 @@ -113,6 +115,14 @@ Global {ABD1ABB5-BB10-4427-AD26-233702D870BD}.Release|x64.Build.0 = Release|x64 {ABD1ABB5-BB10-4427-AD26-233702D870BD}.Release|x86.ActiveCfg = Release|Win32 {ABD1ABB5-BB10-4427-AD26-233702D870BD}.Release|x86.Build.0 = Release|Win32 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Debug|x64.ActiveCfg = Debug|x64 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Debug|x64.Build.0 = Debug|x64 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Debug|x86.ActiveCfg = Debug|Win32 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Debug|x86.Build.0 = Debug|Win32 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Release|x64.ActiveCfg = Release|x64 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Release|x64.Build.0 = Release|x64 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Release|x86.ActiveCfg = Release|Win32 + {B6C09563-1AF1-4226-A6B6-76802399CCDB}.Release|x86.Build.0 = Release|Win32 EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE diff --git a/Robocop UB/RobocopUB.vcxproj b/Robocop UB/RobocopUB.vcxproj new file mode 100644 index 0000000..85cb85b --- /dev/null +++ b/Robocop UB/RobocopUB.vcxproj @@ -0,0 +1,156 @@ + + + + + Debug + Win32 + + + Release + Win32 + + + Debug + x64 + + + Release + x64 + + + + 17.0 + Win32Proj + {B6C09563-1AF1-4226-A6B6-76802399CCDB} + RobocopUB + 10.0 + Robocop UB + + + + DynamicLibrary + true + v143 + Unicode + + + DynamicLibrary + false + v143 + true + Unicode + + + DynamicLibrary + true + v143 + Unicode + + + DynamicLibrary + false + v143 + true + Unicode + + + + + + + + + + + + + + + + + + + + + .addon + + + .addon + + + .addon + + + .addon + + + + Level3 + true + WIN32;_DEBUG;TLOU2_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + $(SolutionDir)external\reshade\include;$(SolutionDir)external\reshade\deps\imgui + MultiThreadedDLL + + + Windows + true + false + + + + + Level3 + true + true + true + WIN32;NDEBUG;TLOU2_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + $(SolutionDir)external\reshade\include;$(SolutionDir)external\reshade\deps\imgui + MultiThreadedDLL + + + Windows + true + false + + + + + Level3 + true + _DEBUG;TLOU2_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + $(SolutionDir)external\reshade\include;$(SolutionDir)external\reshade\deps\imgui + MultiThreadedDLL + + + Windows + true + false + + + + + Level3 + true + true + true + NDEBUG;TLOU2_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + $(SolutionDir)external\reshade\include;$(SolutionDir)external\reshade\deps\imgui + MultiThreadedDLL + + + Windows + true + false + + + + + + + + + \ No newline at end of file diff --git a/Robocop UB/dllmain.cpp b/Robocop UB/dllmain.cpp new file mode 100644 index 0000000..808a72f --- /dev/null +++ b/Robocop UB/dllmain.cpp @@ -0,0 +1,233 @@ +#define IMGUI_DISABLE_INCLUDE_IMCONFIG_H +#define IMGUI_HAS_DOCK 1 + +#include +#include +#include +#include +#include + +// Core game dll functions declarations +typedef void (*SetBoolFn)(bool, bool); +typedef void (*SetIntFn)(int); +typedef float (*GetFloatFn)(); + +static HMODULE fixLib = nullptr; +static SetBoolFn SetFixEnabled = nullptr; +static SetBoolFn SetFOVFixEnabled = nullptr; +static SetBoolFn SetAspectRatioFixEnabled = nullptr; +static SetBoolFn SetDOFFixEnabled = nullptr; +static SetIntFn SetFOV = nullptr; +static GetFloatFn GetFOVIn = nullptr; +static GetFloatFn GetCompensadedFOV = nullptr; +static GetFloatFn GetFOVOut = nullptr; + +// Plugin variables for checkboxes and sliders +static bool fov_fix_enabled = false; +static bool aspect_ratio_fix_enabled = false; +static bool DOF_fix_enabled = false; +static bool fix_enabled = false; +static int worldFOVvalue = 0; +static bool popup_Informations = false; + +// Plugin settings +const char* SETTINGS_FILE = "PluginSettings.ini"; +const char* GENERAL_FIX_SETTING = "GeneralFIX="; +const char* WORLD_FOV_FIX_SETTING = "WorldFOVFIX="; +const char* ASPECT_FIX_SETTING = "AspectFIX="; +const char* DOF_FIX_SETTING = "DOFFIX="; +const char* WORLD_FOV_SETTING = "WorldFOV="; +const char* FIX_VERSION = "1.0.1"; +const char* FIX_INFORMATIONS = "This fix allows to:\n - Control in game FOV.\n - Force aspect ratio.\n - Disable depth of field.\r\nFOV will be compensated when aspect fix is checked."; +const char* DONATION_URL = "https://buymeacoffee.com/k4sh44"; + +// Load and unload game core dll functions /!\ necessary +static void LoadFixDLL() +{ + if (fixLib) return; + fixLib = GetModuleHandleA("RobocopUBCore.asi"); + + if (!fixLib) { + // Optionnel: message d'erreur ou fallback + MessageBoxA(nullptr, "Impossible to load game core dll", "Erreur", MB_OK); + return; + } + SetFixEnabled = (SetBoolFn)GetProcAddress(fixLib, "SetFixEnabled"); + SetFOVFixEnabled = (SetBoolFn)GetProcAddress(fixLib, "SetFOVFixEnabled"); + SetAspectRatioFixEnabled = (SetBoolFn)GetProcAddress(fixLib, "SetAspectRatioFixEnabled"); + SetDOFFixEnabled = (SetBoolFn)GetProcAddress(fixLib, "SetDOFFixEnabled"); + SetFOV = (SetIntFn)GetProcAddress(fixLib, "SetFOV"); + GetFOVIn = (GetFloatFn)GetProcAddress(fixLib, "GetFOVIn"); + GetCompensadedFOV = (GetFloatFn)GetProcAddress(fixLib, "GetCompensatedFOV"); + GetFOVOut = (GetFloatFn)GetProcAddress(fixLib, "GetFOVOut");; + + // Apply initial values loaded from settings + if (SetFOV) SetFOV(worldFOVvalue); + if (SetFOVFixEnabled) SetFOVFixEnabled(fov_fix_enabled, true); + if (SetAspectRatioFixEnabled) SetAspectRatioFixEnabled(aspect_ratio_fix_enabled, true); + if (SetDOFFixEnabled) SetDOFFixEnabled(DOF_fix_enabled, true); + if (SetFixEnabled) SetFixEnabled(fix_enabled, true); +} + +// Addon functions +static void SaveSettings() +{ + std::ofstream file(SETTINGS_FILE); + if (file.is_open()) + { + file << GENERAL_FIX_SETTING << (fix_enabled ? "1" : "0") << "\n"; + file << WORLD_FOV_FIX_SETTING << (fov_fix_enabled ? "1" : "0") << "\n"; + file << ASPECT_FIX_SETTING << (aspect_ratio_fix_enabled ? "1" : "0") << "\n"; + file << DOF_FIX_SETTING << (DOF_fix_enabled ? "1" : "0") << "\n"; + file << WORLD_FOV_SETTING << worldFOVvalue << "\n"; + file.close(); + } +} + +static void LoadSettings() +{ + std::ifstream file(SETTINGS_FILE); + if (file.is_open()) + { + std::string line; + while (std::getline(file, line)) + { + if (line.find(GENERAL_FIX_SETTING) == 0) + { + std::string val = line.substr(strlen(GENERAL_FIX_SETTING)); + fix_enabled = (val == "1" || val == "true"); + } + else if (line.find(WORLD_FOV_FIX_SETTING) == 0) + { + std::string val = line.substr(strlen(WORLD_FOV_FIX_SETTING)); + fov_fix_enabled = (val == "1" || val == "true"); + } + else if (line.find(ASPECT_FIX_SETTING) == 0) + { + std::string val = line.substr(strlen(ASPECT_FIX_SETTING)); + aspect_ratio_fix_enabled = (val == "1" || val == "true"); + } + else if (line.find(DOF_FIX_SETTING) == 0) + { + std::string val = line.substr(strlen(DOF_FIX_SETTING)); + DOF_fix_enabled = (val == "1" || val == "true"); + } + else if (line.find(WORLD_FOV_SETTING) == 0) + worldFOVvalue = std::stoi(line.substr(strlen(WORLD_FOV_SETTING))); + } + file.close(); + } +} + +static void displayFixInformations() { + // Fix version + ImGui::Begin("Informations", &popup_Informations); // + ImGui::SetCursorPos(ImVec2(10, 36)); + ImGui::Text("Version : %s", FIX_VERSION); + ImGui::SetCursorPos(ImVec2(10, 76)); + ImGui::Text(FIX_INFORMATIONS); + ImGui::End(); +} + +// Initialize ImGui widgets for Reshade +static void on_overlay_draw(reshade::api::effect_runtime* runtime) +{ + ImGui::SetNextWindowPos(ImVec2(100, 200), ImGuiCond_Once); + ImGui::SetNextWindowSize(ImVec2(350, 150), ImGuiCond_Once); + + // Donation ? + ImGui::SetCursorPos(ImVec2(10, 36)); + ImGui::Text("Like my work ?"); + ImGui::SetCursorPos(ImVec2(130, 33)); + if (ImGui::Button("consider donation")) + { + ShellExecuteA(NULL, "open", DONATION_URL, NULL, NULL, SW_SHOWNORMAL); + } + + // Fix informations + ImGui::SetCursorPos(ImVec2(270, 33)); + if (ImGui::Button("Fix informations")) + popup_Informations = true; + + if (popup_Informations) + displayFixInformations(); + + // Generic fix + ImGui::SetCursorPos(ImVec2(10, 60)); + ImGui::BeginChild("AllFixesHeader", ImVec2(220, 0), false); // true = bordure + if (ImGui::CollapsingHeader("Enable fixes", ImGuiTreeNodeFlags_DefaultOpen)) { + ImGui::SetCursorPos(ImVec2(5, 30)); + if (ImGui::Checkbox("Fix enabled", &fix_enabled)) { + if (SetFixEnabled) SetFixEnabled(fix_enabled, false); + SaveSettings(); + } + } + ImGui::EndChild(); + + // FOV adjustment + ImGui::SetCursorPos(ImVec2(10, 120)); + ImGui::BeginChild("FOVHeader", ImVec2(220, 0), false); // true = bordure + if (ImGui::CollapsingHeader("In game additional FOV", ImGuiTreeNodeFlags_DefaultOpen)) { + ImGui::SetCursorPos(ImVec2(5, 30)); + ImGui::SetNextItemWidth(150.0f); + if (ImGui::SliderInt("", &worldFOVvalue, -20, 50)) {} + if (ImGui::IsItemDeactivatedAfterEdit()) { + if (SetFOV) SetFOV(worldFOVvalue); + SaveSettings(); + } + } + ImGui::EndChild(); + + // Individual fixes + ImGui::SetCursorPos(ImVec2(240, 60)); + ImGui::BeginChild("IndividualFixesHeader", ImVec2(250, 0), false); // true = bordure + if (ImGui::CollapsingHeader("Individual fixes", ImGuiTreeNodeFlags_DefaultOpen)) { + ImGui::SetCursorPos(ImVec2(5, 30)); + if (ImGui::Checkbox("Enable FOV Fix", &fov_fix_enabled)) { + if (SetFOVFixEnabled) SetFOVFixEnabled(fov_fix_enabled, false); + SaveSettings(); + } + + ImGui::SetCursorPos(ImVec2(5, 55)); + if (ImGui::Checkbox("Enable aspect Ratio Fix", &aspect_ratio_fix_enabled)) { + if (SetAspectRatioFixEnabled) SetAspectRatioFixEnabled(aspect_ratio_fix_enabled, false); + SaveSettings(); + } + + ImGui::SetCursorPos(ImVec2(5, 80)); + if (ImGui::Checkbox("Depth of field Fix", &DOF_fix_enabled)) { + if (SetDOFFixEnabled) SetDOFFixEnabled(DOF_fix_enabled, false); + SaveSettings(); + } + } + ImGui::EndChild(); + + // Fix status + ImGui::SetCursorPos(ImVec2(10, 200)); + ImGui::Text("=============== Fix informations ==============="); + ImGui::Text("FOV In: %.2f, Compensated : %.2f, Out : %.2f", GetFOVIn(), GetCompensadedFOV(), GetFOVOut()); +} + +// Main dll intrance +BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD ul_reason_for_call, LPVOID) +{ + switch (ul_reason_for_call) + { + case DLL_PROCESS_ATTACH: + if (!reshade::register_addon(hinstDLL)) + return FALSE; + + LoadSettings(); + //LoadFixDLL(); + reshade::register_overlay("Robocop Unfinished Business", &on_overlay_draw); + reshade::register_event( + [](reshade::api::effect_runtime* runtime) { + LoadFixDLL(); + }); + break; + case DLL_PROCESS_DETACH: + reshade::unregister_addon(hinstDLL); + break; + } + return TRUE; +} \ No newline at end of file