#include "pch.h" #include "hook.h" #include "../depend/kiero/kiero.h" #include "../depend/kiero/minhook/MinHook.h" #include "../depend/imgui/imgui_impl_dx9.h" #include "../depend/imgui/imgui_impl_dx11.h" #include "../depend/imgui/imgui_impl_win32.h" LRESULT Hook::WndProc(const HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { ImGui_ImplWin32_WndProcHandler(hWnd, uMsg, wParam, lParam); if (ImGui::GetIO().WantTextInput) { #ifdef GTASA Call<0x53F1E0>(); // CPad::ClearKeyboardHistory #endif return 1; } return CallWindowProc(oWndProc, hWnd, uMsg, wParam, lParam); } HRESULT Hook::Reset(IDirect3DDevice9* pDevice, D3DPRESENT_PARAMETERS* pPresentationParameters) { ImGui_ImplDX9_InvalidateDeviceObjects(); return oReset(pDevice, pPresentationParameters); } void Hook::RenderFrame(void* ptr) { if (!ImGui::GetCurrentContext()) { return; } ImGuiIO& io = ImGui::GetIO(); static bool bInit = false; if (bInit) { ShowMouse(m_bShowMouse); // handle window scaling here static ImVec2 fScreenSize = ImVec2(-1, -1); ImVec2 size(screen::GetScreenWidth(), screen::GetScreenHeight()); if (fScreenSize.x != size.x && fScreenSize.y != size.y) { int fontSize = static_cast(size.y / 54.85f); // manually tested io.FontDefault = io.Fonts->AddFontFromFileTTF("C:/Windows/Fonts/trebucbd.ttf", fontSize); io.Fonts->Build(); if (gRenderer == Render_DirectX9) { ImGui_ImplDX9_InvalidateDeviceObjects(); } else { ImGui_ImplDX11_InvalidateDeviceObjects(); } ImGuiStyle* style = &ImGui::GetStyle(); float scaleX = size.x / 1366.0f; float scaleY = size.y / 768.0f; style->FramePadding = ImVec2(5 * scaleX, 3 * scaleY); style->ItemSpacing = ImVec2(8 * scaleX, 4 * scaleY); style->ScrollbarSize = 12 * scaleX; style->IndentSpacing = 20 * scaleX; style->ItemInnerSpacing = ImVec2(4 * scaleX, 4 * scaleY); fScreenSize = size; } ImGui_ImplWin32_NewFrame(); if (gRenderer == Render_DirectX9) { ImGui_ImplDX9_NewFrame(); } else { ImGui_ImplDX11_NewFrame(); } ImGui::NewFrame(); if (windowCallback != nullptr) { windowCallback(); } ImGui::EndFrame(); ImGui::Render(); if (gRenderer == Render_DirectX9) { ImGui_ImplDX9_RenderDrawData(ImGui::GetDrawData()); } else { ImGui_ImplDX11_RenderDrawData(ImGui::GetDrawData()); } } else { bInit = true; ImGui::CreateContext(); ImGuiStyle& style = ImGui::GetStyle(); ImGui_ImplWin32_Init(RsGlobal.ps->window); #ifdef GTASA // shift trigger fix patch::Nop(0x00531155, 5); #endif if (gRenderer == Render_DirectX9) { ImGui_ImplDX9_Init(reinterpret_cast(ptr)); } else { // for dx11 device ptr is swapchain reinterpret_cast(ptr)->GetDevice(__uuidof(ID3D11Device), &ptr); ID3D11DeviceContext* context; reinterpret_cast(ptr)->GetImmediateContext(&context); ImGui_ImplDX11_Init(reinterpret_cast(ptr), context); } ImGui_ImplWin32_EnableDpiAwareness(); io.IniFilename = nullptr; io.LogFilename = nullptr; io.ConfigFlags |= ImGuiConfigFlags_NavEnableGamepad; style.WindowTitleAlign = ImVec2(0.5, 0.5); oWndProc = (WNDPROC)SetWindowLongPtr(RsGlobal.ps->window, GWL_WNDPROC, (LRESULT)WndProc); } } HRESULT Hook::Dx9Handler(IDirect3DDevice9* pDevice) { RenderFrame(pDevice); return oEndScene(pDevice); } HRESULT Hook::Dx11Handler(IDXGISwapChain* pSwapChain, UINT SyncInterval, UINT Flags) { RenderFrame(pSwapChain); return oPresent11(pSwapChain, SyncInterval, Flags); } void Hook::ShowMouse(bool state) { // Disable player controls for controllers #if GTASA bool bMouseDisabled = false; if (patch::Get(0xBA6818) && (m_bShowMouse || bMouseDisabled)) { CPlayerPed *player = FindPlayerPed(); if (player && player->GetPadFromPlayer()) { if (m_bShowMouse) { bMouseDisabled = true; player->GetPadFromPlayer()->DisablePlayerControls = true; } else { bMouseDisabled = false; player->GetPadFromPlayer()->DisablePlayerControls = false; } } } #endif if (m_bMouseVisibility != m_bShowMouse) { ImGui::GetIO().MouseDrawCursor = state; #ifdef GTASA Hook::ApplyMouseFix(); // Reapply the patches #elif GTAVC if (m_bShowMouse) { patch::SetUChar(0x6020A0, 0xC3); // psSetMousePos patch::Nop(0x4AB6CA, 5); // don't call CPad::UpdateMouse() } else { patch::SetUChar(0x6020A0, 0x53); patch::SetRaw(0x4AB6CA, (char*)"\xE8\x51\x21\x00\x00", 5); } #endif CPad::NewMouseControllerState.X = 0; CPad::NewMouseControllerState.Y = 0; CPad::ClearMouseHistory(); CPad::UpdatePads(); m_bMouseVisibility = m_bShowMouse; } } Hook::Hook() { ImGui::CreateContext(); // Nvidia Overlay crash fix if (init(kiero::RenderType::D3D9) == kiero::Status::Success) { gRenderer = Render_DirectX9; kiero::bind(16, (void**)&oReset, Reset); kiero::bind(42, (void**)&oEndScene, Dx9Handler); } else { // gtaRenderHook if (init(kiero::RenderType::D3D11) == kiero::Status::Success) { gRenderer = Render_DirectX11; kiero::bind(8, (void**)&oPresent11, Dx11Handler); } } } Hook::~Hook() { SetWindowLongPtr(RsGlobal.ps->window, GWL_WNDPROC, (LRESULT)oWndProc); ImGui_ImplDX9_Shutdown(); ImGui_ImplWin32_Shutdown(); ImGui::DestroyContext(); kiero::shutdown(); } #ifdef GTASA struct Mouse { unsigned int x, y; unsigned int wheelDelta; unsigned char buttons[8]; }; struct MouseInfo { int x, y, wheelDelta; } mouseInfo; static BOOL __stdcall _SetCursorPos(int X, int Y) { if (Hook::m_bShowMouse || GetActiveWindow() != RsGlobal.ps->window) { return 1; } mouseInfo.x = X; mouseInfo.y = Y; return SetCursorPos(X, Y); } static LRESULT __stdcall _DispatchMessage(MSG* lpMsg) { if (lpMsg->message == WM_MOUSEWHEEL && !Hook::m_bShowMouse) { mouseInfo.wheelDelta += GET_WHEEL_DELTA_WPARAM(lpMsg->wParam); } return DispatchMessageA(lpMsg); } static int _cdecl _GetMouseState(Mouse* pMouse) { if (Hook::m_bShowMouse) { return -1; } struct tagPOINT Point; pMouse->x = 0; pMouse->y = 0; pMouse->wheelDelta = mouseInfo.wheelDelta; GetCursorPos(&Point); if (mouseInfo.x >= 0) { pMouse->x = int(1.6f*(Point.x - mouseInfo.x)); // hacky fix } if (mouseInfo.y >= 0) { pMouse->y = int(Point.y - mouseInfo.y); } mouseInfo.wheelDelta = 0; pMouse->buttons[0] = (GetAsyncKeyState(1) >> 8); pMouse->buttons[1] = (GetAsyncKeyState(2) >> 8); pMouse->buttons[2] = (GetAsyncKeyState(4) >> 8); pMouse->buttons[3] = (GetAsyncKeyState(5) >> 8); pMouse->buttons[4] = (GetAsyncKeyState(6) >> 8); return 0; } void Hook::ApplyMouseFix() { patch::ReplaceFunctionCall(0x53F417, _GetMouseState); patch::Nop(0x57C59B, 1); patch::ReplaceFunctionCall(0x57C59C, _SetCursorPos); patch::Nop(0x81E5D4, 1); patch::ReplaceFunctionCall(0x81E5D5, _SetCursorPos); patch::Nop(0x74542D, 1); patch::ReplaceFunctionCall(0x74542E, _SetCursorPos); patch::Nop(0x748A7C, 1); patch::ReplaceFunctionCall(0x748A7D, _DispatchMessage); patch::SetChar(0x746A08, 32); // diMouseOffset patch::SetChar(0x746A58, 32); // diDeviceoffset } #endif