Skip to content

Commit bc3c0ce

Browse files
committed
Nav, TreeNode: Pressing Left with ImGuiTreeNodeFlags_NavLeftJumpsBackHere now goes through proper navigation logic: honor scrolling and selection. (ocornut#1079, ocornut#1131)
Added a stack for this purpose which other features might build on (e.g. ocornut#2920). However this is currently gated by many tests and not a performance concern, but making stack happen all the time may be undesirable.
1 parent edebb90 commit bc3c0ce

File tree

5 files changed

+57
-14
lines changed

5 files changed

+57
-14
lines changed

docs/CHANGELOG.txt

+2
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,8 @@ Breaking changes:
4444

4545
Other changes:
4646

47+
- Nav, TreeNode: Pressing Left with ImGuiTreeNodeFlags_NavLeftJumpsBackHere now goes
48+
through proper navigation logic: honor scrolling and selection. (#1079, #1131)
4749
- Sliders: Fixed an integer overflow and div-by-zero in SliderInt() when
4850
v_max=INT_MAX (#6675, #6679) [@jbarthelmes]
4951
- ImDrawList: Fixed OOB access in _CalcCircleAutoSegmentCount when passing excessively

imgui.cpp

+14
Original file line numberDiff line numberDiff line change
@@ -3653,6 +3653,7 @@ void ImGui::Shutdown()
36533653
g.FontStack.clear();
36543654
g.OpenPopupStack.clear();
36553655
g.BeginPopupStack.clear();
3656+
g.NavTreeNodeStack.clear();
36563657

36573658
g.Viewports.clear_delete();
36583659

@@ -11191,6 +11192,19 @@ void ImGui::NavMoveRequestResolveWithLastItem(ImGuiNavItemData* result)
1119111192
NavUpdateAnyRequestFlag();
1119211193
}
1119311194

11195+
// Called by TreePop() to implement ImGuiTreeNodeFlags_NavLeftJumpsBackHere
11196+
void ImGui::NavMoveRequestResolveWithPastTreeNode(ImGuiNavItemData* result, ImGuiNavTreeNodeData* tree_node_data)
11197+
{
11198+
ImGuiContext& g = *GImGui;
11199+
g.NavMoveScoringItems = false;
11200+
g.LastItemData.ID = tree_node_data->ID;
11201+
g.LastItemData.InFlags = tree_node_data->InFlags;
11202+
g.LastItemData.NavRect = tree_node_data->NavRect;
11203+
NavApplyItemToResult(result); // Result this instead of implementing a NavApplyPastTreeNodeToResult()
11204+
NavClearPreferredPosForAxis(ImGuiAxis_Y);
11205+
NavUpdateAnyRequestFlag();
11206+
}
11207+
1119411208
void ImGui::NavMoveRequestCancel()
1119511209
{
1119611210
ImGuiContext& g = *GImGui;

imgui.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@
2626
// Library Version
2727
// (Integer encoded as XYYZZ for use in #if preprocessor conditionals, e.g. '#if IMGUI_VERSION_NUM >= 12345')
2828
#define IMGUI_VERSION "1.89.9 WIP"
29-
#define IMGUI_VERSION_NUM 18981
29+
#define IMGUI_VERSION_NUM 18982
3030
#define IMGUI_HAS_TABLE
3131

3232
/*

imgui_internal.h

+14
Original file line numberDiff line numberDiff line change
@@ -135,6 +135,7 @@ struct ImGuiLastItemData; // Status storage for last submitted items
135135
struct ImGuiLocEntry; // A localization entry.
136136
struct ImGuiMenuColumns; // Simple column measurement, currently used for MenuItem() only
137137
struct ImGuiNavItemData; // Result of a gamepad/keyboard directional navigation move query result
138+
struct ImGuiNavTreeNodeData; // Temporary storage for last TreeNode() being a Left arrow landing candidate.
138139
struct ImGuiMetricsConfig; // Storage for ShowMetricsWindow() and DebugNodeXXX() functions
139140
struct ImGuiNextWindowData; // Storage for SetNextWindow** functions
140141
struct ImGuiNextItemData; // Storage for SetNextItem** functions
@@ -1201,6 +1202,16 @@ struct ImGuiLastItemData
12011202
ImGuiLastItemData() { memset(this, 0, sizeof(*this)); }
12021203
};
12031204

1205+
// Store data emitted by TreeNode() for usage by TreePop() to implement ImGuiTreeNodeFlags_NavLeftJumpsBackHere.
1206+
// This is the minimum amount of data that we need to perform the equivalent of NavApplyItemToResult() and which we can't infer in TreePop()
1207+
// Only stored when the node is a potential candidate for landing on a Left arrow jump.
1208+
struct ImGuiNavTreeNodeData
1209+
{
1210+
ImGuiID ID;
1211+
ImGuiItemFlags InFlags;
1212+
ImRect NavRect;
1213+
};
1214+
12041215
struct IMGUI_API ImGuiStackSizes
12051216
{
12061217
short SizeOfIDStack;
@@ -1865,6 +1876,8 @@ struct ImGuiContext
18651876
ImVector<ImGuiGroupData>GroupStack; // Stack for BeginGroup()/EndGroup() - not inherited by Begin()
18661877
ImVector<ImGuiPopupData>OpenPopupStack; // Which popups are open (persistent)
18671878
ImVector<ImGuiPopupData>BeginPopupStack; // Which level of BeginPopup() we are in (reset every frame)
1879+
ImVector<ImGuiNavTreeNodeData> NavTreeNodeStack; // Stack for TreeNode() when a NavLeft requested is emitted.
1880+
18681881
int BeginMenuCount;
18691882

18701883
// Viewports
@@ -2930,6 +2943,7 @@ namespace ImGui
29302943
IMGUI_API void NavMoveRequestSubmit(ImGuiDir move_dir, ImGuiDir clip_dir, ImGuiNavMoveFlags move_flags, ImGuiScrollFlags scroll_flags);
29312944
IMGUI_API void NavMoveRequestForward(ImGuiDir move_dir, ImGuiDir clip_dir, ImGuiNavMoveFlags move_flags, ImGuiScrollFlags scroll_flags);
29322945
IMGUI_API void NavMoveRequestResolveWithLastItem(ImGuiNavItemData* result);
2946+
IMGUI_API void NavMoveRequestResolveWithPastTreeNode(ImGuiNavItemData* result, ImGuiNavTreeNodeData* tree_node_data);
29332947
IMGUI_API void NavMoveRequestCancel();
29342948
IMGUI_API void NavMoveRequestApplyResult();
29352949
IMGUI_API void NavMoveRequestTryWrapping(ImGuiWindow* window, ImGuiNavMoveFlags move_flags);

imgui_widgets.cpp

+26-13
Original file line numberDiff line numberDiff line change
@@ -6155,18 +6155,29 @@ bool ImGui::TreeNodeBehavior(ImGuiID id, ImGuiTreeNodeFlags flags, const char* l
61556155
if (!display_frame && (flags & (ImGuiTreeNodeFlags_SpanAvailWidth | ImGuiTreeNodeFlags_SpanFullWidth)) == 0)
61566156
interact_bb.Max.x = frame_bb.Min.x + text_width + style.ItemSpacing.x * 2.0f;
61576157

6158-
// Store a flag for the current depth to tell if we will allow closing this node when navigating one of its child.
6159-
// For this purpose we essentially compare if g.NavIdIsAlive went from 0 to 1 between TreeNode() and TreePop().
6160-
// This is currently only support 32 level deep and we are fine with (1 << Depth) overflowing into a zero.
6161-
const bool is_leaf = (flags & ImGuiTreeNodeFlags_Leaf) != 0;
6158+
// Compute open and multi-select states before ItemAdd() as it clear NextItem data.
61626159
bool is_open = TreeNodeUpdateNextOpen(id, flags);
6163-
if (is_open && !g.NavIdIsAlive && (flags & ImGuiTreeNodeFlags_NavLeftJumpsBackHere) && !(flags & ImGuiTreeNodeFlags_NoTreePushOnOpen))
6164-
window->DC.TreeJumpToParentOnPopMask |= (1 << window->DC.TreeDepth);
6165-
61666160
bool item_add = ItemAdd(interact_bb, id);
61676161
g.LastItemData.StatusFlags |= ImGuiItemStatusFlags_HasDisplayRect;
61686162
g.LastItemData.DisplayRect = frame_bb;
61696163

6164+
// If a NavLeft request is happening and ImGuiTreeNodeFlags_NavLeftJumpsBackHere enabled:
6165+
// Store data for the current depth to allow returning to this node from any child item.
6166+
// For this purpose we essentially compare if g.NavIdIsAlive went from 0 to 1 between TreeNode() and TreePop().
6167+
// It will become tempting to enable ImGuiTreeNodeFlags_NavLeftJumpsBackHere by default or move it to ImGuiStyle.
6168+
// Currently only supports 32 level deep and we are fine with (1 << Depth) overflowing into a zero, easy to increase.
6169+
if (is_open && !g.NavIdIsAlive && (flags & ImGuiTreeNodeFlags_NavLeftJumpsBackHere) && !(flags & ImGuiTreeNodeFlags_NoTreePushOnOpen))
6170+
if (g.NavMoveDir == ImGuiDir_Left && g.NavWindow == window && NavMoveRequestButNoResultYet())
6171+
{
6172+
g.NavTreeNodeStack.resize(g.NavTreeNodeStack.Size + 1);
6173+
ImGuiNavTreeNodeData* nav_tree_node_data = &g.NavTreeNodeStack.back();
6174+
nav_tree_node_data->ID = id;
6175+
nav_tree_node_data->InFlags = g.LastItemData.InFlags;
6176+
nav_tree_node_data->NavRect = g.LastItemData.NavRect;
6177+
window->DC.TreeJumpToParentOnPopMask |= (1 << window->DC.TreeDepth);
6178+
}
6179+
6180+
const bool is_leaf = (flags & ImGuiTreeNodeFlags_Leaf) != 0;
61706181
if (!item_add)
61716182
{
61726183
if (is_open && !(flags & ImGuiTreeNodeFlags_NoTreePushOnOpen))
@@ -6336,12 +6347,14 @@ void ImGui::TreePop()
63366347
ImU32 tree_depth_mask = (1 << window->DC.TreeDepth);
63376348

63386349
// Handle Left arrow to move to parent tree node (when ImGuiTreeNodeFlags_NavLeftJumpsBackHere is enabled)
6339-
if (g.NavMoveDir == ImGuiDir_Left && g.NavWindow == window && NavMoveRequestButNoResultYet())
6340-
if (g.NavIdIsAlive && (window->DC.TreeJumpToParentOnPopMask & tree_depth_mask))
6341-
{
6342-
SetNavID(window->IDStack.back(), g.NavLayer, 0, ImRect());
6343-
NavMoveRequestCancel();
6344-
}
6350+
if (window->DC.TreeJumpToParentOnPopMask & tree_depth_mask) // Only set during request
6351+
{
6352+
ImGuiNavTreeNodeData* nav_tree_node_data = &g.NavTreeNodeStack.back();
6353+
IM_ASSERT(nav_tree_node_data->ID == window->IDStack.back());
6354+
if (g.NavIdIsAlive && g.NavMoveDir == ImGuiDir_Left && g.NavWindow == window && NavMoveRequestButNoResultYet())
6355+
NavMoveRequestResolveWithPastTreeNode(&g.NavMoveResultLocal, nav_tree_node_data);
6356+
g.NavTreeNodeStack.pop_back();
6357+
}
63456358
window->DC.TreeJumpToParentOnPopMask &= tree_depth_mask - 1;
63466359

63476360
IM_ASSERT(window->IDStack.Size > 1); // There should always be 1 element in the IDStack (pushed during window creation). If this triggers you called TreePop/PopID too much.

0 commit comments

Comments
 (0)