Skip to content

Commit 8bab3ea

Browse files
committed
Clipper: added SeekCursorForItem() function, for use when using ImGuiListClipper::Begin(INT_MAX). (#1311)
Tagging #3609 just in case we made a mistake introducing a regression (but tests are passing and have been extended).
1 parent 74a1854 commit 8bab3ea

File tree

3 files changed

+30
-14
lines changed

3 files changed

+30
-14
lines changed

docs/CHANGELOG.txt

+4
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,10 @@ Other changes:
7777
Disabling this was previously possible for Selectable() via a direct flag but not for MenuItem().
7878
(#1379, #1468, #2200, #4936, #5216, #7302, #7573)
7979
- This was mostly all previously in imgui_internal.h.
80+
- Clipper: added SeekCursorForItem() function. When using ImGuiListClipper::Begin(INT_MAX) you can
81+
can use the clipper without knowing the amount of items beforehand. (#1311)
82+
In this situation, call ImGuiListClipper::SeekCursorForItem(items_count) as the end of your iteration
83+
loop to position the layout cursor correctly. This is done automatically if provided a count to Begin().
8084
- Style: close button and collapse/window-menu button hover highlight made rectangular instead of round.
8185
- Debug Tools: Added IMGUI_DEBUG_LOG(), ImGui::DebugLog() in public API. (#5855)
8286
Debug log entries add a imgui frame counter prefix + are redirected to ShowDebugLogWindow() and

imgui.cpp

+18-12
Original file line numberDiff line numberDiff line change
@@ -2911,15 +2911,6 @@ static void ImGuiListClipper_SeekCursorAndSetupPrevLine(float pos_y, float line_
29112911
}
29122912
}
29132913

2914-
static void ImGuiListClipper_SeekCursorForItem(ImGuiListClipper* clipper, int item_n)
2915-
{
2916-
// StartPosY starts from ItemsFrozen hence the subtraction
2917-
// Perform the add and multiply with double to allow seeking through larger ranges
2918-
ImGuiListClipperData* data = (ImGuiListClipperData*)clipper->TempData;
2919-
float pos_y = (float)((double)clipper->StartPosY + data->LossynessOffset + (double)(item_n - data->ItemsFrozen) * clipper->ItemsHeight);
2920-
ImGuiListClipper_SeekCursorAndSetupPrevLine(pos_y, clipper->ItemsHeight);
2921-
}
2922-
29232914
ImGuiListClipper::ImGuiListClipper()
29242915
{
29252916
memset(this, 0, sizeof(*this));
@@ -2956,6 +2947,7 @@ void ImGuiListClipper::Begin(int items_count, float items_height)
29562947
data->Reset(this);
29572948
data->LossynessOffset = window->DC.CursorStartPosLossyness.y;
29582949
TempData = data;
2950+
StartSeekOffsetY = data->LossynessOffset;
29592951
}
29602952

29612953
void ImGuiListClipper::End()
@@ -2966,7 +2958,7 @@ void ImGuiListClipper::End()
29662958
ImGuiContext& g = *Ctx;
29672959
IMGUI_DEBUG_LOG_CLIPPER("Clipper: End() in '%s'\n", g.CurrentWindow->Name);
29682960
if (ItemsCount >= 0 && ItemsCount < INT_MAX && DisplayStart >= 0)
2969-
ImGuiListClipper_SeekCursorForItem(this, ItemsCount);
2961+
SeekCursorForItem(ItemsCount);
29702962

29712963
// Restore temporary buffer and fix back pointers which may be invalidated when nesting
29722964
IM_ASSERT(data->ListClipper == this);
@@ -2990,6 +2982,17 @@ void ImGuiListClipper::IncludeItemsByIndex(int item_begin, int item_end)
29902982
data->Ranges.push_back(ImGuiListClipperRange::FromIndices(item_begin, item_end));
29912983
}
29922984

2985+
// This is already called while stepping.
2986+
// The ONLY reason you may want to call this is if you passed INT_MAX to ImGuiListClipper::Begin() because you couldn't step item count beforehand.
2987+
void ImGuiListClipper::SeekCursorForItem(int item_n)
2988+
{
2989+
// - Perform the add and multiply with double to allow seeking through larger ranges.
2990+
// - StartPosY starts from ItemsFrozen, by adding SeekOffsetY we generally cancel that out (SeekOffsetY == LossynessOffset - ItemsFrozen * ItemsHeight).
2991+
// - The reason we store SeekOffsetY instead of inferring it, is because we want to allow user to perform Seek after the last step, where ImGuiListClipperData is already done.
2992+
float pos_y = (float)((double)StartPosY + StartSeekOffsetY + (double)item_n * ItemsHeight);
2993+
ImGuiListClipper_SeekCursorAndSetupPrevLine(pos_y, ItemsHeight);
2994+
}
2995+
29932996
static bool ImGuiListClipper_StepInternal(ImGuiListClipper* clipper)
29942997
{
29952998
ImGuiContext& g = *clipper->Ctx;
@@ -3053,6 +3056,9 @@ static bool ImGuiListClipper_StepInternal(ImGuiListClipper* clipper)
30533056
const int already_submitted = clipper->DisplayEnd;
30543057
if (calc_clipping)
30553058
{
3059+
// Record seek offset, this is so ImGuiListClipper::Seek() can be called after ImGuiListClipperData is done
3060+
clipper->StartSeekOffsetY = (double)data->LossynessOffset - data->ItemsFrozen * (double)clipper->ItemsHeight;
3061+
30563062
if (g.LogEnabled)
30573063
{
30583064
// If logging is active, do not perform any clipping
@@ -3100,7 +3106,7 @@ static bool ImGuiListClipper_StepInternal(ImGuiListClipper* clipper)
31003106
clipper->DisplayStart = ImMax(data->Ranges[data->StepNo].Min, already_submitted);
31013107
clipper->DisplayEnd = ImMin(data->Ranges[data->StepNo].Max, clipper->ItemsCount);
31023108
if (clipper->DisplayStart > already_submitted) //-V1051
3103-
ImGuiListClipper_SeekCursorForItem(clipper, clipper->DisplayStart);
3109+
clipper->SeekCursorForItem(clipper->DisplayStart);
31043110
data->StepNo++;
31053111
if (clipper->DisplayStart == clipper->DisplayEnd && data->StepNo < data->Ranges.Size)
31063112
continue;
@@ -3110,7 +3116,7 @@ static bool ImGuiListClipper_StepInternal(ImGuiListClipper* clipper)
31103116
// After the last step: Let the clipper validate that we have reached the expected Y position (corresponding to element DisplayEnd),
31113117
// Advance the cursor to the end of the list and then returns 'false' to end the loop.
31123118
if (clipper->ItemsCount < INT_MAX)
3113-
ImGuiListClipper_SeekCursorForItem(clipper, clipper->ItemsCount);
3119+
clipper->SeekCursorForItem(clipper->ItemsCount);
31143120

31153121
return false;
31163122
}

imgui.h

+8-2
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828
// Library Version
2929
// (Integer encoded as XYYZZ for use in #if preprocessor conditionals, e.g. '#if IMGUI_VERSION_NUM >= 12345')
3030
#define IMGUI_VERSION "1.91.0 WIP"
31-
#define IMGUI_VERSION_NUM 19094
31+
#define IMGUI_VERSION_NUM 19095
3232
#define IMGUI_HAS_TABLE
3333

3434
/*
@@ -2601,9 +2601,10 @@ struct ImGuiListClipper
26012601
int ItemsCount; // [Internal] Number of items
26022602
float ItemsHeight; // [Internal] Height of item after a first step and item submission can calculate it
26032603
float StartPosY; // [Internal] Cursor position at the time of Begin() or after table frozen rows are all processed
2604+
double StartSeekOffsetY; // [Internal] Account for frozen rows in a table and initial loss of precision in very large windows.
26042605
void* TempData; // [Internal] Internal data
26052606

2606-
// items_count: Use INT_MAX if you don't know how many items you have (in which case the cursor won't be advanced in the final step)
2607+
// items_count: Use INT_MAX if you don't know how many items you have (in which case the cursor won't be advanced in the final step, and you can call SeekCursorForItem() manually if you need)
26072608
// items_height: Use -1.0f to be calculated automatically on first step. Otherwise pass in the distance between your items, typically GetTextLineHeightWithSpacing() or GetFrameHeightWithSpacing().
26082609
IMGUI_API ImGuiListClipper();
26092610
IMGUI_API ~ImGuiListClipper();
@@ -2616,6 +2617,11 @@ struct ImGuiListClipper
26162617
inline void IncludeItemByIndex(int item_index) { IncludeItemsByIndex(item_index, item_index + 1); }
26172618
IMGUI_API void IncludeItemsByIndex(int item_begin, int item_end); // item_end is exclusive e.g. use (42, 42+1) to make item 42 never clipped.
26182619

2620+
// Seek cursor toward given item. This is automatically called while stepping.
2621+
// - The only reason to call this is: you can use ImGuiListClipper::Begin(INT_MAX) if you don't know item count ahead of time.
2622+
// - In this case, after all steps are done, you'll want to call SeekCursorForItem(item_count).
2623+
IMGUI_API void SeekCursorForItem(int item_index);
2624+
26192625
#ifndef IMGUI_DISABLE_OBSOLETE_FUNCTIONS
26202626
inline void IncludeRangeByIndices(int item_begin, int item_end) { IncludeItemsByIndex(item_begin, item_end); } // [renamed in 1.89.9]
26212627
inline void ForceDisplayRangeByIndices(int item_begin, int item_end) { IncludeItemsByIndex(item_begin, item_end); } // [renamed in 1.89.6]

0 commit comments

Comments
 (0)