@ -959,7 +959,8 @@ static bool UpdateWindowManualResize(ImGuiWindow* window, const ImVe
static void RenderWindowOuterBorders ( ImGuiWindow * window ) ;
static void RenderWindowOuterBorders ( ImGuiWindow * window ) ;
static void RenderWindowDecorations ( ImGuiWindow * window , const ImRect & title_bar_rect , bool title_bar_is_highlight , bool handle_borders_and_resize_grips , int resize_grip_count , const ImU32 resize_grip_col [ 4 ] , float resize_grip_draw_size ) ;
static void RenderWindowDecorations ( ImGuiWindow * window , const ImRect & title_bar_rect , bool title_bar_is_highlight , bool handle_borders_and_resize_grips , int resize_grip_count , const ImU32 resize_grip_col [ 4 ] , float resize_grip_draw_size ) ;
static void RenderWindowTitleBarContents ( ImGuiWindow * window , const ImRect & title_bar_rect , const char * name , bool * p_open ) ;
static void RenderWindowTitleBarContents ( ImGuiWindow * window , const ImRect & title_bar_rect , const char * name , bool * p_open ) ;
static void EndFrameDrawDimmedBackgrounds ( ) ;
static void RenderDimmedBackgroundBehindWindow ( ImGuiWindow * window , ImU32 col ) ;
static void RenderDimmedBackgrounds ( ) ;
// Viewports
// Viewports
const ImGuiID IMGUI_VIEWPORT_DEFAULT_ID = 0x11111111 ; // Using an arbitrary constant instead of e.g. ImHashStr("ViewportDefault", 0); so it's easier to spot in the debugger. The exact value doesn't matter.
const ImGuiID IMGUI_VIEWPORT_DEFAULT_ID = 0x11111111 ; // Using an arbitrary constant instead of e.g. ImHashStr("ViewportDefault", 0); so it's easier to spot in the debugger. The exact value doesn't matter.
@ -4623,58 +4624,86 @@ static ImGuiWindow* FindFrontMostVisibleChildWindow(ImGuiWindow* window)
return window ;
return window ;
}
}
static void ImGui : : EndFrameDrawDimmedBackgrounds( )
static void ImGui : : RenderDimmedBackgroundBehindWindow( ImGuiWindow * window , ImU32 col )
{
{
ImGuiContext & g = * GImGui ;
if ( ( col & IM_COL32_A_MASK ) = = 0 )
return ;
// Draw modal whitening background on _other_ viewports than the one the modal is one
ImGuiViewportP * viewport = window - > Viewport ;
ImGuiWindow * modal_window = GetTopMostPopupModal ( ) ;
ImRect viewport_rect = viewport - > GetMainRect ( ) ;
const bool dim_bg_for_modal = ( modal_window ! = NULL ) ;
const bool dim_bg_for_window_list = ( g . NavWindowingTargetAnim ! = NULL ) ;
// Draw behind window by moving the draw command at the FRONT of the draw list
if ( dim_bg_for_modal | | dim_bg_for_window_list )
for ( int viewport_n = 0 ; viewport_n < g . Viewports . Size ; viewport_n + + )
{
{
ImGuiViewportP * viewport = g . Viewports [ viewport_n ] ;
ImDrawList * draw_list = window - > RootWindowDockTree - > DrawList ;
if ( modal_window & & viewport = = modal_window - > Viewport )
draw_list - > AddDrawCmd ( ) ;
continue ;
draw_list - > PushClipRect ( viewport_rect . Min - ImVec2 ( 1 , 1 ) , viewport_rect . Max + ImVec2 ( 1 , 1 ) , false ) ; // Ensure ImDrawCmd are not merged
if ( g . NavWindowingListWindow & & viewport = = g . NavWindowingListWindow - > Viewport )
draw_list - > AddRectFilled ( viewport_rect . Min , viewport_rect . Max , col ) ;
continue ;
ImDrawCmd cmd = draw_list - > CmdBuffer . back ( ) ;
if ( g . NavWindowingTargetAnim & & viewport = = g . NavWindowingTargetAnim - > Viewport )
IM_ASSERT ( cmd . ElemCount = = 6 ) ;
continue ;
draw_list - > CmdBuffer . pop_back ( ) ;
if ( viewport - > Window & & modal_window & & IsWindowAbove ( viewport - > Window , modal_window ) )
draw_list - > CmdBuffer . push_front ( cmd ) ;
continue ;
ImDrawList * draw_list = GetForegroundDrawList ( viewport ) ;
const ImU32 dim_bg_col = GetColorU32 ( dim_bg_for_modal ? ImGuiCol_ModalWindowDimBg : ImGuiCol_NavWindowingDimBg , g . DimBgRatio ) ;
draw_list - > AddRectFilled ( viewport - > Pos , viewport - > Pos + viewport - > Size , dim_bg_col ) ;
}
}
// Draw modal whitening background behind CTRL-TAB list
// Draw over sibling docking nodes in a same docking tree
if ( dim_bg_for_window_list & & g . NavWindowingTargetAnim - > Active)
if ( window - > RootWindow - > DockIsActive )
{
{
// Choose a draw list that will be front-most across all our children
// In the unlikely case that the window wasn't made active we can't rely on its drawlist and skip rendering all-together.
ImGuiWindow * window = g . NavWindowingTargetAnim ;
ImDrawList * draw_list = FindFrontMostVisibleChildWindow ( window - > RootWindowDockTree ) - > DrawList ;
ImDrawList * draw_list = FindFrontMostVisibleChildWindow ( window - > RootWindowDockTree ) - > DrawList ;
draw_list - > PushClipRectFullScreen ( ) ;
draw_list - > PushClipRect ( viewport_rect . Min , viewport_rect . Max , false ) ;
//if (window->RootWindowDockTree != window->RootWindow)
RenderRectFilledWithHole ( draw_list , window - > RootWindowDockTree - > Rect ( ) , window - > RootWindow - > Rect ( ) , col , 0.0f ) ; // window->RootWindowDockTree->WindowRounding);
draw_list - > PopClipRect ( ) ;
}
}
// Docking: draw modal whitening background on other nodes of a same dock tree
static void ImGui : : RenderDimmedBackgrounds ( )
// For CTRL+TAB within a docking node we need to render the dimming background in 8 steps
{
// (Because the root node renders the background in one shot, in order to avoid flickering when a child dock node is not submitted)
ImGuiContext & g = * GImGui ;
if ( window - > RootWindow - > DockIsActive )
ImGuiWindow * modal_window = GetTopMostAndVisiblePopupModal ( ) ;
if ( window - > RootWindowDockTree ! = window - > RootWindow )
const bool dim_bg_for_modal = ( modal_window ! = NULL ) ;
RenderRectFilledWithHole ( draw_list , window - > RootWindowDockTree - > Rect ( ) , window - > RootWindow - > Rect ( ) , GetColorU32 ( ImGuiCol_NavWindowingDimBg , g . DimBgRatio ) , g . Style . WindowRounding ) ;
const bool dim_bg_for_window_list = ( g . NavWindowingTargetAnim ! = NULL & & g . NavWindowingTargetAnim - > Active ) ;
if ( ! dim_bg_for_modal & & ! dim_bg_for_window_list )
return ;
// Draw navigation selection/windowing rectangle border
ImGuiViewport * viewports_already_dimmed [ 2 ] = { NULL , NULL } ;
float rounding = ImMax ( window - > WindowRounding , g . Style . WindowRounding ) ;
if ( dim_bg_for_modal )
ImRect bb = window - > Rect ( ) ;
bb . Expand ( g . FontSize ) ;
if ( ! window - > Viewport - > GetMainRect ( ) . Contains ( bb ) ) // If a window fits the entire viewport, adjust its highlight inward
{
{
bb . Expand ( - g . FontSize - 1.0f ) ;
// Draw dimming behind modal
rounding = window - > WindowRounding ;
RenderDimmedBackgroundBehindWindow ( modal_window , GetColorU32 ( ImGuiCol_ModalWindowDimBg , g . DimBgRatio ) ) ;
viewports_already_dimmed [ 0 ] = modal_window - > Viewport ;
}
}
draw_list - > AddRect ( bb . Min , bb . Max , GetColorU32 ( ImGuiCol_NavWindowingHighlight , g . NavWindowingHighlightAlpha ) , rounding , 0 , 3.0f ) ;
else if ( dim_bg_for_window_list )
draw_list - > PopClipRect ( ) ;
{
// Draw dimming behind CTRL+Tab target window and behind CTRL+Tab UI window
RenderDimmedBackgroundBehindWindow ( g . NavWindowingTargetAnim , GetColorU32 ( ImGuiCol_NavWindowingDimBg , g . DimBgRatio ) ) ;
if ( g . NavWindowingListWindow ! = NULL & & g . NavWindowingListWindow - > Viewport ! = g . NavWindowingTargetAnim - > Viewport )
RenderDimmedBackgroundBehindWindow ( g . NavWindowingListWindow , GetColorU32 ( ImGuiCol_NavWindowingDimBg , g . DimBgRatio ) ) ;
viewports_already_dimmed [ 0 ] = g . NavWindowingTargetAnim - > Viewport ;
viewports_already_dimmed [ 1 ] = g . NavWindowingListWindow ? g . NavWindowingListWindow - > Viewport : NULL ;
// Draw border around CTRL+Tab target window
ImGuiWindow * window = g . NavWindowingTargetAnim ;
ImGuiViewport * viewport = window - > Viewport ;
float distance = g . FontSize ;
ImRect bb = window - > Rect ( ) ;
bb . Expand ( distance ) ;
if ( bb . GetWidth ( ) > = viewport - > Size . x & & bb . GetHeight ( ) > = viewport - > Size . y )
bb . Expand ( - distance - 1.0f ) ; // If a window fits the entire viewport, adjust its highlight inward
window - > DrawList - > PushClipRect ( viewport - > Pos , viewport - > Pos + viewport - > Size ) ;
window - > DrawList - > AddRect ( bb . Min , bb . Max , GetColorU32 ( ImGuiCol_NavWindowingHighlight , g . NavWindowingHighlightAlpha ) , window - > WindowRounding , 0 , 3.0f ) ;
window - > DrawList - > PopClipRect ( ) ;
}
// Draw dimming background on _other_ viewports than the ones our windows are in
for ( int viewport_n = 0 ; viewport_n < g . Viewports . Size ; viewport_n + + )
{
ImGuiViewportP * viewport = g . Viewports [ viewport_n ] ;
if ( viewport = = viewports_already_dimmed [ 0 ] | | viewport = = viewports_already_dimmed [ 1 ] )
continue ;
if ( modal_window & & viewport - > Window & & IsWindowAbove ( viewport - > Window , modal_window ) )
continue ;
ImDrawList * draw_list = GetForegroundDrawList ( viewport ) ;
const ImU32 dim_bg_col = GetColorU32 ( dim_bg_for_modal ? ImGuiCol_ModalWindowDimBg : ImGuiCol_NavWindowingDimBg , g . DimBgRatio ) ;
draw_list - > AddRectFilled ( viewport - > Pos , viewport - > Pos + viewport - > Size , dim_bg_col ) ;
}
}
}
}
@ -4740,9 +4769,6 @@ void ImGui::EndFrame()
// Initiate moving window + handle left-click and right-click focus
// Initiate moving window + handle left-click and right-click focus
UpdateMouseMovingWindowEndFrame ( ) ;
UpdateMouseMovingWindowEndFrame ( ) ;
// Draw modal/window whitening backgrounds
EndFrameDrawDimmedBackgrounds ( ) ;
// Update user-facing viewport list (g.Viewports -> g.PlatformIO.Viewports after filtering out some)
// Update user-facing viewport list (g.Viewports -> g.PlatformIO.Viewports after filtering out some)
UpdateViewportsEndFrame ( ) ;
UpdateViewportsEndFrame ( ) ;
@ -4785,6 +4811,7 @@ void ImGui::Render()
if ( g . FrameCountEnded ! = g . FrameCount )
if ( g . FrameCountEnded ! = g . FrameCount )
EndFrame ( ) ;
EndFrame ( ) ;
const bool first_render_of_frame = ( g . FrameCountRendered ! = g . FrameCount ) ;
g . FrameCountRendered = g . FrameCount ;
g . FrameCountRendered = g . FrameCount ;
g . IO . MetricsRenderWindows = 0 ;
g . IO . MetricsRenderWindows = 0 ;
@ -4814,6 +4841,10 @@ void ImGui::Render()
if ( windows_to_render_top_most [ n ] & & IsWindowActiveAndVisible ( windows_to_render_top_most [ n ] ) ) // NavWindowingTarget is always temporarily displayed as the top-most window
if ( windows_to_render_top_most [ n ] & & IsWindowActiveAndVisible ( windows_to_render_top_most [ n ] ) ) // NavWindowingTarget is always temporarily displayed as the top-most window
AddRootWindowToDrawData ( windows_to_render_top_most [ n ] ) ;
AddRootWindowToDrawData ( windows_to_render_top_most [ n ] ) ;
// Draw modal/window whitening backgrounds
if ( first_render_of_frame )
RenderDimmedBackgrounds ( ) ;
ImVec2 mouse_cursor_offset , mouse_cursor_size , mouse_cursor_uv [ 4 ] ;
ImVec2 mouse_cursor_offset , mouse_cursor_size , mouse_cursor_uv [ 4 ] ;
if ( g . IO . MouseDrawCursor & & g . MouseCursor ! = ImGuiMouseCursor_None )
if ( g . IO . MouseDrawCursor & & g . MouseCursor ! = ImGuiMouseCursor_None )
g . IO . Fonts - > GetMouseCursorTexData ( g . MouseCursor , & mouse_cursor_offset , & mouse_cursor_size , & mouse_cursor_uv [ 0 ] , & mouse_cursor_uv [ 2 ] ) ;
g . IO . Fonts - > GetMouseCursorTexData ( g . MouseCursor , & mouse_cursor_offset , & mouse_cursor_size , & mouse_cursor_uv [ 0 ] , & mouse_cursor_uv [ 2 ] ) ;
@ -4827,7 +4858,7 @@ void ImGui::Render()
// Draw software mouse cursor if requested by io.MouseDrawCursor flag
// Draw software mouse cursor if requested by io.MouseDrawCursor flag
// (note we scale cursor by current viewport/monitor, however Windows 10 for its own hardware cursor seems to be using a different scale factor)
// (note we scale cursor by current viewport/monitor, however Windows 10 for its own hardware cursor seems to be using a different scale factor)
if ( mouse_cursor_size . x > 0.0f & & mouse_cursor_size . y > 0.0f )
if ( mouse_cursor_size . x > 0.0f & & mouse_cursor_size . y > 0.0f & & first_render_of_frame )
{
{
float scale = g . Style . MouseCursorScale * viewport - > DpiScale ;
float scale = g . Style . MouseCursorScale * viewport - > DpiScale ;
if ( viewport - > GetMainRect ( ) . Overlaps ( ImRect ( g . IO . MousePos , g . IO . MousePos + ImVec2 ( mouse_cursor_size . x + 2 , mouse_cursor_size . y + 2 ) * scale ) ) )
if ( viewport - > GetMainRect ( ) . Overlaps ( ImRect ( g . IO . MousePos , g . IO . MousePos + ImVec2 ( mouse_cursor_size . x + 2 , mouse_cursor_size . y + 2 ) * scale ) ) )
@ -6723,20 +6754,6 @@ bool ImGui::Begin(const char* name, bool* p_open, ImGuiWindowFlags flags)
window - > DrawList - > PushTextureID ( g . Font - > ContainerAtlas - > TexID ) ;
window - > DrawList - > PushTextureID ( g . Font - > ContainerAtlas - > TexID ) ;
PushClipRect ( host_rect . Min , host_rect . Max , false ) ;
PushClipRect ( host_rect . Min , host_rect . Max , false ) ;
// Draw modal or window list full viewport dimming background (for other viewports we'll render them in EndFrame)
ImGuiWindow * window_window_list = g . NavWindowingListWindow ;
const bool dim_bg_for_modal = ( flags & ImGuiWindowFlags_Modal ) & & window = = GetTopMostPopupModal ( ) & & window - > HiddenFramesCannotSkipItems < = 0 ;
const bool dim_bg_for_window_list = g . NavWindowingTargetAnim & & ( ( window = = g . NavWindowingTargetAnim - > RootWindowDockTree ) | | ( window = = window_window_list & & window_window_list - > Viewport ! = g . NavWindowingTargetAnim - > Viewport ) ) ;
if ( dim_bg_for_modal | | dim_bg_for_window_list )
{
const ImU32 dim_bg_col = GetColorU32 ( dim_bg_for_modal ? ImGuiCol_ModalWindowDimBg : ImGuiCol_NavWindowingDimBg , g . DimBgRatio ) ;
if ( window - > DockIsActive | | ( flags & ImGuiWindowFlags_DockNodeHost ) )
window - > DrawList - > ChannelsSetCurrent ( 0 ) ;
window - > DrawList - > AddRectFilled ( viewport_rect . Min , viewport_rect . Max , dim_bg_col ) ;
if ( window - > DockIsActive | | ( flags & ImGuiWindowFlags_DockNodeHost ) )
window - > DrawList - > ChannelsSetCurrent ( 1 ) ;
}
// Child windows can render their decoration (bg color, border, scrollbars, etc.) within their parent to save a draw call (since 1.71)
// Child windows can render their decoration (bg color, border, scrollbars, etc.) within their parent to save a draw call (since 1.71)
// When using overlapping child windows, this will break the assumption that child z-order is mapped to submission order.
// When using overlapping child windows, this will break the assumption that child z-order is mapped to submission order.
// FIXME: User code may rely on explicit sorting of overlapping child window and would need to disable this somehow. Please get in contact if you are affected (github #4493)
// FIXME: User code may rely on explicit sorting of overlapping child window and would need to disable this somehow. Please get in contact if you are affected (github #4493)
@ -8985,6 +9002,16 @@ ImGuiWindow* ImGui::GetTopMostPopupModal()
return NULL ;
return NULL ;
}
}
ImGuiWindow * ImGui : : GetTopMostAndVisiblePopupModal ( )
{
ImGuiContext & g = * GImGui ;
for ( int n = g . OpenPopupStack . Size - 1 ; n > = 0 ; n - - )
if ( ImGuiWindow * popup = g . OpenPopupStack . Data [ n ] . Window )
if ( ( popup - > Flags & ImGuiWindowFlags_Modal ) & & IsWindowActiveAndVisible ( popup ) )
return popup ;
return NULL ;
}
void ImGui : : OpenPopup ( const char * str_id , ImGuiPopupFlags popup_flags )
void ImGui : : OpenPopup ( const char * str_id , ImGuiPopupFlags popup_flags )
{
{
ImGuiContext & g = * GImGui ;
ImGuiContext & g = * GImGui ;