@@ -6155,18 +6155,29 @@ bool ImGui::TreeNodeBehavior(ImGuiID id, ImGuiTreeNodeFlags flags, const char* l
6155
6155
if (!display_frame && (flags & (ImGuiTreeNodeFlags_SpanAvailWidth | ImGuiTreeNodeFlags_SpanFullWidth)) == 0)
6156
6156
interact_bb.Max.x = frame_bb.Min.x + text_width + style.ItemSpacing.x * 2.0f;
6157
6157
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.
6162
6159
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
-
6166
6160
bool item_add = ItemAdd(interact_bb, id);
6167
6161
g.LastItemData.StatusFlags |= ImGuiItemStatusFlags_HasDisplayRect;
6168
6162
g.LastItemData.DisplayRect = frame_bb;
6169
6163
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;
6170
6181
if (!item_add)
6171
6182
{
6172
6183
if (is_open && !(flags & ImGuiTreeNodeFlags_NoTreePushOnOpen))
@@ -6336,12 +6347,14 @@ void ImGui::TreePop()
6336
6347
ImU32 tree_depth_mask = (1 << window->DC.TreeDepth);
6337
6348
6338
6349
// 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
+ }
6345
6358
window->DC.TreeJumpToParentOnPopMask &= tree_depth_mask - 1;
6346
6359
6347
6360
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