@ -918,7 +918,7 @@ static void CheckStacksSize(ImGuiWindow* window, bool write);
static ImVec2 CalcNextScrollFromScrollTargetAndClamp ( ImGuiWindow * window , bool snap_on_edges ) ;
static void AddDrawListToDrawData ( ImVector < ImDrawList * > * out_list , ImDrawList * draw_list ) ;
static void AddWindowToSort ed Buffer( ImVector < ImGuiWindow * > * out_sorted_windows , ImGuiWindow * window ) ;
static void AddWindowToSort Buffer( ImVector < ImGuiWindow * > * out_sorted_windows , ImGuiWindow * window ) ;
// Settings
static void * SettingsHandlerWindow_ReadOpen ( ImGuiContext * , ImGuiSettingsHandler * , const char * name ) ;
@ -3385,6 +3385,7 @@ void ImGui::NewFrame()
g . NavIdTabCounter = INT_MAX ;
// Mark all windows as not visible
IM_ASSERT ( g . WindowsFocusOrder . Size = = g . Windows . Size ) ;
for ( int i = 0 ; i ! = g . Windows . Size ; i + + )
{
ImGuiWindow * window = g . Windows [ i ] ;
@ -3395,9 +3396,8 @@ void ImGui::NewFrame()
}
// Closing the focused window restore focus to the first active root window in descending z-order
// FIXME: Because z-order and nav-order are correlated, this will focus the wrong window if we are part of a hierarchy with NoBringToFrontOnFocus flag.
if ( g . NavWindow & & ! g . NavWindow - > WasActive )
Focus FrontMostActive WindowIgnoringOne( NULL ) ;
Focus Previous WindowIgnoringOne( NULL ) ;
// No window should be open at the beginning of the frame.
// But in order to allow the user to call NewFrame() multiple times without calling Render(), we are doing an explicit clear.
@ -3475,6 +3475,7 @@ void ImGui::Shutdown(ImGuiContext* context)
for ( int i = 0 ; i < g . Windows . Size ; i + + )
IM_DELETE ( g . Windows [ i ] ) ;
g . Windows . clear ( ) ;
g . WindowsFocusOrder . clear ( ) ;
g . WindowsSortBuffer . clear ( ) ;
g . CurrentWindow = NULL ;
g . CurrentWindowStack . clear ( ) ;
@ -3524,7 +3525,7 @@ static int IMGUI_CDECL ChildWindowComparer(const void* lhs, const void* rhs)
return ( a - > BeginOrderWithinParent - b - > BeginOrderWithinParent ) ;
}
static void AddWindowToSort ed Buffer( ImVector < ImGuiWindow * > * out_sorted_windows , ImGuiWindow * window )
static void AddWindowToSort Buffer( ImVector < ImGuiWindow * > * out_sorted_windows , ImGuiWindow * window )
{
out_sorted_windows - > push_back ( window ) ;
if ( window - > Active )
@ -3536,7 +3537,7 @@ static void AddWindowToSortedBuffer(ImVector<ImGuiWindow*>* out_sorted_windows,
{
ImGuiWindow * child = window - > DC . ChildWindows [ i ] ;
if ( child - > Active )
AddWindowToSort ed Buffer( out_sorted_windows , child ) ;
AddWindowToSort Buffer( out_sorted_windows , child ) ;
}
}
}
@ -3807,7 +3808,7 @@ void ImGui::EndFrame()
ImGuiWindow * window = g . Windows [ i ] ;
if ( window - > Active & & ( window - > Flags & ImGuiWindowFlags_ChildWindow ) ) // if a child is active its parent will add it
continue ;
AddWindowToSort ed Buffer( & g . WindowsSortBuffer , window ) ;
AddWindowToSort Buffer( & g . WindowsSortBuffer , window ) ;
}
IM_ASSERT ( g . Windows . Size = = g . WindowsSortBuffer . Size ) ; // we done something wrong
@ -4503,8 +4504,9 @@ static ImGuiWindow* CreateNewWindow(const char* name, ImVec2 size, ImGuiWindowFl
window - > AutoFitOnlyGrows = ( window - > AutoFitFramesX > 0 ) | | ( window - > AutoFitFramesY > 0 ) ;
}
g . WindowsFocusOrder . push_back ( window ) ;
if ( flags & ImGuiWindowFlags_NoBringToFrontOnFocus )
g . Windows . insert( g . Windows . begin ( ) , window ) ; // Quite slow but rare and only once
g . Windows . push_front( window ) ; // Quite slow but rare and only once
else
g . Windows . push_back ( window ) ;
return window ;
@ -5621,7 +5623,21 @@ void ImGui::End()
SetCurrentViewport ( g . CurrentWindow , g . CurrentWindow - > Viewport ) ;
}
void ImGui : : BringWindowToFront ( ImGuiWindow * window )
void ImGui : : BringWindowToFocusFront ( ImGuiWindow * window )
{
ImGuiContext & g = * GImGui ;
if ( g . WindowsFocusOrder . back ( ) = = window )
return ;
for ( int i = g . WindowsFocusOrder . Size - 2 ; i > = 0 ; i - - ) // We can ignore the front most window
if ( g . WindowsFocusOrder [ i ] = = window )
{
memmove ( & g . WindowsFocusOrder [ i ] , & g . WindowsFocusOrder [ i + 1 ] , ( size_t ) ( g . WindowsFocusOrder . Size - i - 1 ) * sizeof ( ImGuiWindow * ) ) ;
g . WindowsFocusOrder [ g . WindowsFocusOrder . Size - 1 ] = window ;
break ;
}
}
void ImGui : : BringWindowToDisplayFront ( ImGuiWindow * window )
{
ImGuiContext & g = * GImGui ;
ImGuiWindow * current_front_window = g . Windows . back ( ) ;
@ -5630,13 +5646,13 @@ void ImGui::BringWindowToFront(ImGuiWindow* window)
for ( int i = g . Windows . Size - 2 ; i > = 0 ; i - - ) // We can ignore the front most window
if ( g . Windows [ i ] = = window )
{
g. Windows . erase ( g . Windows . Data + i ) ;
g . Windows .push_back ( window ) ;
memmove( & g . Windows [ i ] , & g . Windows [ i + 1 ] , ( size_t ) ( g . Windows . Size - i - 1 ) * sizeof ( ImGuiWindow * ) ) ;
g . Windows [g . Windows . Size - 1 ] = window ;
break ;
}
}
void ImGui : : BringWindowTo Back( ImGuiWindow * window )
void ImGui : : BringWindowTo Display Back( ImGuiWindow * window )
{
ImGuiContext & g = * GImGui ;
if ( g . Windows [ 0 ] = = window )
@ -5681,21 +5697,25 @@ void ImGui::FocusWindow(ImGuiWindow* window)
ClearActiveID ( ) ;
// Bring to front
BringWindowToFocusFront ( window ) ;
if ( ! ( window - > Flags & ImGuiWindowFlags_NoBringToFrontOnFocus ) )
BringWindowTo Front( window ) ;
BringWindowTo Display Front( window ) ;
}
void ImGui : : Focus FrontMostActive WindowIgnoringOne( ImGuiWindow * ignore_window )
void ImGui : : Focus Previous WindowIgnoringOne( ImGuiWindow * ignore_window )
{
ImGuiContext & g = * GImGui ;
for ( int i = g . Windows . Size - 1 ; i > = 0 ; i - - )
if ( g . Windows [ i ] ! = ignore_window & & g . Windows [ i ] - > WasActive & & ! ( g . Windows [ i ] - > Flags & ImGuiWindowFlags_ChildWindow ) )
for ( int i = g . WindowsFocusOrder . Size - 1 ; i > = 0 ; i - - )
{
ImGuiWindow * focus_window = NavRestoreLastChildNavWindow ( g . Windows [ i ] ) ;
ImGuiWindow * window = g . WindowsFocusOrder [ i ] ;
if ( window ! = ignore_window & & window - > WasActive & & ! ( window - > Flags & ImGuiWindowFlags_ChildWindow ) )
{
ImGuiWindow * focus_window = NavRestoreLastChildNavWindow ( window ) ;
FocusWindow ( focus_window ) ;
return ;
}
}
}
void ImGui : : PushItemWidth ( float item_width )
{
@ -8639,11 +8659,11 @@ static float ImGui::NavUpdatePageUpPageDown(int allowed_dir_flags)
return 0.0f ;
}
static int FindWindow Index( ImGuiWindow * window ) // FIXME-OPT O(N)
static int FindWindow Focus Index( ImGuiWindow * window ) // FIXME-OPT O(N)
{
ImGuiContext & g = * GImGui ;
for ( int i = g . Windows . Size - 1 ; i > = 0 ; i - - )
if ( g . Windows [ i ] = = window )
for ( int i = g . Windows FocusOrder . Size - 1 ; i > = 0 ; i - - )
if ( g . Windows FocusOrder [ i ] = = window )
return i ;
return - 1 ;
}
@ -8651,9 +8671,9 @@ static int FindWindowIndex(ImGuiWindow* window) // FIXME-OPT O(N)
static ImGuiWindow * FindWindowNavFocusable ( int i_start , int i_stop , int dir ) // FIXME-OPT O(N)
{
ImGuiContext & g = * GImGui ;
for ( int i = i_start ; i > = 0 & & i < g . Windows . Size & & i ! = i_stop ; i + = dir )
if ( ImGui : : IsWindowNavFocusable ( g . Windows [ i ] ) )
return g . Windows [ i ] ;
for ( int i = i_start ; i > = 0 & & i < g . Windows FocusOrder . Size & & i ! = i_stop ; i + = dir )
if ( ImGui : : IsWindowNavFocusable ( g . Windows FocusOrder [ i ] ) )
return g . Windows FocusOrder [ i ] ;
return NULL ;
}
@ -8664,10 +8684,10 @@ static void NavUpdateWindowingHighlightWindow(int focus_change_dir)
if ( g . NavWindowingTarget - > Flags & ImGuiWindowFlags_Modal )
return ;
const int i_current = FindWindow Index( g . NavWindowingTarget ) ;
const int i_current = FindWindow Focus Index( g . NavWindowingTarget ) ;
ImGuiWindow * window_target = FindWindowNavFocusable ( i_current + focus_change_dir , - INT_MAX , focus_change_dir ) ;
if ( ! window_target )
window_target = FindWindowNavFocusable ( ( focus_change_dir < 0 ) ? ( g . Windows . Size - 1 ) : 0 , i_current , focus_change_dir ) ;
window_target = FindWindowNavFocusable ( ( focus_change_dir < 0 ) ? ( g . Windows FocusOrder . Size - 1 ) : 0 , i_current , focus_change_dir ) ;
if ( window_target ) // Don't reset windowing target if there's a single window in the list
g . NavWindowingTarget = g . NavWindowingTargetAnim = window_target ;
g . NavWindowingToggleLayer = false ;
@ -8699,7 +8719,7 @@ static void ImGui::NavUpdateWindowing()
bool start_windowing_with_gamepad = ! g . NavWindowingTarget & & IsNavInputPressed ( ImGuiNavInput_Menu , ImGuiInputReadMode_Pressed ) ;
bool start_windowing_with_keyboard = ! g . NavWindowingTarget & & g . IO . KeyCtrl & & IsKeyPressedMap ( ImGuiKey_Tab ) & & ( g . IO . ConfigFlags & ImGuiConfigFlags_NavEnableKeyboard ) ;
if ( start_windowing_with_gamepad | | start_windowing_with_keyboard )
if ( ImGuiWindow * window = g . NavWindow ? g . NavWindow : FindWindowNavFocusable ( g . Windows . Size - 1 , - INT_MAX , - 1 ) )
if ( ImGuiWindow * window = g . NavWindow ? g . NavWindow : FindWindowNavFocusable ( g . Windows FocusOrder . Size - 1 , - INT_MAX , - 1 ) )
{
g . NavWindowingTarget = g . NavWindowingTargetAnim = window ;
g . NavWindowingTimer = g . NavWindowingHighlightAlpha = 0.0f ;
@ -8843,9 +8863,9 @@ void ImGui::NavUpdateWindowingList()
SetNextWindowPos ( viewport - > Pos + viewport - > Size * 0.5f , ImGuiCond_Always , ImVec2 ( 0.5f , 0.5f ) ) ;
PushStyleVar ( ImGuiStyleVar_WindowPadding , g . Style . WindowPadding * 2.0f ) ;
Begin ( " ###NavWindowingList " , NULL , ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoFocusOnAppearing | ImGuiWindowFlags_NoNav | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoInputs | ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoSavedSettings ) ;
for ( int n = g . Windows . Size - 1 ; n > = 0 ; n - - )
for ( int n = g . Windows FocusOrder . Size - 1 ; n > = 0 ; n - - )
{
ImGuiWindow * window = g . Windows [ n ] ;
ImGuiWindow * window = g . Windows FocusOrder [ n ] ;
if ( ! IsWindowNavFocusable ( window ) )
continue ;
const char * label = window - > Name ;