-
-
Notifications
You must be signed in to change notification settings - Fork 10.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Input Latency in Window Event #8541
Comments
Very interesting. I was one 'enter key' press away from searching about 'Mouse Input Lag' issues when I noticed your post at the top. I'm currently trying to implement ImGui into Unreal Engine. While there are many existing implementations, they all use the same two options: either require the user to actively enable/disable interaction with ImGui vs. Unreal Engine or use an InputPreProcessor that causes issues when an ImGui widget is ZOrder-wise behind an Unreal Engine widget. I got it as far as having one Slate Widget per ImGui window, but I have a lot of issues with Positioning and Rendering while dragging said window. I'm using the I can somewhat understand that the visuals of the Widget lag behind, due to the RenderThread not being "instant", but the actual I could solve some of my problems by still using the InputPreProcessor, to feed ImGui the mouse position at all times, but I would much rather have the What I'm struggling with looks very similar to your attached video and I wasn't sure if it's ImGui being used within UE that naturally causes this, or if I'm doing something wrong, or if the 'issue' is within ImGui. That said, opening the interactive "ImGui Manual" shows the same problem when dragging a window. Mostly just posting here to share experiences and stay up-to-date with any future responses. |
Ha, sounds like your latency issue might be even more severe than what I described. Let me try to break this down based on your information and my limited UE knowledge. The message pipeline appears to be:
My issue stops at step 3. My solution is to inform the operating system at stage 2 that the window is about to be moved. The Win32 modal loop then takes over and automatically handles the window movement. This ensures that the relative position between the mouse drag point and the window remains consistent on every frame during movement, resulting in extremely low latency. Many applications indeed use this approach. |
Hey, thanks for the reply. I didn't mean to pull the problem away towards Unreal Engine-specifics, but I appreciate you trying to re-interpret the general issue. Throughout yesterday, I managed to resolve a bunch of issues that I had caused, which I had expected anyway. Let me try to describe what's going on on my end a bit more then. The following wall of text is Unreal Engine-specific. If whoever reads this doesn't care about Unreal Engine, just ignore it. It ultimately touches on the Input Latency problem, but this will be a bit of text before that happens. I packed it into a collapsed section so it doesn't spam too much. <-- Expand to read about ImGui + Unreal Engine and its reasons for Input Latency.How ImGui wants the order of actions per "frame" to work (I believe...)
Overview of Unreal Engine's
|
Hey. If the video is in 1:1 real-time and the rendering refresh rate is calculated at 75 FPS, a 2-frame delay would indeed feel "rubberbandy" as you described. You can estimate that if you move the mouse at a speed of about 1000 pixels per second, in 20ms your mouse could move approximately 20 pixels. From what I understand, if you don’t let the OS take over window movement, it’s impossible to make the window follow the mouse perfectly, and the reason for this is what I mentioned earlier. Enabling triple buffering and increasing the frame rate can significantly alleviate this issue. So, you still need to fix the issue with stopping the drag :D. Based on my experience, I suspect you might not be using the backend code provided by ImGui but instead modified it yourself. The reason for the drag signal loss is likely because you didn’t tell the OS to use SetCapture() in the mouse click callback and ReleaseCapture() when releasing it. Although I’m not very familiar with the mechanisms of Unreal Engine, I hope my response can be helpful to you. |
Yeah, which I probably won't be able to and won't bother. Maybe in the future.
That sounds reasonable, as I'm indeed not capturing the Mouse at that moment. I was going to resolve it by generally feeding ImGui the Position and MouseButtonUp events, instead of only if the Mouse is above the SWidget. Maybe capturing it will already resolve this. I will play around with both options. EDIT: For what it's worth, generally feeding ImGui those two events resolves this issue, but I will try with capturing the mouse too. EDIT2: Good call, using EDIT3: I'll leave your issue post alone then. The Latency Issue is still "the same", and it's good to know that I'm not alone with it, generally, but I also don't think I can figure this out, given Unreal Engine handles a large portion already. |
Version/Branch of Dear ImGui:
docking
Back-ends:
Win32 + Vulkan
Compiler, OS:
Windows + MSVC
Full config/build information:
Details:
When moving a floating windows, I can clearly perceive higher latency compared to many other applications I use (e.g., Chrome, Visual Studio, VS Code). Even with triple buffering enabled in the swap chain (FPS reaching 4000+), there remains latency when compared to other softwares.
In slow-motion video analysis, the relative position between the mouse cursor's clicked point and the window shifts during movement, while other applications maintain consistent alignment. This is because ImGui's event mechanism follows the 'collect -> process + respond' pattern, and ImGui's collection phase is already a response to the Win32 modal loop. A common approach is to notify the OS in the Win32 message callback that the window movement is starting.
Flowing is my experimental codes added in
ImGui_ImplWin32_WndProcHandler_PlatformWindow
function:After adding these lines of code, the window no longer feels laggy when moved! However, this introduces another issue: when the Win32 modal loop takes over the window movement event, the window's owning thread gets blocked. If the UI updates run on the same thread, all ImGui windows stop updating. To break this blocking, an alternative update mechanism is needed. I suggest using
SetTimer()
to make the window's thread trigger a callback function periodically to redraw the UI. With this approach, the window movement experience becomes much smoother.My experimental code does not take other features into account globally, so I don’t plan to submit a pull request. From a code maintenance perspective, would it be necessary to add an enum value (e.g., ImGuiConfigFlags_OSTakeOverWindowEvent) to enable this feature?
Screenshots/Video:
Desktop.2025.04.01.-.13.56.15.02.mp4
Minimal, Complete and Verifiable Example code:
ImGui::NewFrame();
....
ImGui::Begin("Circle");
if (ImGui::IsMouseDown(0))
ImGui::Text("IsMouseDown(0) = true");
else
ImGui::Text("IsMouseDown(0) = false");
ImDrawList * draw_list = ImGui::GetWindowDrawList();
draw_list->AddCircle(ImGui::GetMousePos(), 5.0f, IM_COL32_WHITE, 200);
draw_list->AddCircle(ImGui::GetMousePos(), 15.0f, IM_COL32_WHITE, 200);
draw_list->AddCircle(ImGui::GetMousePos(), 25.0f, IM_COL32_WHITE, 200);
draw_list->AddCircle(ImGui::GetMousePos(), 35.0f, IM_COL32_WHITE, 200);
ImGui::End();
...
ImGui::Render();
The text was updated successfully, but these errors were encountered: