ImGui Integration
Update Swapchain
to expose its image format:
[[nodiscard]] auto get_format() const -> vk::Format {
return m_ci.imageFormat;
}
class App
can now store a std::optional<DearImGui>
member and add/call its create function:
void App::create_imgui() {
auto const imgui_ci = DearImGui::CreateInfo{
.window = m_window.get(),
.api_version = vk_version_v,
.instance = *m_instance,
.physical_device = m_gpu.device,
.queue_family = m_gpu.queue_family,
.device = *m_device,
.queue = m_queue,
.color_format = m_swapchain->get_format(),
.samples = vk::SampleCountFlagBits::e1,
};
m_imgui.emplace(imgui_ci);
}
Start a new ImGui frame after resetting the render fence, and show the demo window:
m_device->resetFences(*render_sync.drawn);
m_imgui->new_frame();
// ...
command_buffer.beginRendering(rendering_info);
ImGui::ShowDemoWindow();
// draw stuff here.
command_buffer.endRendering();
ImGui doesn't draw anything here (the actual draw command requires the Command Buffer), it's just a good customization point for all higher level logic.
We use a separate render pass for Dear ImGui, again for isolation, and to enable us to change the main render pass later, eg by adding a depth buffer attachment (DearImGui
is setup assuming its render pass will only use a single color attachment).
m_imgui->end_frame();
// we don't want to clear the image again, instead load it intact after the
// previous pass.
color_attachment.setLoadOp(vk::AttachmentLoadOp::eLoad);
rendering_info.setColorAttachments(color_attachment)
.setPDepthAttachment(nullptr);
command_buffer.beginRendering(rendering_info);
m_imgui->render(command_buffer);
command_buffer.endRendering();