#include "imgui.hpp" #include "webgpu/gpu.hpp" #include "internal.hpp" #include "window.hpp" #include #include #include "../imgui/backends/imgui_impl_sdl.cpp" // NOLINT(bugprone-suspicious-include) #include "../imgui/backends/imgui_impl_sdlrenderer.cpp" // NOLINT(bugprone-suspicious-include) #include "../imgui/backends/imgui_impl_wgpu.cpp" // NOLINT(bugprone-suspicious-include) namespace aurora::imgui { static float g_scale; static std::string g_imguiSettings{}; static std::string g_imguiLog{}; static bool g_useSdlRenderer = false; static std::vector g_sdlTextures; static std::vector g_wgpuTextures; void create_context() noexcept { IMGUI_CHECKVERSION(); ImGui::CreateContext(); ImGuiIO& io = ImGui::GetIO(); g_imguiSettings = std::string{g_config.configPath} + "/imgui.ini"; g_imguiLog = std::string{g_config.configPath} + "/imgui.log"; io.IniFilename = g_imguiSettings.c_str(); io.LogFilename = g_imguiLog.c_str(); } void initialize() noexcept { SDL_Renderer* renderer = window::get_sdl_renderer(); ImGui_ImplSDL2_Init(window::get_sdl_window(), renderer); #ifdef __APPLE__ // Disable MouseCanUseGlobalState for scaling purposes ImGui_ImplSDL2_GetBackendData()->MouseCanUseGlobalState = false; #endif g_useSdlRenderer = renderer != nullptr; if (g_useSdlRenderer) { ImGui_ImplSDLRenderer_Init(renderer); } else { ImGui_ImplWGPU_Init(webgpu::g_device, 1, webgpu::g_graphicsConfig.colorFormat); } } void shutdown() noexcept { if (g_useSdlRenderer) { ImGui_ImplSDLRenderer_Shutdown(); } else { ImGui_ImplWGPU_Shutdown(); } ImGui_ImplSDL2_Shutdown(); ImGui::DestroyContext(); for (const auto& texture : g_sdlTextures) { SDL_DestroyTexture(texture); } g_sdlTextures.clear(); for (const auto& texture : g_wgpuTextures) { wgpuTextureDestroy(texture); } g_wgpuTextures.clear(); } void process_event(const SDL_Event& event) noexcept { #ifdef __APPLE__ if (event.type == SDL_MOUSEMOTION) { auto& io = ImGui::GetIO(); // Scale up mouse coordinates io.AddMousePosEvent(static_cast(event.motion.x) * g_scale, static_cast(event.motion.y) * g_scale); return; } #endif ImGui_ImplSDL2_ProcessEvent(&event); } void new_frame(const AuroraWindowSize& size) noexcept { if (g_useSdlRenderer) { ImGui_ImplSDLRenderer_NewFrame(); } else { if (g_scale != size.scale) { if (g_scale > 0.f) { // TODO wgpu backend bug: doesn't clear bind groups on invalidate g_resources.ImageBindGroups.Clear(); ImGui_ImplWGPU_CreateDeviceObjects(); } g_scale = size.scale; } ImGui_ImplWGPU_NewFrame(); } ImGui_ImplSDL2_NewFrame(); // Render at full DPI ImGui::GetIO().DisplaySize = { static_cast(size.fb_width), static_cast(size.fb_height), }; ImGui::NewFrame(); } void render(WGPURenderPassEncoder pass) noexcept { ImGui::Render(); auto* data = ImGui::GetDrawData(); // io.DisplayFramebufferScale is informational; we're rendering at full DPI data->FramebufferScale = {1.f, 1.f}; if (g_useSdlRenderer) { SDL_Renderer* renderer = ImGui_ImplSDLRenderer_GetBackendData()->SDLRenderer; SDL_RenderClear(renderer); ImGui_ImplSDLRenderer_RenderDrawData(data); SDL_RenderPresent(renderer); } else { ImGui_ImplWGPU_RenderDrawData(data, pass); } } ImTextureID add_texture(uint32_t width, uint32_t height, const uint8_t* data) noexcept { if (g_useSdlRenderer) { SDL_Renderer* renderer = ImGui_ImplSDLRenderer_GetBackendData()->SDLRenderer; SDL_Texture* texture = SDL_CreateTexture(renderer, SDL_PIXELFORMAT_RGBA32, SDL_TEXTUREACCESS_STATIC, width, height); SDL_UpdateTexture(texture, nullptr, data, width * 4); SDL_SetTextureScaleMode(texture, SDL_ScaleModeLinear); g_sdlTextures.push_back(texture); return texture; } const auto size = WGPUExtent3D{ .width = width, .height = height, .depthOrArrayLayers = 1, }; const auto textureDescriptor = WGPUTextureDescriptor{ .label = "imgui texture", .usage = WGPUTextureUsage_TextureBinding | WGPUTextureUsage_CopyDst, .dimension = WGPUTextureDimension_2D, .size = size, .format = WGPUTextureFormat_RGBA8Unorm, .mipLevelCount = 1, .sampleCount = 1, }; const auto textureViewDescriptor = WGPUTextureViewDescriptor{ .label = "imgui texture view", .format = WGPUTextureFormat_RGBA8Unorm, .dimension = WGPUTextureViewDimension_2D, .mipLevelCount = WGPU_MIP_LEVEL_COUNT_UNDEFINED, .arrayLayerCount = WGPU_ARRAY_LAYER_COUNT_UNDEFINED, }; auto texture = wgpuDeviceCreateTexture(webgpu::g_device, &textureDescriptor); auto textureView = wgpuTextureCreateView(texture, &textureViewDescriptor); { const auto dstView = WGPUImageCopyTexture{ .texture = texture, }; const auto dataLayout = WGPUTextureDataLayout{ .bytesPerRow = 4 * width, .rowsPerImage = height, }; wgpuQueueWriteTexture(webgpu::g_queue, &dstView, data, width * height * 4, &dataLayout, &size); } g_wgpuTextures.push_back(texture); return textureView; } } // namespace aurora::imgui // C bindings extern "C" { ImTextureID aurora_imgui_add_texture(uint32_t width, uint32_t height, const void* rgba8) { return aurora::imgui::add_texture(width, height, static_cast(rgba8)); } }