#include "trackball.h" #include #include #include "util/detour.h" #include "util/logging.h" #include "rawinput/rawinput.h" #include "games/io.h" #include "io.h" namespace games::ccj { bool MOUSE_TRACKBALL = false; bool MOUSE_TRACKBALL_USE_TOGGLE = false; uint8_t TRACKBALL_SENSITIVITY = 10; static HANDLE fakeHandle = (HANDLE)0xDEADBEEF; static HWND hWnd = nullptr; static WNDPROC wndProc = nullptr; static std::thread *tbThread = nullptr; static bool tbThreadRunning = false; static const wchar_t *fakeDeviceName = L"VID_1241&PID_1111"; static const wchar_t *windowName = L"ChaseProject"; static decltype(GetRawInputDeviceList) *GetRawInputDeviceList_orig = nullptr; static decltype(GetRawInputDeviceInfoW) *GetRawInputDeviceInfoW_orig = nullptr; static decltype(SetWindowLongPtrW) *SetWindowLongPtrW_orig = nullptr; static decltype(GetRawInputData) *GetRawInputData_orig = nullptr; static decltype(RegisterRawInputDevices) *RegisterRawInputDevices_orig = nullptr; static UINT WINAPI GetRawInputDeviceList_hook(PRAWINPUTDEVICELIST pRawInputDeviceList, PUINT puiNumDevices, UINT cbSize) { auto result = GetRawInputDeviceList_orig(pRawInputDeviceList, puiNumDevices, cbSize); if (result == 0xFFFFFFFF) return result; if (pRawInputDeviceList == NULL) { (*puiNumDevices)++; } else if (result < *puiNumDevices) { pRawInputDeviceList[result] = {fakeHandle, 0}; result++; } return result; } static UINT WINAPI GetRawInputDeviceInfoW_hook(HANDLE hDevice, UINT uiCommand, LPVOID pData, PUINT pcbSize) { if (hDevice != fakeHandle || uiCommand != RIDI_DEVICENAME) return GetRawInputDeviceInfoW_orig(hDevice, uiCommand, pData, pcbSize); const auto requiredLen = (wcslen(fakeDeviceName) + 1) * sizeof(wchar_t); if (*pcbSize < requiredLen) { *pcbSize = requiredLen; return 0xFFFFFFFF; } if (pData == NULL) { *pcbSize = requiredLen; return 0; } wcscpy((wchar_t*)pData, fakeDeviceName); return requiredLen; } static LONG_PTR WINAPI SetWindowLongPtrW_hook(HWND _hWnd, int nIndex, LONG_PTR dwNewLong) { wchar_t buffer[256]; if (nIndex != GWLP_WNDPROC || GetWindowTextW(_hWnd, buffer, 256) == 0 || !wcswcs(buffer, windowName)) return SetWindowLongPtrW_orig(_hWnd, nIndex, dwNewLong); hWnd = _hWnd; wndProc = (WNDPROC)dwNewLong; return SetWindowLongPtrW_orig(_hWnd, nIndex, dwNewLong); } static UINT WINAPI GetRawInputData_hook(HRAWINPUT hRawInput, UINT uiCommand, LPVOID pData, PUINT pcbSize, UINT cbSizeHeader) { if (hRawInput != fakeHandle) return GetRawInputData_orig(hRawInput, uiCommand, pData, pcbSize, cbSizeHeader); if (pData == NULL) { if (uiCommand == RID_HEADER) *pcbSize = sizeof(RAWINPUTHEADER); else *pcbSize = sizeof(RAWINPUT); return 0; } const RAWINPUTHEADER header = { RIM_TYPEMOUSE, sizeof(RAWINPUT), fakeHandle, 0 }; if (uiCommand == RID_HEADER) { if (*pcbSize < sizeof(RAWINPUTHEADER)) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return 0xFFFFFFFF; } *((RAWINPUTHEADER*)pData) = header; return sizeof(RAWINPUTHEADER); } else if (uiCommand == RID_INPUT) { if (*pcbSize < sizeof(RAWINPUT)) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return 0xFFFFFFFF; } RAWMOUSE rawMouse {}; if (MOUSE_TRACKBALL) { static bool active = false; static bool lastState = false; static std::chrono::time_point lastModified = std::chrono::steady_clock::now(); static std::chrono::milliseconds debounceDuration(100); auto currentTime = std::chrono::steady_clock::now(); bool pressed = GetAsyncKeyState(VK_RBUTTON) & 0x8000; bool focused = GetForegroundWindow() == hWnd; if (focused && MOUSE_TRACKBALL_USE_TOGGLE && pressed && (currentTime - lastModified > debounceDuration)) { active = !active; lastModified = currentTime; } if (focused && ((MOUSE_TRACKBALL_USE_TOGGLE && active) || (!MOUSE_TRACKBALL_USE_TOGGLE && pressed))) { POINT cursor; RECT client; GetClientRect(hWnd, &client); int sx = client.right - client.left; int sy = client.bottom - client.top; GetCursorPos(&cursor); ScreenToClient(hWnd, &cursor); static int lastX = cursor.x; static int lastY = cursor.y; if (!lastState) { lastX = cursor.x; lastY = cursor.y; lastState = true; } rawMouse.usFlags = MOUSE_MOVE_RELATIVE; rawMouse.lLastX = (int)((float)(cursor.x - lastX) * (float)TRACKBALL_SENSITIVITY / 20.0f); rawMouse.lLastY = (int)((float)(lastY - cursor.y) * (float)TRACKBALL_SENSITIVITY / 20.0f); bool updateCursor = false; if (cursor.x <= 0) { updateCursor = true; cursor.x = sx - 5; } else if (cursor.x >= sx - 1) { updateCursor = true; cursor.x = 5; } if (cursor.y <= 0) { updateCursor = true; cursor.y = sy - 5; } else if (cursor.y >= sy - 1) { updateCursor = true; cursor.y = 5; } lastX = cursor.x; lastY = cursor.y; if (updateCursor) { ClientToScreen(hWnd, &cursor); SetCursorPos(cursor.x, cursor.y); } } else if (lastState && !active) { lastState = false; } } else { rawMouse.usFlags = MOUSE_MOVE_RELATIVE; auto &analogs = get_analogs(); if (analogs[Analogs::Trackball_DX].isSet() || analogs[Analogs::Trackball_DY].isSet()) { float x = GameAPI::Analogs::getState(RI_MGR, analogs[Analogs::Trackball_DX]) * 2.0f - 1.0f; float y = GameAPI::Analogs::getState(RI_MGR, analogs[Analogs::Trackball_DY]) * 2.0f - 1.0f; rawMouse.lLastX = (long) (x * (float) TRACKBALL_SENSITIVITY); rawMouse.lLastY = (long) (-y * (float) TRACKBALL_SENSITIVITY); } auto &buttons = get_buttons(); if (GameAPI::Buttons::getState(RI_MGR, buttons[Buttons::Trackball_Up])) rawMouse.lLastY = TRACKBALL_SENSITIVITY; if (GameAPI::Buttons::getState(RI_MGR, buttons[Buttons::Trackball_Down])) rawMouse.lLastY = -TRACKBALL_SENSITIVITY; if (GameAPI::Buttons::getState(RI_MGR, buttons[Buttons::Trackball_Left])) rawMouse.lLastX = -TRACKBALL_SENSITIVITY; if (GameAPI::Buttons::getState(RI_MGR, buttons[Buttons::Trackball_Right])) rawMouse.lLastX = TRACKBALL_SENSITIVITY; } *((RAWINPUT*)pData) = { header, { rawMouse } }; return sizeof(RAWINPUT); } return 0xFFFFFFFF; } static BOOL WINAPI RegisterRawInputDevices_hook(PCRAWINPUTDEVICE pRawInputDevices, UINT uiNumDevices, UINT cbSize) { if (uiNumDevices == 2 && pRawInputDevices[1].usUsage == HID_USAGE_GENERIC_GAMEPAD) uiNumDevices = 1; return RegisterRawInputDevices_orig(pRawInputDevices, uiNumDevices, cbSize); } void trackball_hook_init() { // avoid double init static bool initialized = false; if (initialized) { return; } else { initialized = true; } // announce log_info("trackball", "init"); // user32 const auto user32Dll = "user32.dll"; detour::trampoline_try(user32Dll, "GetRawInputDeviceList", GetRawInputDeviceList_hook, &GetRawInputDeviceList_orig); detour::trampoline_try(user32Dll, "GetRawInputDeviceInfoW", GetRawInputDeviceInfoW_hook, &GetRawInputDeviceInfoW_orig); detour::trampoline_try(user32Dll, "SetWindowLongPtrW", SetWindowLongPtrW_hook, &SetWindowLongPtrW_orig); detour::trampoline_try(user32Dll, "GetRawInputData", GetRawInputData_hook, &GetRawInputData_orig); detour::trampoline_try(user32Dll, "RegisterRawInputDevices", RegisterRawInputDevices_hook, &RegisterRawInputDevices_orig); } void trackball_thread_start() { using namespace std::chrono_literals; tbThreadRunning = true; log_info("trackball", "thread start, use mouse: {}, toggle: {}", MOUSE_TRACKBALL, MOUSE_TRACKBALL_USE_TOGGLE); tbThread = new std::thread([&] { while (tbThreadRunning) { if (hWnd && wndProc) { wndProc(hWnd, WM_INPUT, RIM_INPUT, (LPARAM)fakeHandle); } if (!tbThreadRunning) break; std::this_thread::sleep_for(10ms); } }); } void trackball_thread_stop() { tbThreadRunning = false; if (tbThread) tbThread->join(); log_info("trackball", "thread stop"); delete tbThread; } }