23
23
// CHANGELOG
24
24
// (minor and older changes stripped away, please see git history for details)
25
25
// 2024-XX-XX: Platform: Added support for multiple windows via the ImGuiPlatformIO interface.
26
+ // 2024-11-21: [Docking] Fixed a crash when multiple processes are running with multi-viewports, caused by misusage of GetProp(). (#8162, #8069)
26
27
// 2024-10-28: [Docking] Rely on property stored inside HWND to retrieve context/viewport, should facilitate attempt to use this for parallel contexts. (#8069)
27
28
// 2024-09-16: [Docking] Inputs: fixed an issue where a viewport destroyed while clicking would hog mouse tracking and temporary lead to incorrect update of HoveredWindow. (#7971)
28
29
// 2024-07-08: Inputs: Fixed ImGuiMod_Super being mapped to VK_APPS instead of VK_LWIN||VK_RWIN. (#7768)
@@ -192,8 +193,10 @@ static bool ImGui_ImplWin32_InitEx(void* hwnd, bool platform_has_own_dc)
192
193
// Our mouse update function expect PlatformHandle to be filled for the main viewport
193
194
ImGuiViewport* main_viewport = ImGui::GetMainViewport ();
194
195
main_viewport->PlatformHandle = main_viewport->PlatformHandleRaw = (void *)bd->hWnd ;
196
+
197
+ // Be aware that GetPropA()/SetPropA() may be accessed from other processes.
198
+ // So as we store a pointer in IMGUI_CONTEXT we need to make sure we only call GetPropA() on windows owned by our process.
195
199
::SetPropA (bd->hWnd, " IMGUI_CONTEXT" , ImGui::GetCurrentContext());
196
- ::SetPropA (bd->hWnd, " IMGUI_VIEWPORT" , main_viewport);
197
200
ImGui_ImplWin32_InitMultiViewportSupport (platform_has_own_dc);
198
201
199
202
// Dynamically load XInput library
@@ -238,7 +241,6 @@ void ImGui_ImplWin32_Shutdown()
238
241
ImGuiIO& io = ImGui::GetIO ();
239
242
240
243
::SetPropA (bd->hWnd, " IMGUI_CONTEXT" , nullptr );
241
- ::SetPropA (bd->hWnd, " IMGUI_VIEWPORT" , nullptr );
242
244
ImGui_ImplWin32_ShutdownMultiViewportSupport ();
243
245
244
246
// Unload XInput library
@@ -319,17 +321,20 @@ static void ImGui_ImplWin32_UpdateKeyModifiers(ImGuiIO& io)
319
321
io.AddKeyEvent (ImGuiMod_Super, IsVkDown (VK_LWIN) || IsVkDown (VK_RWIN));
320
322
}
321
323
322
- static ImGuiViewport* ImGui_ImplWin32_FindViewportByPlatformHandle (HWND hwnd)
324
+ static ImGuiViewport* ImGui_ImplWin32_FindViewportByPlatformHandle (ImGuiPlatformIO& platform_io, HWND hwnd)
323
325
{
324
- // We could use ImGui::FindViewportByPlatformHandle() but GetPropA() gets us closer to parallel multi-contexts,
325
- // until we can pass an explicit context to FindViewportByPlatformHandle() .
326
+ // We cannot use ImGui::FindViewportByPlatformHandle() because it doesn't take a context.
327
+ // When called from ImGui_ImplWin32_WndProcHandler_PlatformWindow() we don't assume that context is bound .
326
328
// return ImGui::FindViewportByPlatformHandle((void*)hwnd);
327
- return (ImGuiViewport*)::GetPropA (hwnd, " IMGUI_VIEWPORT" );
329
+ for (ImGuiViewport* viewport : platform_io.Viewports )
330
+ if (viewport->PlatformHandle == hwnd)
331
+ return viewport;
332
+ return nullptr ;
328
333
}
329
334
330
335
// This code supports multi-viewports (multiple OS Windows mapped into different Dear ImGui viewports)
331
336
// Because of that, it is a little more complicated than your typical single-viewport binding code!
332
- static void ImGui_ImplWin32_UpdateMouseData (ImGuiIO& io)
337
+ static void ImGui_ImplWin32_UpdateMouseData (ImGuiIO& io, ImGuiPlatformIO& platform_io )
333
338
{
334
339
ImGui_ImplWin32_Data* bd = ImGui_ImplWin32_GetBackendData (io);
335
340
IM_ASSERT (bd->hWnd != 0 );
@@ -338,7 +343,7 @@ static void ImGui_ImplWin32_UpdateMouseData(ImGuiIO& io)
338
343
bool has_mouse_screen_pos = ::GetCursorPos (&mouse_screen_pos) != 0 ;
339
344
340
345
HWND focused_window = ::GetForegroundWindow ();
341
- const bool is_app_focused = (focused_window && (focused_window == bd->hWnd || ::IsChild (focused_window, bd->hWnd ) || ImGui_ImplWin32_FindViewportByPlatformHandle (focused_window)));
346
+ const bool is_app_focused = (focused_window && (focused_window == bd->hWnd || ::IsChild (focused_window, bd->hWnd ) || ImGui_ImplWin32_FindViewportByPlatformHandle (platform_io, focused_window)));
342
347
if (is_app_focused)
343
348
{
344
349
// (Optional) Set OS mouse position from Dear ImGui if requested (rarely used, only when io.ConfigNavMoveSetMousePos is enabled by user)
@@ -376,7 +381,7 @@ static void ImGui_ImplWin32_UpdateMouseData(ImGuiIO& io)
376
381
ImGuiID mouse_viewport_id = 0 ;
377
382
if (has_mouse_screen_pos)
378
383
if (HWND hovered_hwnd = ::WindowFromPoint (mouse_screen_pos))
379
- if (ImGuiViewport* viewport = ImGui_ImplWin32_FindViewportByPlatformHandle (hovered_hwnd))
384
+ if (ImGuiViewport* viewport = ImGui_ImplWin32_FindViewportByPlatformHandle (platform_io, hovered_hwnd))
380
385
mouse_viewport_id = viewport->ID ;
381
386
io.AddMouseViewportEvent (mouse_viewport_id);
382
387
}
@@ -475,6 +480,7 @@ void ImGui_ImplWin32_NewFrame()
475
480
ImGui_ImplWin32_Data* bd = ImGui_ImplWin32_GetBackendData ();
476
481
IM_ASSERT (bd != nullptr && " Context or backend not initialized? Did you call ImGui_ImplWin32_Init()?" );
477
482
ImGuiIO& io = ImGui::GetIO ();
483
+ ImGuiPlatformIO& platform_io = ImGui::GetPlatformIO ();
478
484
479
485
// Setup display size (every frame to accommodate for window resizing)
480
486
RECT rect = { 0 , 0 , 0 , 0 };
@@ -490,7 +496,7 @@ void ImGui_ImplWin32_NewFrame()
490
496
bd->Time = current_time;
491
497
492
498
// Update OS mouse position
493
- ImGui_ImplWin32_UpdateMouseData (io);
499
+ ImGui_ImplWin32_UpdateMouseData (io, platform_io );
494
500
495
501
// Process workarounds for known Windows key handling issues
496
502
ImGui_ImplWin32_ProcessKeyEventsWorkarounds (io);
@@ -1091,7 +1097,6 @@ static void ImGui_ImplWin32_CreateWindow(ImGuiViewport* viewport)
1091
1097
1092
1098
// Secondary viewports store their imgui context
1093
1099
::SetPropA (vd->Hwnd, " IMGUI_CONTEXT" , ImGui::GetCurrentContext());
1094
- ::SetPropA (vd->Hwnd, " IMGUI_VIEWPORT" , viewport);
1095
1100
}
1096
1101
1097
1102
static void ImGui_ImplWin32_DestroyWindow (ImGuiViewport* viewport)
@@ -1303,7 +1308,7 @@ static void ImGui_ImplWin32_OnChangedViewport(ImGuiViewport* viewport)
1303
1308
#endif
1304
1309
}
1305
1310
1306
- namespace ImGui { extern ImGuiIO& GetIOEx (ImGuiContext*); }
1311
+ namespace ImGui { extern ImGuiIO& GetIOEx (ImGuiContext*); extern ImGuiPlatformIO& GetPlatformIOEx (ImGuiContext*); }
1307
1312
1308
1313
static LRESULT CALLBACK ImGui_ImplWin32_WndProcHandler_PlatformWindow (HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
1309
1314
{
@@ -1313,10 +1318,11 @@ static LRESULT CALLBACK ImGui_ImplWin32_WndProcHandler_PlatformWindow(HWND hWnd,
1313
1318
return DefWindowProc (hWnd, msg, wParam, lParam); // unlike ImGui_ImplWin32_WndProcHandler() we are called directly by Windows, we can't just return 0.
1314
1319
1315
1320
ImGuiIO& io = ImGui::GetIOEx (ctx);
1321
+ ImGuiPlatformIO& platform_io = ImGui::GetPlatformIOEx (ctx);
1316
1322
LRESULT result = 0 ;
1317
1323
if (ImGui_ImplWin32_WndProcHandlerEx (hWnd, msg, wParam, lParam, io))
1318
1324
result = true ;
1319
- else if (ImGuiViewport* viewport = ImGui_ImplWin32_FindViewportByPlatformHandle (hWnd))
1325
+ else if (ImGuiViewport* viewport = ImGui_ImplWin32_FindViewportByPlatformHandle (platform_io, hWnd))
1320
1326
{
1321
1327
switch (msg)
1322
1328
{
0 commit comments