NcEngine
ImGuiUtility.h
Go to the documentation of this file.
1
10#pragma once
11
12#include "ncengine/type/EngineTypes.h"
14
15#include "imgui.h"
16#include "ncmath/Vector.h"
17#include "ncmath/Geometry.h"
18
19#include <algorithm>
20#include <functional>
21#include <numbers>
22#include <span>
23#include <string>
24#include <string_view>
25
27#define IMGUI_SCOPE_UNIQUE_NAME_HELPER(name, lineNumber) name ## lineNumber
28#define IMGUI_SCOPE_UNIQUE_NAME(name, lineNumber) IMGUI_SCOPE_UNIQUE_NAME_HELPER(name, lineNumber)
32#define IMGUI_SCOPE(Property, ...) auto IMGUI_SCOPE_UNIQUE_NAME(imguiProp, __LINE__) = Property(__VA_ARGS__)
33
34namespace nc::ui
35{
36constexpr auto g_defaultItemWidth = 50.0f;
37constexpr auto g_maxPos = 5000.0f;
38constexpr auto g_minPos = -g_maxPos;
39constexpr auto g_maxAngle = std::numbers::pi_v<float> * 2.0f;
40constexpr auto g_minAngle = -g_maxAngle;
41constexpr auto g_minScale = 0.1f;
42constexpr auto g_maxScale = 1000.0f;
43
45template<class F>
46void Window(const char* label, F&& drawContents);
47
49template<class F>
50void Window(const char* label, ImGuiWindowFlags flags, F&& drawContents);
51
53template<class F>
54void ChildWindow(const char* label, F&& drawContents);
55
57auto IsWindowBackgroundClicked() -> bool;
58
60auto InputInt(int& value, const char* label, int step = 1) -> bool;
61
63auto InputU8(uint8_t& value, const char* label) -> bool;
64
66auto InputU16(uint16_t& value, const char* label) -> bool;
67
69auto InputU32(uint32_t& value, const char* label) -> bool;
70
72auto InputU64(uint64_t& value, const char* label) -> bool;
73
75auto DragFloat(float& value, const char* label, float speed = 1.0f, float min = 0.0f, float max = 1000.0f) -> bool;
76
78auto Checkbox(bool& value, const char* name) -> bool;
79
81auto InputVector3(Vector3& value, const char* label, float speed = 1.0f, float min = 0.0f, float max = 100.0f) -> bool;
82
84auto InputPosition(Vector3& value, const char* label) -> bool;
85
87auto InputAngles(Vector3& value, const char* label) -> bool;
88
90auto InputScale(Vector3& value, const char* label, float min = g_minScale, float max = g_maxScale) -> bool;
91
93auto InputAxis(Vector3& value, const char* label, float min = -1.0f, float max = 1.0f) -> bool;
94
96auto InputReferenceFrame(Vector3& basis, Vector3& normal, const char* basisLabel, const char* normalLabel) -> bool;
97
99auto InputColor3(Vector3& value, const char* label) -> bool;
100
102auto InputColor4(Vector4& value, const char* label) -> bool;
103
105auto Combobox(std::string& value, const char* label, std::span<const std::string_view> items);
106
108auto FilteredCombobox(std::string& value, const char* label, std::span<const std::string_view> items, auto&& disableIf);
109
111auto InputText(std::string& value, const char* label) -> bool;
112
114auto SelectableWidget(bool selected, const ImVec2& size, auto&& widget) -> bool;
115
121template<class Getter, class Setter>
123{
124 Getter get;
125 Setter set;
126 const char* name = "";
127};
128
142auto PropertyWidget(const auto& property, auto& instance, auto&& widget, auto&&... args) -> bool;
143
145template<class T>
146void DragAndDropSource(T* item);
147
149template<class T, std::invocable<T*> F>
150void DragAndDropTarget(F&& func);
151
153void SameLineSpaced();
154
156void SameLineRightAligned(float itemWidth);
157
159void SeparatorSpaced();
160
163{
164 explicit ImGuiId(const char* id) { ImGui::PushID(id); }
165 explicit ImGuiId(int id) { ImGui::PushID(id); }
166 ~ImGuiId() noexcept { ImGui::PopID(); }
167};
168
170struct Indent
171{
172 explicit Indent() { ImGui::Indent(); }
173 ~Indent() noexcept { ImGui::Unindent(); }
174};
175
178{
179 explicit StyleColor(ImGuiCol index, ImU32 col) { ImGui::PushStyleColor(index, col); }
180 explicit StyleColor(ImGuiCol index, const ImVec4& col) { ImGui::PushStyleColor(index, col); }
181 explicit StyleColor(ImGuiCol index, const Vector4& col) { ImGui::PushStyleColor(index, col); }
182 ~StyleColor() noexcept { ImGui::PopStyleColor(); }
183};
184
187{
188 explicit ItemWidth(float itemWidth) { ImGui::PushItemWidth(itemWidth); }
189 ~ItemWidth() noexcept { ImGui::PopItemWidth(); }
190};
191
194{
195 explicit DisableIf(bool condition) { ImGui::BeginDisabled(condition); }
196 ~DisableIf() noexcept { ImGui::EndDisabled(); }
197};
198
201inline auto IsCapturingKeyboard() -> bool
202{
203 return ImGui::GetIO().WantCaptureKeyboard;
204}
205
208inline auto IsCapturingMouse() -> bool
209{
210 return ImGui::GetIO().WantCaptureMouse;
211}
212
213template<class F>
214void Window(const char* label, ImGuiWindowFlags flags, F&& drawContents)
215{
216 if (ImGui::Begin(label, nullptr, flags))
217 {
218 drawContents();
219 }
220
221 ImGui::End();
222}
223
224template<class F>
225void Window(const char* label, F&& drawContents)
226{
227 Window(label, ImGuiWindowFlags_None, std::forward<F>(drawContents));
228}
229
230template<class F>
231void ChildWindow(const char* label, F&& drawContents)
232{
233 if (ImGui::BeginChild(label, {0, 0}, true))
234 {
235 drawContents();
236 };
237
238 ImGui::EndChild();
239}
240
241inline auto IsWindowBackgroundClicked() -> bool
242{
243 return ImGui::IsWindowHovered() && ImGui::IsMouseClicked(0) && !ImGui::IsAnyItemHovered();
244}
245
246inline void SameLineSpaced()
247{
248 ImGui::SameLine();
249 ImGui::Spacing();
250 ImGui::SameLine();
251}
252
253inline void SameLineRightAligned(float itemWidth)
254{
255 const auto availableWidth = ImGui::GetContentRegionAvail().x;
256 itemWidth += + ImGui::GetStyle().FramePadding.x * 2.0f;
257 ImGui::SameLine(ImGui::GetCursorPosX() + availableWidth - itemWidth);
258}
259
260inline void SeparatorSpaced()
261{
262 ImGui::Spacing();
263 ImGui::Separator();
264 ImGui::Spacing();
265}
266
267inline auto InputInt(int& value, const char* label, int step) -> bool
268{
269 return ImGui::InputInt(label, &value, step);
270}
271
272inline auto InputU8(uint8_t& value, const char* label) -> bool
273{
274 constexpr uint8_t step = 1;
275 return ImGui::InputScalar(label, ImGuiDataType_U8, &value, &step);
276}
277
278inline auto InputU16(uint16_t& value, const char* label) -> bool
279{
280 constexpr uint16_t step = 1;
281 return ImGui::InputScalar(label, ImGuiDataType_U16, &value, &step);
282}
283
284inline auto InputU32(unsigned& value, const char* label) -> bool
285{
286 constexpr uint32_t step = 1;
287 return ImGui::InputScalar(label, ImGuiDataType_U32, &value, &step);
288}
289
290inline auto InputU64(uint64_t& value, const char* label) -> bool
291{
292 constexpr uint64_t step = 1;
293 return ImGui::InputScalar(label, ImGuiDataType_U64, &value, &step);
294}
295
296inline auto DragFloat(float& value, const char* label, float speed, float min, float max) -> bool
297{
298 return ImGui::DragFloat(label, &value, speed, min, max, "%.3f", ImGuiSliderFlags_AlwaysClamp);
299}
300
301inline auto Checkbox(bool& value, const char* name) -> bool
302{
303 return ImGui::Checkbox(name, &value);
304}
305
306inline auto InputVector3(Vector3& value, const char* label, float speed, float min, float max) -> bool
307{
308 return ImGui::DragFloat3(label, &value.x, speed, min, max, "%.3f", ImGuiSliderFlags_AlwaysClamp);
309}
310
311inline auto InputPosition(Vector3& value, const char* label) -> bool
312{
313 return ImGui::DragFloat3(label, &value.x, 1.0f, g_minPos, g_maxPos, "%.3f", ImGuiSliderFlags_AlwaysClamp);
314}
315
316inline auto InputAngles(Vector3& value, const char* label) -> bool
317{
318 return ImGui::DragFloat3(label, &value.x, 0.1f, g_minAngle, g_maxAngle, "%.3f", ImGuiSliderFlags_AlwaysClamp);
319}
320
321inline auto InputScale(Vector3& value, const char* label, float min, float max) -> bool
322{
323 return ImGui::DragFloat3(label, &value.x, 0.5f, min, max, "%.3f", ImGuiSliderFlags_AlwaysClamp);
324}
325
326inline auto InputAxis(Vector3& value, const char* label, float min, float max) -> bool
327{
328 const auto previous = value;
329 if (ImGui::DragFloat3(label, &value.x, 0.1f, min, max, "%.3f", ImGuiSliderFlags_AlwaysClamp))
330 {
331 // When a component is changed to a maximum, other values are still potentially non-zero. These cases need to
332 // be hard reset to the correct axis otherwise normalization prevents ever being able to reach it.
333 if (value.x != previous.x && std::fabs(value.x) == 1.0f)
334 value = Vector3{value.x, 0.0f, 0.0f};
335 else if (value.y != previous.y && std::fabs(value.y) == 1.0f)
336 value = Vector3{0.0f, value.y, 0.0f};
337 else if (value.z != previous.z && std::fabs(value.z) == 1.0f)
338 value = Vector3{0.0f, 0.0f, value.z};
339 else if (value == Vector3::Zero()) // prevent problems when 0 is directly enterered
340 value = previous;
341 else
342 value = Normalize(value);
343
344 return true;
345 }
346
347 return false;
348}
349
350inline auto InputReferenceFrame(Vector3& basis, Vector3& normal, const char* basisLabel, const char* normalLabel) -> bool
351{
352 auto modified = false;
353 if (InputAxis(basis, basisLabel))
354 {
355 normal = ClosestOrthogonal(normal, basis);
356 modified = true;
357 }
358
359 if (InputAxis(normal, normalLabel))
360 {
361 basis = ClosestOrthogonal(basis, normal);
362 modified = true;
363 }
364
365 return modified;
366}
367
368inline auto InputColor(Vector3& value, const char* label) -> bool
369{
370 return ImGui::ColorEdit3(label, &value.x, ImGuiColorEditFlags_NoInputs);
371}
372
373inline auto InputColor3(Vector3& value, const char* label) -> bool
374{
375 return ImGui::ColorEdit3(label, &value.x, ImGuiColorEditFlags_NoInputs);
376}
377
378inline auto InputColor4(Vector4& value, const char* label) -> bool
379{
380 return ImGui::ColorEdit4(label, &value.x, ImGuiColorEditFlags_NoInputs);
381}
382
383inline auto Combobox(std::string& value, const char* label, std::span<const std::string_view> items)
384{
385 if (ImGui::BeginCombo(label, value.c_str()))
386 {
387 const auto selected = std::ranges::find_if(items, [](const auto& text)
388 {
389 return ImGui::Selectable(text.data());
390 });
391
392 ImGui::EndCombo();
393
394 if (selected != std::cend(items))
395 {
396 value = selected->data();
397 return true;
398 }
399 }
400
401 return false;
402}
403
404inline auto FilteredCombobox(std::string& value, const char* label, std::span<const std::string_view> items, auto&& disableIf)
405{
406 if (ImGui::BeginCombo(label, value.c_str()))
407 {
408 const auto selected = std::ranges::find_if(items, [&disableIf](const auto& text)
409 {
410 IMGUI_SCOPE(DisableIf, disableIf(text));
411 return ImGui::Selectable(text.data());
412 });
413
414 ImGui::EndCombo();
415
416 if (selected != std::cend(items))
417 {
418 value = selected->data();
419 return true;
420 }
421 }
422
423 return false;
424}
425
426namespace internal
427{
429{
430 std::string* string;
431};
432
433inline auto InputTextCallback(ImGuiInputTextCallbackData* data) -> int
434{
435 if(data->EventFlag == ImGuiInputTextFlags_CallbackResize)
436 {
437 InputTextCallbackUserData* userData = static_cast<InputTextCallbackUserData*>(data->UserData);
438 auto* str = userData->string;
439 str->resize(data->BufTextLen);
440 data->Buf = (char*)str->c_str();
441 }
442
443 return 0;
444}
445} // namespace internal
446
447inline auto InputText(std::string& text, const char* label) -> bool
448{
449 auto flags = ImGuiInputTextFlags_CallbackResize | ImGuiInputTextFlags_EnterReturnsTrue;
450 auto userData = internal::InputTextCallbackUserData{.string = &text};
451 return ImGui::InputText(label, text.data(), text.capacity() + 1, flags, internal::InputTextCallback, &userData);
452}
453
454auto SelectableWidget(bool selected, const ImVec2& size, auto&& widget) -> bool
455{
456 static constexpr auto flags = ImGuiSelectableFlags_AllowDoubleClick |
457 ImGuiSelectableFlags_AllowItemOverlap;
458 auto clicked = false;
459 if (ImGui::Selectable("##select", selected, flags, size))
460 clicked = true;
461
462 ImGui::SameLine();
463 widget();
464 if (ImGui::IsItemClicked(0))
465 {
466 clicked = true;
467 ImGui::SetKeyboardFocusHere(-1);
468 }
469
470 return clicked;
471};
472
473auto PropertyWidget(const auto& property, auto& instance, auto&& widget, auto&&... args) -> bool
474{
475 auto value = std::invoke(property.get, instance);
476 if (widget(value, property.name, std::forward<decltype(args)>(args)...))
477 {
478 std::invoke(property.set, instance, value);
479 return true;
480 }
481
482 return false;
483}
484
485template<class T>
486void DragAndDropSource(T* item)
487{
488 if(ImGui::BeginDragDropSource(ImGuiDragDropFlags_None))
489 {
490 ImGui::SetDragDropPayload(type::Type<T>::name, item, sizeof(T));
491 ImGui::Text(type::Type<T>::name);
492 ImGui::EndDragDropSource();
493 }
494}
495
496template<class T, std::invocable<T*> F>
497void DragAndDropTarget(F&& func)
498{
499 if(ImGui::BeginDragDropTarget())
500 {
501 if(const auto* payload = ImGui::AcceptDragDropPayload(type::Type<T>::name))
502 {
503 func(static_cast<std::add_pointer_t<T>>(payload->Data));
504 }
505
506 ImGui::EndDragDropTarget();
507 }
508}
509} // namespace nc::ui
void SameLineRightAligned(float itemWidth)
UI layout helper for right aligning an item on the same line as the previous item.
Definition: ImGuiUtility.h:253
auto IsCapturingMouse() -> bool
Check if the UI is currently using mouse events.
Definition: ImGuiUtility.h:208
auto InputU32(uint32_t &value, const char *label) -> bool
Input field UI widget for uint32_t.
auto InputColor4(Vector4 &value, const char *label) -> bool
RGBA color picker UI widget.
Definition: ImGuiUtility.h:378
void SameLineSpaced()
UI layout helper for spacing.
Definition: ImGuiUtility.h:246
auto PropertyWidget(const auto &property, auto &instance, auto &&widget, auto &&... args) -> bool
Invoke a widget function with a Property and object instance.
Definition: ImGuiUtility.h:473
auto InputReferenceFrame(Vector3 &basis, Vector3 &normal, const char *basisLabel, const char *normalLabel) -> bool
Vector3 UI widget constrained for two orthonormal axis inputs.
Definition: ImGuiUtility.h:350
auto FilteredCombobox(std::string &value, const char *label, std::span< const std::string_view > items, auto &&disableIf)
Combobox UI widget that allows disabling selection of entries.
Definition: ImGuiUtility.h:404
void DragAndDropTarget(F &&func)
Introduce a drag and drop target that can accept a payload of type T*.
Definition: ImGuiUtility.h:497
auto InputPosition(Vector3 &value, const char *label) -> bool
Vector3 UI widget constrained for position inputs.
Definition: ImGuiUtility.h:311
auto Combobox(std::string &value, const char *label, std::span< const std::string_view > items)
Combobox UI widget.
Definition: ImGuiUtility.h:383
void ChildWindow(const char *label, F &&drawContents)
Create a child window.
Definition: ImGuiUtility.h:231
void Window(const char *label, F &&drawContents)
Create a top-level window.
Definition: ImGuiUtility.h:225
auto InputText(std::string &value, const char *label) -> bool
Text input UI widget.
Definition: ImGuiUtility.h:447
auto InputScale(Vector3 &value, const char *label, float min=g_minScale, float max=g_maxScale) -> bool
Vector3 UI widget constrained for scale inputs.
Definition: ImGuiUtility.h:321
auto InputU16(uint16_t &value, const char *label) -> bool
Input field UI widget for uint16_t.
Definition: ImGuiUtility.h:278
auto InputU64(uint64_t &value, const char *label) -> bool
Input field UI widget for uint64_t.
Definition: ImGuiUtility.h:290
void SeparatorSpaced()
UI layout helper for separator.
Definition: ImGuiUtility.h:260
void DragAndDropSource(T *item)
Introduce a drag and drop source with item as its payload.
Definition: ImGuiUtility.h:486
auto IsWindowBackgroundClicked() -> bool
Check if the current window background is clicked on.
Definition: ImGuiUtility.h:241
auto DragFloat(float &value, const char *label, float speed=1.0f, float min=0.0f, float max=1000.0f) -> bool
Drag slider UI widget for float.
Definition: ImGuiUtility.h:296
auto SelectableWidget(bool selected, const ImVec2 &size, auto &&widget) -> bool
Wrap a widget with a selectable - allows for things like list box of input widgets.
Definition: ImGuiUtility.h:454
auto InputInt(int &value, const char *label, int step=1) -> bool
Input field UI widget for int.
Definition: ImGuiUtility.h:267
#define IMGUI_SCOPE(Property,...)
Create a variable to hold a scoped property.
Definition: ImGuiUtility.h:32
auto IsCapturingKeyboard() -> bool
Check if the UI is currently using keyboard events.
Definition: ImGuiUtility.h:201
auto InputU8(uint8_t &value, const char *label) -> bool
Input field UI widget for uint8_t.
Definition: ImGuiUtility.h:272
auto InputAngles(Vector3 &value, const char *label) -> bool
Vector3 UI widget constrained for angular (radians) inputs.
Definition: ImGuiUtility.h:316
auto InputAxis(Vector3 &value, const char *label, float min=-1.0f, float max=1.0f) -> bool
Vector3 UI widget constrained for normalized axis inputs.
Definition: ImGuiUtility.h:326
auto InputColor3(Vector3 &value, const char *label) -> bool
RGB color picker UI widget.
Definition: ImGuiUtility.h:373
auto Checkbox(bool &value, const char *name) -> bool
Checkbox UI widget for bool.
Definition: ImGuiUtility.h:301
auto InputVector3(Vector3 &value, const char *label, float speed=1.0f, float min=0.0f, float max=100.0f) -> bool
Vector3 UI widget with three float drag sliders.
Definition: ImGuiUtility.h:306
auto Normalize(const Quaternion &quat) -> Quaternion
Return a normalized quaternion.
A three component vector.
Definition: Vector.h:29
A four component vector.
Definition: Vector.h:48
Definition: Type.h:11
RAII wrapper for conditionally disabling widgets within a scope.
Definition: ImGuiUtility.h:194
RAII wrapper for scoped uids.
Definition: ImGuiUtility.h:163
RAII wrapper for scoped indentation.
Definition: ImGuiUtility.h:171
RAII wrapper for scoped item width.
Definition: ImGuiUtility.h:187
Simple getter/setter-based property wrapper.
Definition: ImGuiUtility.h:123
RAII wrapper for scoped style color.
Definition: ImGuiUtility.h:178
Definition: ImGuiUtility.h:429