Unreal ImGui ============ [![MIT licensed](https://img.shields.io/badge/license-MIT-blue.svg)](LICENSE.md) Unreal ImGui is an Unreal Engine 5 plug-in that integrates [Dear ImGui](https://github.com/ocornut/imgui) developed by Omar Cornut. Dear ImGui is an immediate-mode graphical user interface library that is very lightweight and easy to use. It can be very useful when creating debugging tools. :stop_button: Read Me First --------------------------- This is a fork of [https://github.com/benui-dev/UnrealImGui](https://github.com/benui-dev/UnrealImGui), which itself is a fork of the original UnrealImGui project by [https://github.com/segross/UnrealImGui](https://github.com/segross/UnrealImGui). You can view the original readme.md, please see this link: [UnrealImGui ReadMe.md](https://github.com/benui-dev/UnrealImGui/blob/master/README.md). Fork Additions/Fixes -------------------- - Updated Dear ImGui to 1.91.5-docking - Added configurable docking enabled setting TO-DO ----- - [X] Update Dear ImGui to latest 'docking' branch release (which as of writing is 1.91.5-docking) - [X] Update input state to use new ImGuiKey API - [ ] Update gamepad navigation to use new ImGuiKey API - [X] Enable docking - [ ] Support full-screen dock space - [ ] Add NetImGui support Status ------ UnrealImGui Version: 1.22 ImGui version: 1.91.5-docking ImPlot version: v0.13 WIP Supported Unreal Engine version: 5.4* \* *The original repository and Ben's fork that this project is based on has support for Unreal 4.26 to 5.3. As of writing I am currently using this plugin on Unreal 5.4 with no issues. I cannot guarantee compatibility with versions other than the ones previously listed, so use this plugin with other versions of Unreal at your own risk.* How to Set up ------------- On top of reading the base repository's [How to Set up](https://github.com/segross/UnrealImGui/blob/master/README.md#how-to-set-up) segment, you'll need to add the following line to your `[GameName].Build.cs` file otherwise you'll get linking errors: ```cpp // Tell the compiler we want to import the ImPlot symbols when linking against ImGui plugin PrivateDefinitions.Add(string.Format("IMPLOT_API=DLLIMPORT")); ``` # Additional Knowledge ## Using ImPlot It's pretty easy to use ImPlot, it's pretty much the same drill as using Dear ImGui with the UnrealImGui plugin. You can see documentation on how to use ImPlot here: [ImPlot](https://github.com/epezent/implot). The only thing you won't need to do is call the `ImPlot::CreateContext()` and `ImPlot::DestroyContext` routines as they're already called when ImGui's context is created within UnrealImGui's guts. ## Drawing a UTextureRenderTarget2D One might want to render viewports into the world in an ImGui window. You can do this pretty simply by generating a `UTextureRenderTarget2D` then assigning that to a `ASceneCapture2D` actor in your world. Here's some sample code for generating an correctly managing the `UTextureRenderTarget2D`: ```cpp void Init() { TextureRenderTarget = NewObject(); if(IsValid(TextureRenderTarget)) { TextureRenderTarget->InitAutoFormat(512, 512); TextureRenderTarget->UpdateResourceImmediate(true); } // ... Generate a unique TextureName here // Register this render target as an ImGui interop handled texture ImGuiTextureHandle = FImGuiModule::Get().FindTextureHandle(TextureName); if(!ImGuiTextureHandle.IsValid()) { if(IsValid(TextureRenderTarget)) { ImGuiTextureHandle = FImGuiModule::Get().RegisterTexture(TextureName, TextureRenderTarget, true); } } } ~Class() { // Requires releasing to avoid memory leak FImGuiModule::Get().ReleaseTexture(ImGuiTextureHandle); } void Render() { // Actually submit the draw command to ImGui to render the quad with the texture if(ImGuiTextureHandle.IsValid()) { ImGui::Image(ImGuiTextureHandle.GetTextureId(), {512.f, 512.f}); } } ``` Then generating the `ASceneCapture2D`: ```cpp void Init() { FActorSpawnParameters SpawnInfo; SceneCapture2D = World->SpawnActor(FVector::ZeroVector, FRotator::ZeroRotator, SpawnInfo); SceneCaptureComponent2D->TextureTarget = TextureRenderTarget; SceneCaptureComponent2D->UpdateContent(); // Need to use this in order to force capture to use tone curve and also set alpha to scene alpha (1) SceneCaptureComponent2D->CaptureSource = ESceneCaptureSource::SCS_FinalToneCurveHDR; } ``` ### Troubleshooting If you're using a scene capture and your quad is not drawing at all, make sure your scene capture "Capture Source" is set to "Final Color (with tone curve) in Linear sRGB gamut" to avoid alpha being set to 0 (since there's no way to instruct ImGui to ignore alpha without modding the core UnrealImGui plugin). If you're getting crashes or seg faults during rendering, make sure you're using `UPROPERTY()` on your class variables! ## Adding custom fonts ### FontAwesome Adding custom fonts is fairly simple. As a more complex and more commonly done, we're going to embed and build FontAwesome 6 into the font atlas. First thing you'll need is a binary C version of the FontAwesome font along with the necessary [companion descriptors](https://github.com/juliettef/IconFontCppHeaders/blob/main/IconsFontAwesome6.h). The descriptors are pre-generated, however if you have a new version of FA you wish to use, then use the Python script in that repository. As for the binary C, you'll need to compile Dear ImGui's [binary_to_compressed_c.cpp](https://github.com/ocornut/imgui/blob/master/misc/fonts/binary_to_compressed_c.cpp). Once you have the necessary files, you'll need to encode your FontAwesome font using the command: ``` binary_to_compressed_c.exe -nocompress fa-solid-900.ttf FontAwesomeFont > FontAwesomeFont.h ``` The only mandatory field here is `-nocompress` as this instructs the encoder to create an uncompressed binary file (1:1) since currently there's no immediate support for compressed fonts. Move over your descriptors and your binary C font file to an appropriate location for inclusion in your Unreal Engine project. The following code is how to instruct ImGui to build the font atlas with your FontAwesome font: ```cpp #include "FontAwesomeFont.h" #include "IconsFontAwesome6.h" // Add FontAwesome font glyphs from memory if(TSharedPtr FAFontConfig = MakeShareable(new ImFontConfig())) { static const ImWchar IconRange[] = { ICON_MIN_FA, ICON_MAX_FA, 0 }; FAFontConfig->FontDataOwnedByAtlas = false; // Global font data lifetime FAFontConfig->FontData = (void*)FontAwesomeData; // Declared in binary C .h file FAFontConfig->FontDataSize = FontAwesomeSize; // Declared in binary C .h file FAFontConfig->SizePixels = 16; FAFontConfig->MergeMode = true; // Forces ImGui to place this font into the same atlas as the previous font FAFontConfig->GlyphRanges = IconRange; // Required; instructs ImGui to use these glyphs FAFontConfig->GlyphMinAdvanceX = 16.f; // Use for monospaced icons FAFontConfig->PixelSnapH = true; // Better rendering (align to pixel grid) FAFontConfig->GlyphOffset = {0, 3}; // Moves icons around, for alignment with general typesets FImGuiModule::Get().GetProperties().AddCustomFont("FontAwesome", FAFontConfig); FImGuiModule::Get().RebuildFontAtlas(); } ``` That's it. Make sure you execute the above code once at the beginning of the first ImGui frame (or at any point of your framework where the ImGui context has been initialized correctly) and it should build the main atlas with FontAwesome inside it. ImFontConfig lifetime is currently managed via reference counting (`TSharedPtr`). ### Using the icons ```cpp #include "IconsFontAwesome6.h" void OnPaint() { ImGui::Text(ICON_FA_AWARD); } ``` Pretty simple. Building FStrings that incorporate FontAwesome icons however gets a little trickier: ```cpp FString Str = "Hello " + FString(UTF8_TO_TCHAR(ICON_FA_WAVE_SQUARE)) " World"; ImGui::TextUnformatted(TCHAR_TO_UTF8(*Str)); ``` ### More info - [Dear ImGui: Using Fonts](https://github.com/ocornut/imgui/blob/master/docs/FONTS.md) - [IconFontCppHeaders](https://github.com/juliettef/IconFontCppHeaders) - [FontAwesome with general Dear ImGui](https://pixtur.github.io/mkdocs-for-imgui/site/FONTS/) # Misc See also -------- - [Fork this project is based on](https://github.com/benui-dev/UnrealImGui) - [Original project by segross](https://github.com/segross/UnrealImGui) - [Dear ImGui](https://github.com/ocornut/imgui) - [ImPlot](https://github.com/epezent/implot) License ------- Unreal ImGui (and this fork) is licensed under the MIT License, see LICENSE for more information.