add first ui input elements and in-game logging
This commit is contained in:
parent
3dbf4f2255
commit
61a3d20dad
2 changed files with 260 additions and 29 deletions
|
@ -4,8 +4,6 @@
|
||||||
#include <backends/imgui_impl_sdlrenderer.h>
|
#include <backends/imgui_impl_sdlrenderer.h>
|
||||||
#include <SDL.h>
|
#include <SDL.h>
|
||||||
|
|
||||||
#include <array>
|
|
||||||
#include <cstdio>
|
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
@ -21,9 +19,10 @@ void dp_sdl_fatal(const char *const msg)
|
||||||
|
|
||||||
struct DoomState
|
struct DoomState
|
||||||
{
|
{
|
||||||
|
bool alive = false; // FIXME for testing only
|
||||||
};
|
};
|
||||||
|
|
||||||
static_assert(std::is_trivial<DoomState>::value, "DoomState must be a trivial type");
|
static_assert(std::is_trivially_copyable<DoomState>::value, "DoomState must be a trivially copyable type");
|
||||||
|
|
||||||
struct ControllerContext
|
struct ControllerContext
|
||||||
{
|
{
|
||||||
|
@ -33,9 +32,37 @@ struct ControllerContext
|
||||||
SDL_Renderer *renderer;
|
SDL_Renderer *renderer;
|
||||||
std::vector<DoomState> dooms;
|
std::vector<DoomState> dooms;
|
||||||
bool quit = false;
|
bool quit = false;
|
||||||
|
ExampleAppLog appLog;
|
||||||
|
int columns = 4;
|
||||||
};
|
};
|
||||||
|
|
||||||
void show_ui(ControllerContext *ctx)
|
struct ControllerActions
|
||||||
|
{
|
||||||
|
int doomsToSpawn = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
void spawn_doom(ControllerContext &ctx)
|
||||||
|
{
|
||||||
|
size_t nextId = ctx.dooms.size();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
DoomState ds;
|
||||||
|
ds.alive = true;
|
||||||
|
ctx.dooms.emplace_back(ds);
|
||||||
|
}
|
||||||
|
|
||||||
|
void perform_actions(ControllerContext &ctx, const ControllerActions &actions)
|
||||||
|
{
|
||||||
|
if (actions.doomsToSpawn)
|
||||||
|
{
|
||||||
|
log_info("Spawning %d new dooms", actions.doomsToSpawn);
|
||||||
|
for (int i=0; i<actions.doomsToSpawn; ++i)
|
||||||
|
spawn_doom(ctx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ControllerActions run_ui(ControllerContext &ctx)
|
||||||
{
|
{
|
||||||
#ifndef IMGUI_DISABLE_DEBUG_TOOLS
|
#ifndef IMGUI_DISABLE_DEBUG_TOOLS
|
||||||
const bool has_debug_tools = true;
|
const bool has_debug_tools = true;
|
||||||
|
@ -44,6 +71,7 @@ void show_ui(ControllerContext *ctx)
|
||||||
#endif
|
#endif
|
||||||
static bool show_app_metrics = false;
|
static bool show_app_metrics = false;
|
||||||
static bool show_app_debug_log = false;
|
static bool show_app_debug_log = false;
|
||||||
|
static bool show_log_window = true;
|
||||||
|
|
||||||
if (show_app_metrics)
|
if (show_app_metrics)
|
||||||
ImGui::ShowMetricsWindow(&show_app_metrics);
|
ImGui::ShowMetricsWindow(&show_app_metrics);
|
||||||
|
@ -51,32 +79,39 @@ void show_ui(ControllerContext *ctx)
|
||||||
ImGui::ShowDebugLogWindow(&show_app_debug_log);
|
ImGui::ShowDebugLogWindow(&show_app_debug_log);
|
||||||
|
|
||||||
const ImGuiViewport* main_viewport = ImGui::GetMainViewport();
|
const ImGuiViewport* main_viewport = ImGui::GetMainViewport();
|
||||||
|
|
||||||
|
if (show_log_window)
|
||||||
|
{
|
||||||
|
ImGui::SetNextWindowPos(ImVec2(main_viewport->WorkPos.x + 666, main_viewport->WorkPos.y + 20), ImGuiCond_FirstUseEver);
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(500, 400), ImGuiCond_FirstUseEver);
|
||||||
|
ctx.appLog.Draw("log");
|
||||||
|
}
|
||||||
|
|
||||||
ImGui::SetNextWindowPos(ImVec2(main_viewport->WorkPos.x + 20, main_viewport->WorkPos.y + 20), ImGuiCond_FirstUseEver);
|
ImGui::SetNextWindowPos(ImVec2(main_viewport->WorkPos.x + 20, main_viewport->WorkPos.y + 20), ImGuiCond_FirstUseEver);
|
||||||
ImGui::SetNextWindowSize(ImVec2(550, 340), ImGuiCond_FirstUseEver);
|
ImGui::SetNextWindowSize(ImVec2(420, 340), ImGuiCond_FirstUseEver);
|
||||||
ImGuiWindowFlags window_flags = ImGuiWindowFlags_MenuBar;
|
ImGuiWindowFlags window_flags = ImGuiWindowFlags_MenuBar;
|
||||||
|
|
||||||
|
static std::array<char, 1024> strbuf;
|
||||||
auto &io = ImGui::GetIO();
|
auto &io = ImGui::GetIO();
|
||||||
std::array<char, 1024> windowTitle;
|
|
||||||
std::snprintf(windowTitle.data(), windowTitle.size(),
|
aprintf(strbuf, "doompanning - #dooms=%zu, %.2f ms/frame (%.1f fps)###doompanning",
|
||||||
"doompanning - #dooms=%zu, %.2f ms/frame (%.1f fps)###doompanning",
|
ctx.dooms.size(), 1000.0f / io.Framerate, io.Framerate);
|
||||||
0ul, 1000.0f / io.Framerate, io.Framerate);
|
|
||||||
|
|
||||||
// Main body of the doompanning window starts here.
|
// Main body of the doompanning window starts here.
|
||||||
if (!ImGui::Begin(windowTitle.data(), nullptr, window_flags))
|
if (!ImGui::Begin(strbuf.data(), nullptr, window_flags))
|
||||||
{
|
{
|
||||||
// Early out if the window is collapsed, as an optimization.
|
// Early out if the window is collapsed, as an optimization.
|
||||||
ImGui::End();
|
ImGui::End();
|
||||||
return;
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
ImGui::PushItemWidth(ImGui::GetFontSize() * -12);
|
|
||||||
|
|
||||||
// Menu Bar
|
// Menu Bar
|
||||||
if (ImGui::BeginMenuBar())
|
if (ImGui::BeginMenuBar())
|
||||||
{
|
{
|
||||||
if (ImGui::BeginMenu("Menu"))
|
if (ImGui::BeginMenu("Menu"))
|
||||||
{
|
{
|
||||||
ImGui::MenuItem("Quit", "Ctrl+Q", &ctx->quit, true);
|
ImGui::MenuItem("Log Window", nullptr, &show_log_window);
|
||||||
|
ImGui::MenuItem("Quit", "Ctrl+Q", &ctx.quit, true);
|
||||||
ImGui::EndMenu();
|
ImGui::EndMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -90,15 +125,32 @@ void show_ui(ControllerContext *ctx)
|
||||||
ImGui::EndMenuBar();
|
ImGui::EndMenuBar();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Window contents
|
||||||
|
|
||||||
|
ControllerActions result = {};
|
||||||
|
static int doomsToSpawn = 1;
|
||||||
|
|
||||||
|
ImGui::PushItemWidth(ImGui::GetFontSize() * -16); // affects stuff like slider widths
|
||||||
|
|
||||||
|
ImGui::SliderInt("Layout columns##columns", &ctx.columns, 1, 32, "%d", ImGuiSliderFlags_AlwaysClamp);
|
||||||
|
|
||||||
|
ImGui::SliderInt("##dooms", &doomsToSpawn, 1, 16, "%d", ImGuiSliderFlags_AlwaysClamp);
|
||||||
|
aprintf(strbuf, "Spawn %d more doom%s###spawnmore", doomsToSpawn, doomsToSpawn > 1 ? "s" : "");
|
||||||
|
if (ImGui::SameLine(); ImGui::Button(strbuf.data()))
|
||||||
|
result.doomsToSpawn = doomsToSpawn;
|
||||||
|
|
||||||
ImGui::PopItemWidth();
|
ImGui::PopItemWidth();
|
||||||
ImGui::End();
|
ImGui::End();
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
int doom_controller_loop(ControllerContext *ctx)
|
int doom_controller_loop(ControllerContext &ctx)
|
||||||
{
|
{
|
||||||
static constexpr ImVec4 clear_color = ImVec4(0.45f, 0.55f, 0.60f, 1.00f);
|
static constexpr ImVec4 clear_color = ImVec4(0.45f, 0.55f, 0.60f, 1.00f);
|
||||||
|
ControllerActions actions = {};
|
||||||
|
|
||||||
while (!ctx->quit)
|
while (!ctx.quit)
|
||||||
{
|
{
|
||||||
SDL_Event event;
|
SDL_Event event;
|
||||||
while (SDL_PollEvent(&event))
|
while (SDL_PollEvent(&event))
|
||||||
|
@ -106,41 +158,64 @@ int doom_controller_loop(ControllerContext *ctx)
|
||||||
ImGui_ImplSDL2_ProcessEvent(&event);
|
ImGui_ImplSDL2_ProcessEvent(&event);
|
||||||
|
|
||||||
if (event.type == SDL_QUIT)
|
if (event.type == SDL_QUIT)
|
||||||
ctx->quit = true;
|
ctx.quit = true;
|
||||||
|
|
||||||
if (event.type == SDL_WINDOWEVENT
|
if (event.type == SDL_WINDOWEVENT
|
||||||
&& event.window.event == SDL_WINDOWEVENT_CLOSE
|
&& event.window.event == SDL_WINDOWEVENT_CLOSE
|
||||||
&& event.window.windowID == SDL_GetWindowID(ctx->window))
|
&& event.window.windowID == SDL_GetWindowID(ctx.window))
|
||||||
{
|
{
|
||||||
ctx->quit = true;
|
ctx.quit = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Process input events not consumed by ImGui
|
||||||
|
if (auto &io = ImGui::GetIO();
|
||||||
|
!io.WantCaptureKeyboard)
|
||||||
|
{
|
||||||
|
if (io.KeyCtrl && ImGui::IsKeyDown(ImGuiKey_Q))
|
||||||
|
{
|
||||||
|
ctx.quit = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
perform_actions(ctx, actions);
|
||||||
|
|
||||||
// Start the Dear ImGui frame
|
// Start the Dear ImGui frame
|
||||||
ImGui_ImplSDLRenderer_NewFrame();
|
ImGui_ImplSDLRenderer_NewFrame();
|
||||||
ImGui_ImplSDL2_NewFrame();
|
ImGui_ImplSDL2_NewFrame();
|
||||||
ImGui::NewFrame();
|
ImGui::NewFrame();
|
||||||
|
|
||||||
// Do ImGui things here
|
actions = run_ui(ctx);
|
||||||
show_ui(ctx);
|
|
||||||
|
|
||||||
// Rendering
|
// Rendering
|
||||||
ImGui::Render();
|
ImGui::Render();
|
||||||
auto [r, g, b, a] = imvec4_to_rgba(clear_color);
|
auto [r, g, b, a] = imvec4_to_rgba(clear_color);
|
||||||
SDL_SetRenderDrawColor(ctx->renderer, r, g, b, a);
|
SDL_SetRenderDrawColor(ctx.renderer, r, g, b, a);
|
||||||
SDL_RenderClear(ctx->renderer);
|
SDL_RenderClear(ctx.renderer);
|
||||||
ImGui_ImplSDLRenderer_RenderDrawData(ImGui::GetDrawData());
|
ImGui_ImplSDLRenderer_RenderDrawData(ImGui::GetDrawData());
|
||||||
SDL_RenderPresent(ctx->renderer);
|
SDL_RenderPresent(ctx.renderer);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void log_to_context(log_Event *ev)
|
||||||
|
{
|
||||||
|
auto ctx = reinterpret_cast<ControllerContext *>(ev->udata);
|
||||||
|
ctx->appLog.AddLog(ev->fmt, ev->ap);
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
(void) argc;
|
(void) argc;
|
||||||
(void) argv;
|
(void) argv;
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
|
log_set_level(LOG_TRACE);
|
||||||
|
#else
|
||||||
|
log_set_level(LOG_DEBUG);
|
||||||
|
#endif
|
||||||
|
|
||||||
log_info("doompanning ctrl starting");
|
log_info("doompanning ctrl starting");
|
||||||
|
|
||||||
if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_TIMER))
|
if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_TIMER))
|
||||||
|
@ -150,7 +225,7 @@ int main(int argc, char *argv[])
|
||||||
SDL_SetHint(SDL_HINT_IME_SHOW_UI, "1");
|
SDL_SetHint(SDL_HINT_IME_SHOW_UI, "1");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
const auto windowFlags = SDL_WINDOW_RESIZABLE | SDL_WINDOW_ALLOW_HIGHDPI;
|
const auto windowFlags = SDL_WINDOW_RESIZABLE | SDL_WINDOW_ALLOW_HIGHDPI | SDL_WINDOW_OPENGL;
|
||||||
auto window = SDL_CreateWindow("doompanning", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 1280, 720, windowFlags);
|
auto window = SDL_CreateWindow("doompanning", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 1280, 720, windowFlags);
|
||||||
if (!window)
|
if (!window)
|
||||||
dp_sdl_fatal("SDL_CreateWindow");
|
dp_sdl_fatal("SDL_CreateWindow");
|
||||||
|
@ -172,16 +247,20 @@ int main(int argc, char *argv[])
|
||||||
auto pubSock = make_ctrl_pub(CtrlUrl);
|
auto pubSock = make_ctrl_pub(CtrlUrl);
|
||||||
auto subSock = make_ctrl_sub(DoomUrl);
|
auto subSock = make_ctrl_sub(DoomUrl);
|
||||||
|
|
||||||
ControllerContext ctx
|
ControllerContext ctx =
|
||||||
{
|
{
|
||||||
pubSock,
|
pubSock,
|
||||||
subSock,
|
subSock,
|
||||||
window,
|
window,
|
||||||
renderer,
|
renderer,
|
||||||
{}
|
{},
|
||||||
|
{},
|
||||||
|
{},
|
||||||
};
|
};
|
||||||
|
|
||||||
int ret = doom_controller_loop(&ctx);
|
log_add_callback(log_to_context, &ctx, LOG_TRACE);
|
||||||
|
|
||||||
|
int ret = doom_controller_loop(ctx);
|
||||||
|
|
||||||
nng_close(pubSock);
|
nng_close(pubSock);
|
||||||
nng_close(subSock);
|
nng_close(subSock);
|
||||||
|
|
154
src/dp_util.hpp
154
src/dp_util.hpp
|
@ -1,8 +1,13 @@
|
||||||
#ifndef SRC_DP_UTIL_HPP
|
#ifndef SRC_DP_UTIL_HPP
|
||||||
#define SRC_DP_UTIL_HPP
|
#define SRC_DP_UTIL_HPP
|
||||||
|
|
||||||
#include <imgui.h>
|
#include <array>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstdarg>
|
||||||
#include <tuple>
|
#include <tuple>
|
||||||
|
|
||||||
|
#include <imgui.h>
|
||||||
|
|
||||||
#include "dp_types.h"
|
#include "dp_types.h"
|
||||||
|
|
||||||
inline std::tuple<u8, u8, u8, u8> imvec4_to_rgba(const ImVec4 &color)
|
inline std::tuple<u8, u8, u8, u8> imvec4_to_rgba(const ImVec4 &color)
|
||||||
|
@ -16,4 +21,151 @@ inline std::tuple<u8, u8, u8, u8> imvec4_to_rgba(const ImVec4 &color)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<size_t Size>
|
||||||
|
[[maybe_unused]] const char *aprintf(std::array<char, Size> &buf, const char *fmt, ...)
|
||||||
|
{
|
||||||
|
std::va_list args;
|
||||||
|
va_start(args, fmt);
|
||||||
|
std::vsnprintf(buf.data(), buf.size(), fmt, args);
|
||||||
|
va_end(args);
|
||||||
|
return buf.data();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Usage:
|
||||||
|
// static ExampleAppLog my_log;
|
||||||
|
// my_log.AddLog("Hello %d world\n", 123);
|
||||||
|
// my_log.Draw("title");
|
||||||
|
struct ExampleAppLog
|
||||||
|
{
|
||||||
|
ImGuiTextBuffer Buf;
|
||||||
|
ImGuiTextFilter Filter;
|
||||||
|
ImVector<int> LineOffsets; // Index to lines offset. We maintain this with AddLog() calls.
|
||||||
|
bool AutoScroll; // Keep scrolling if already at the bottom.
|
||||||
|
|
||||||
|
ExampleAppLog()
|
||||||
|
{
|
||||||
|
AutoScroll = true;
|
||||||
|
Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Clear()
|
||||||
|
{
|
||||||
|
Buf.clear();
|
||||||
|
LineOffsets.clear();
|
||||||
|
LineOffsets.push_back(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddLog(const char* fmt, ...) IM_FMTARGS(2)
|
||||||
|
{
|
||||||
|
int old_size = Buf.size();
|
||||||
|
va_list args;
|
||||||
|
va_start(args, fmt);
|
||||||
|
Buf.appendfv(fmt, args);
|
||||||
|
va_end(args);
|
||||||
|
for (int new_size = Buf.size(); old_size < new_size; old_size++)
|
||||||
|
if (Buf[old_size] == '\n')
|
||||||
|
LineOffsets.push_back(old_size + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddLog(const char *fmt, va_list args)
|
||||||
|
{
|
||||||
|
int old_size = Buf.size();
|
||||||
|
Buf.appendfv(fmt, args);
|
||||||
|
Buf.append("\n");
|
||||||
|
for (int new_size = Buf.size(); old_size < new_size; old_size++)
|
||||||
|
if (Buf[old_size] == '\n')
|
||||||
|
LineOffsets.push_back(old_size + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Draw(const char* title, bool* p_open = NULL)
|
||||||
|
{
|
||||||
|
if (!ImGui::Begin(title, p_open))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Options menu
|
||||||
|
if (ImGui::BeginPopup("Options"))
|
||||||
|
{
|
||||||
|
ImGui::Checkbox("Auto-scroll", &AutoScroll);
|
||||||
|
ImGui::EndPopup();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Main window
|
||||||
|
if (ImGui::Button("Options"))
|
||||||
|
ImGui::OpenPopup("Options");
|
||||||
|
ImGui::SameLine();
|
||||||
|
bool clear = ImGui::Button("Clear");
|
||||||
|
ImGui::SameLine();
|
||||||
|
bool copy = ImGui::Button("Copy");
|
||||||
|
ImGui::SameLine();
|
||||||
|
Filter.Draw("Filter", -100.0f);
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
if (ImGui::BeginChild("scrolling", ImVec2(0, 0), false, ImGuiWindowFlags_HorizontalScrollbar))
|
||||||
|
{
|
||||||
|
if (clear)
|
||||||
|
Clear();
|
||||||
|
if (copy)
|
||||||
|
ImGui::LogToClipboard();
|
||||||
|
|
||||||
|
ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing, ImVec2(0, 0));
|
||||||
|
const char* buf = Buf.begin();
|
||||||
|
const char* buf_end = Buf.end();
|
||||||
|
if (Filter.IsActive())
|
||||||
|
{
|
||||||
|
// In this example we don't use the clipper when Filter is enabled.
|
||||||
|
// This is because we don't have random access to the result of our filter.
|
||||||
|
// A real application processing logs with ten of thousands of entries may want to store the result of
|
||||||
|
// search/filter.. especially if the filtering function is not trivial (e.g. reg-exp).
|
||||||
|
for (int line_no = 0; line_no < LineOffsets.Size; line_no++)
|
||||||
|
{
|
||||||
|
const char* line_start = buf + LineOffsets[line_no];
|
||||||
|
const char* line_end = (line_no + 1 < LineOffsets.Size) ? (buf + LineOffsets[line_no + 1] - 1) : buf_end;
|
||||||
|
if (Filter.PassFilter(line_start, line_end))
|
||||||
|
ImGui::TextUnformatted(line_start, line_end);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// The simplest and easy way to display the entire buffer:
|
||||||
|
// ImGui::TextUnformatted(buf_begin, buf_end);
|
||||||
|
// And it'll just work. TextUnformatted() has specialization for large blob of text and will fast-forward
|
||||||
|
// to skip non-visible lines. Here we instead demonstrate using the clipper to only process lines that are
|
||||||
|
// within the visible area.
|
||||||
|
// If you have tens of thousands of items and their processing cost is non-negligible, coarse clipping them
|
||||||
|
// on your side is recommended. Using ImGuiListClipper requires
|
||||||
|
// - A) random access into your data
|
||||||
|
// - B) items all being the same height,
|
||||||
|
// both of which we can handle since we have an array pointing to the beginning of each line of text.
|
||||||
|
// When using the filter (in the block of code above) we don't have random access into the data to display
|
||||||
|
// anymore, which is why we don't use the clipper. Storing or skimming through the search result would make
|
||||||
|
// it possible (and would be recommended if you want to search through tens of thousands of entries).
|
||||||
|
ImGuiListClipper clipper;
|
||||||
|
clipper.Begin(LineOffsets.Size);
|
||||||
|
while (clipper.Step())
|
||||||
|
{
|
||||||
|
for (int line_no = clipper.DisplayStart; line_no < clipper.DisplayEnd; line_no++)
|
||||||
|
{
|
||||||
|
const char* line_start = buf + LineOffsets[line_no];
|
||||||
|
const char* line_end = (line_no + 1 < LineOffsets.Size) ? (buf + LineOffsets[line_no + 1] - 1) : buf_end;
|
||||||
|
ImGui::TextUnformatted(line_start, line_end);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
clipper.End();
|
||||||
|
}
|
||||||
|
ImGui::PopStyleVar();
|
||||||
|
|
||||||
|
// Keep up at the bottom of the scroll region if we were already at the bottom at the beginning of the frame.
|
||||||
|
// Using a scrollbar or mouse-wheel will take away from the bottom edge.
|
||||||
|
if (AutoScroll && ImGui::GetScrollY() >= ImGui::GetScrollMaxY())
|
||||||
|
ImGui::SetScrollHereY(1.0f);
|
||||||
|
}
|
||||||
|
ImGui::EndChild();
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
#endif // SRC_DP_UTIL_HPP
|
#endif // SRC_DP_UTIL_HPP
|
Loading…
Reference in a new issue