#include #include #include #include #include #include #include #include #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::value, "DoomState must be a trivially copyable type"); struct ControllerContext { nng_socket pub; nng_socket sub; SDL_Window *window; SDL_Renderer *renderer; std::vector dooms; bool quit = false; ExampleAppLog appLog; int columns = 4; }; 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; iWorkPos.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::SetNextWindowSize(ImVec2(420, 340), ImGuiCond_FirstUseEver); ImGuiWindowFlags window_flags = ImGuiWindowFlags_MenuBar; static std::array strbuf; auto &io = ImGui::GetIO(); aprintf(strbuf, "doompanning - #dooms=%zu, %.2f ms/frame (%.1f fps)###doompanning", ctx.dooms.size(), 1000.0f / io.Framerate, io.Framerate); // 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) { 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; if (event.type == SDL_WINDOWEVENT && event.window.event == SDL_WINDOWEVENT_CLOSE && event.window.windowID == SDL_GetWindowID(ctx.window)) { 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 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_imgui(log_Event *ev) { auto ctx = reinterpret_cast(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); dp_nng_init_limits(1, 1, 1); auto pubSock = make_ctrl_pub(CtrlUrl); auto subSock = make_ctrl_sub(DoomUrl); ControllerContext ctx = { pubSock, subSock, window, renderer, {}, {}, {}, }; log_add_callback(log_to_imgui, &ctx, LOG_TRACE); int ret = doom_controller_loop(ctx); nng_close(pubSock); nng_close(subSock); return ret; }