doompanning/src/doompanning.cc

269 lines
7 KiB
C++
Raw Normal View History

#include <nng/nng.h>
#include <imgui.h>
#include <backends/imgui_impl_sdl.h>
#include <backends/imgui_impl_sdlrenderer.h>
2023-02-12 17:39:26 +01:00
#include <SDL2/SDL.h>
#include <cstdlib>
#include <type_traits>
#include <vector>
#include "dp_common.h"
#include "dp_util.hpp"
void dp_sdl_fatal(const char *const msg)
{
log_fatal("%s: %s", msg, SDL_GetError());
abort();
}
struct DoomState
{
bool alive = false; // FIXME for testing only
};
static_assert(std::is_trivially_copyable<DoomState>::value, "DoomState must be a trivially copyable type");
2023-02-12 13:40:41 +01:00
struct ControllerContext
{
nng_socket pub;
nng_socket sub;
SDL_Window *window;
SDL_Renderer *renderer;
std::vector<DoomState> dooms;
bool quit = false;
ExampleAppLog appLog;
int columns = 4;
2023-02-12 13:40:41 +01:00
};
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
const bool has_debug_tools = true;
#else
const bool has_debug_tools = false;
#endif
static bool show_app_metrics = false;
static bool show_app_debug_log = false;
static bool show_log_window = true;
if (show_app_metrics)
ImGui::ShowMetricsWindow(&show_app_metrics);
if (show_app_debug_log)
ImGui::ShowDebugLogWindow(&show_app_debug_log);
2023-02-12 16:12:01 +01:00
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");
}
2023-02-12 16:12:01 +01:00
ImGui::SetNextWindowPos(ImVec2(main_viewport->WorkPos.x + 20, main_viewport->WorkPos.y + 20), ImGuiCond_FirstUseEver);
ImGui::SetNextWindowSize(ImVec2(420, 340), ImGuiCond_FirstUseEver);
2023-02-12 16:12:01 +01:00
ImGuiWindowFlags window_flags = ImGuiWindowFlags_MenuBar;
static std::array<char, 1024> strbuf;
2023-02-12 16:12:01 +01:00
auto &io = ImGui::GetIO();
aprintf(strbuf, "doompanning - #dooms=%zu, %.2f ms/frame (%.1f fps)###doompanning",
ctx.dooms.size(), 1000.0f / io.Framerate, io.Framerate);
2023-02-12 16:12:01 +01:00
// Main body of the doompanning window starts here.
if (!ImGui::Begin(strbuf.data(), nullptr, window_flags))
{
// Early out if the window is collapsed, as an optimization.
ImGui::End();
return {};
}
// Menu Bar
if (ImGui::BeginMenuBar())
{
if (ImGui::BeginMenu("Menu"))
{
ImGui::MenuItem("Log Window", nullptr, &show_log_window);
ImGui::MenuItem("Quit", "Ctrl+Q", &ctx.quit, true);
ImGui::EndMenu();
}
if (ImGui::BeginMenu("Tools"))
{
ImGui::MenuItem("Dear ImGui Metrics/Debugger", NULL, &show_app_metrics, has_debug_tools);
ImGui::MenuItem("Dear ImGui Debug Log", NULL, &show_app_debug_log, has_debug_tools);
ImGui::EndMenu();
}
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::End();
return result;
}
int doom_controller_loop(ControllerContext &ctx)
{
2023-02-12 13:40:41 +01:00
static constexpr ImVec4 clear_color = ImVec4(0.45f, 0.55f, 0.60f, 1.00f);
ControllerActions actions = {};
while (!ctx.quit)
{
SDL_Event event;
while (SDL_PollEvent(&event))
{
ImGui_ImplSDL2_ProcessEvent(&event);
if (event.type == SDL_QUIT)
ctx.quit = true;
2023-02-12 13:40:41 +01:00
if (event.type == SDL_WINDOWEVENT
&& event.window.event == SDL_WINDOWEVENT_CLOSE
&& event.window.windowID == SDL_GetWindowID(ctx.window))
2023-02-12 13:40:41 +01:00
{
ctx.quit = true;
2023-02-12 13:40:41 +01:00
}
}
// 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
ImGui_ImplSDLRenderer_NewFrame();
ImGui_ImplSDL2_NewFrame();
ImGui::NewFrame();
actions = run_ui(ctx);
// Rendering
ImGui::Render();
auto [r, g, b, a] = imvec4_to_rgba(clear_color);
SDL_SetRenderDrawColor(ctx.renderer, r, g, b, a);
SDL_RenderClear(ctx.renderer);
ImGui_ImplSDLRenderer_RenderDrawData(ImGui::GetDrawData());
SDL_RenderPresent(ctx.renderer);
}
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[])
{
(void) argc;
(void) argv;
#ifndef NDEBUG
log_set_level(LOG_TRACE);
#else
log_set_level(LOG_DEBUG);
#endif
log_info("doompanning ctrl starting");
if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_TIMER))
dp_sdl_fatal("SDL_Init");
#ifdef SDL_HINT_IME_SHOW_UI
SDL_SetHint(SDL_HINT_IME_SHOW_UI, "1");
#endif
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);
if (!window)
dp_sdl_fatal("SDL_CreateWindow");
auto renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_PRESENTVSYNC | SDL_RENDERER_ACCELERATED);
if (!renderer)
dp_sdl_fatal("SDL_CreateRenderer");
IMGUI_CHECKVERSION();
ImGui::CreateContext();
ImGui::GetIO().IniFilename = "doompanning_ui.ini";
ImGui::StyleColorsDark();
ImGui_ImplSDL2_InitForSDLRenderer(window, renderer);
ImGui_ImplSDLRenderer_Init(renderer);
2023-02-12 13:40:41 +01:00
dp_nng_init_limits(1, 1, 1);
2023-02-04 01:56:30 +01:00
2023-02-12 13:40:41 +01:00
auto pubSock = make_ctrl_pub(CtrlUrl);
auto subSock = make_ctrl_sub(DoomUrl);
2023-02-04 01:56:30 +01:00
ControllerContext ctx =
2023-02-04 01:56:30 +01:00
{
2023-02-12 13:40:41 +01:00
pubSock,
subSock,
window,
renderer,
{},
{},
{},
2023-02-12 13:40:41 +01:00
};
2023-02-04 01:56:30 +01:00
log_add_callback(log_to_context, &ctx, LOG_TRACE);
int ret = doom_controller_loop(ctx);
2023-02-04 01:56:30 +01:00
2023-02-12 13:40:41 +01:00
nng_close(pubSock);
nng_close(subSock);
2023-02-04 01:56:30 +01:00
2023-02-12 13:40:41 +01:00
return ret;
}