Skip to content

Commit

Permalink
Work on adding support for netImgui
Browse files Browse the repository at this point in the history
  • Loading branch information
pthom committed Apr 10, 2024
1 parent aa78fab commit 7c4e3b2
Show file tree
Hide file tree
Showing 13 changed files with 497 additions and 6 deletions.
6 changes: 6 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,12 @@ option(HELLOIMGUI_HAS_NULL "Null rendering backend" OFF) # for testing and remo
option(HELLOIMGUI_NULL_BACKEND "Use Null rendering/platform backend" OFF)
# </Backends> # do not remove this line (used by the script that generates the documentation)

#------------------------------------------------------------------------------
# Options / Remoting
#------------------------------------------------------------------------------
# Using https://github.com/sammyfreg/netImgui, you can use HelloImGui with remote rendering
# (this is useful for example to render ImGui on a remote device, or to render ImGui in a web browser)
option(HELLOIMGUI_WITH_NETIMGUI "Use netImgui for remote rendering" OFF)

#------------------------------------------------------------------------------
# Options / Freetype
Expand Down
10 changes: 6 additions & 4 deletions CMakePresets.json
Original file line number Diff line number Diff line change
Expand Up @@ -29,14 +29,16 @@
"HELLOIMGUI_HAS_VULKAN": "ON",
"HELLOIMGUI_HAS_DIRECTX11": "ON",
"HELLOIMGUI_HAS_DIRECTX12": "ON",
"HELLOIMGUI_HAS_NULL": "ON"
"HELLOIMGUI_HAS_NULL": "ON",
"HELLOIMGUI_WITH_NETIMGUI": "ON"
}
},
{
"name": "build_null_backend",
"description": "Build with null backend.",
"name": "build_netimgui",
"description": "Build with netImgui.",
"cacheVariables": {
"HELLOIMGUI_NULL_BACKEND": "ON"
"HELLOIMGUI_NULL_BACKEND": "ON",
"HELLOIMGUI_WITH_NETIMGUI": "ON"
}
},
{
Expand Down
1 change: 1 addition & 0 deletions external/.gitignore
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
SDL
SDL2-*/
qtimgui/
netImgui/
45 changes: 43 additions & 2 deletions hello_imgui_cmake/hello_imgui_build_lib.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -1031,6 +1031,42 @@ function(_him_fetch_glfw_if_needed)
endfunction()


###################################################################################################
# Remoting with https://github.com/sammyfreg/netImgui: API = him_with_netimgui
###################################################################################################
function(him_with_netimgui)
target_compile_definitions(${HELLOIMGUI_TARGET} PUBLIC HELLOIMGUI_WITH_NETIMGUI)

# message(STATUS "HelloImGui: downloading and building netImgui")
# include(FetchContent)
# # Set(FETCHCONTENT_QUIET FALSE)
# FetchContent_Declare(net_imgui
# GIT_REPOSITORY https://github.com/pthom/netImgui.git
# GIT_TAG cmake_multiplatform
# GIT_PROGRESS TRUE
# )
# FetchContent_MakeAvailable(net_imgui)


# Add netImgui to the project
set(NETIMGUI_DIR ${HELLOIMGUI_BASEPATH}/external/netImgui CACHE STRING "" FORCE)
set(NETIMGUI_BUILD_IMGUI OFF CACHE BOOL "" FORCE)
set(NETIMGUI_BUILD_CLIENT ON CACHE BOOL "" FORCE)
set(NETIMGUI_BUILD_SERVER_LIB ON CACHE BOOL "" FORCE)

#set(NETIMGUI_BUILD_SERVER_APP_SOKOL ON CACHE BOOL "" FORCE)
set(NETIMGUI_BUILD_SERVER_APP_SOKOL OFF CACHE BOOL "" FORCE)
set(NETIMGUI_SERVER_APP_BACKEND_GLFW_GL3 ON CACHE BOOL "" FORCE)

set(NETIMGUI_BUILD_SAMPLES OFF CACHE BOOL "" FORCE)
add_subdirectory(${NETIMGUI_DIR} netimgui)

target_link_libraries(${HELLOIMGUI_TARGET} PUBLIC net_imgui_client)
him_add_installable_dependency(net_imgui_client)
endfunction()



###################################################################################################
# Miscellaneous: API = him_add_misc_options
###################################################################################################
Expand Down Expand Up @@ -1083,8 +1119,9 @@ function(him_log_configuration)
===========================================================================
Hello ImGui build options:
===========================================================================
Platform Backend(s): ${active_platform_backends}
Rendering Backend(s): ${active_rendering_backends}
Platform Backend(s): ${active_platform_backends}
Rendering Backend(s): ${active_rendering_backends}
Use netImGui: ${HELLOIMGUI_WITH_NETIMGUI}
---------------------------------------------------------------------------
Options:
HELLOIMGUI_USE_FREETYPE: ${HELLOIMGUI_USE_FREETYPE} (${HELLOIMGUI_FREETYPE_SELECTED_INFO})
Expand Down Expand Up @@ -1185,6 +1222,10 @@ function(him_main_add_hello_imgui_library)
target_compile_definitions(${HELLOIMGUI_TARGET} PUBLIC HELLOIMGUI_HAS_NULL)
endif()

if (HELLOIMGUI_WITH_NETIMGUI)
him_with_netimgui()
endif()

him_add_apple_options()
him_add_linux_options()
him_add_windows_options()
Expand Down
3 changes: 3 additions & 0 deletions src/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -6,3 +6,6 @@ endif()
if(HELLOIMGUI_BUILD_TESTS)
add_subdirectory(hello_imgui_tests)
endif()
if(HELLOIMGUI_WITH_NETIMGUI AND HELLOIMGUI_HAS_OPENGL3 AND HELLOIMGUI_USE_GLFW3)
add_subdirectory(netimgui_remote_display)
endif()
6 changes: 6 additions & 0 deletions src/hello_imgui/impl/hello_imgui.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -197,6 +197,8 @@ std::string PlatformBackendTypeToString(PlatformBackendType platformBackendType)
return "Glfw";
else if (platformBackendType == PlatformBackendType::Sdl)
return "Sdl";
else if (platformBackendType == PlatformBackendType::Null)
return "Null";
else
return "Unknown platform backend";
}
Expand All @@ -213,13 +215,17 @@ std::string RendererBackendTypeToString(RendererBackendType rendererBackendType)
return "DirectX11";
else if (rendererBackendType == RendererBackendType::DirectX12)
return "DirectX12";
else if (rendererBackendType == RendererBackendType::Null)
return "Null";
else
return "Unknown renderer backend";
}

std::string GetBackendDescription()
{
const auto& params = GetRunnerParams();
if (params->remoteParams.enableRemoting)
return "Remote";
std::string platformBackend = PlatformBackendTypeToString(params->platformBackendType);
std::string rendererBackend = RendererBackendTypeToString(params->rendererBackendType);
return platformBackend + " - " + rendererBackend;
Expand Down
106 changes: 106 additions & 0 deletions src/hello_imgui/internal/backend_impls/abstract_runner.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,11 +21,16 @@
#define SCOPED_RELEASE_GIL_ON_MAIN_THREAD
#endif

#ifdef HELLOIMGUI_WITH_NETIMGUI
#include "NetImgui_Api.h"
#endif

#include <chrono>
#include <cassert>
#include <filesystem>
#include <cstdio>
#include <optional>
#include <thread>

#if __APPLE__
#include <TargetConditionals.h>
Expand Down Expand Up @@ -680,8 +685,97 @@ void AbstractRunner::Setup()
style.Colors[ImGuiCol_TitleBgCollapsed].w = 1.f;
}
params.callbacks.SetupImGuiStyle();

#ifdef HELLOIMGUI_WITH_NETIMGUI
if (params.remoteParams.enableRemoting)
{
if (!NetImGui_Connect())
{
IM_ASSERT(false && "NetImGui_Connect() failed! Please launch the server/display app before!");
exit(1);
}
}
#endif
}


void AbstractRunner::NetImGui_LogConnectionStatusOnce()
{
if (! params.remoteParams.enableRemoting)
return;
#ifdef HELLOIMGUI_WITH_NETIMGUI
// Log connection status and return
if (NetImgui::IsConnectionPending())
{
static bool wasMessageShown = false;
if (!wasMessageShown)
{
wasMessageShown = true;
printf("Waiting for NetImgui connection...\n");
}
}
else if (NetImgui::IsConnected())
{
static bool wasMessageShown = false;
if (!wasMessageShown)
{
wasMessageShown = true;
printf("Connected to NetImgui server\n");
}
}
#endif
}


bool AbstractRunner::NetImGui_Connect()
{
#ifdef HELLOIMGUI_WITH_NETIMGUI
// 0. Startup NetImgui
NetImgui::Startup();
// 1. Connect to NetImgui server
std::string clientName = params.remoteParams.clientName;
if (clientName.empty())
clientName = params.appWindowParams.windowTitle;
if (clientName.empty())
clientName = "HelloImGui";
bool clientActiveImmediately = NetImgui::ConnectToApp(clientName.c_str(), params.remoteParams.serverHost.c_str(), params.remoteParams.serverPort);

// 2. Wait a bit for the connection to be established (this is optional, just to display the messages after)
if (!clientActiveImmediately)
{
printf("Sleeping 0.1s to wait for NetImgui connection...\n");
std::this_thread::sleep_for(std::chrono::milliseconds(100));
}

// 3. Log connection status (this is optional also, since this is called also inside CreateFramesAndRender)
NetImGui_LogConnectionStatusOnce();
if (!NetImgui::IsConnected() && !NetImgui::IsConnectionPending())
{
printf("Connection to NetImgui server failed!\n");
return false;
}

// 4. Send Font texture to NetImgui server
const ImFontAtlas* pFonts = ImGui::GetIO().Fonts;
if( pFonts->TexPixelsAlpha8) // && (pFonts->TexPixelsAlpha8 != client.mpFontTextureData || client.mFontTextureID != pFonts->TexID ))
{
uint8_t* pPixelData(nullptr); int width(0), height(0);
ImGui::GetIO().Fonts->GetTexDataAsAlpha8(&pPixelData, &width, &height);
NetImgui::SendDataTexture(pFonts->TexID, pPixelData, static_cast<uint16_t>(width), static_cast<uint16_t>(height), NetImgui::eTexFormat::kTexFmtA8);
}
if( pFonts->TexPixelsRGBA32) // && (pFonts->TexPixelsAlpha8 != client.mpFontTextureData || client.mFontTextureID != pFonts->TexID ))
{
uint8_t* pPixelData(nullptr); int width(0), height(0);
ImGui::GetIO().Fonts->GetTexDataAsRGBA32(&pPixelData, &width, &height);
NetImgui::SendDataTexture(pFonts->TexID, pPixelData, static_cast<uint16_t>(width), static_cast<uint16_t>(height), NetImgui::eTexFormat::kTexFmtRGBA8);
}
return true;
#else
return false;
#endif
}


void AbstractRunner::SetLayoutResetIfNeeded()
{
if (params.imGuiWindowParams.defaultImGuiWindowType == DefaultImGuiWindowType::ProvideFullScreenDockSpace)
Expand Down Expand Up @@ -775,6 +869,11 @@ void AbstractRunner::CreateFramesAndRender()
// (it is here only to make the code more readable, and to enable to collapse blocks of code)
constexpr bool foldable_region = true;

#ifdef HELLOIMGUI_WITH_NETIMGUI
if (params.remoteParams.enableRemoting)
NetImGui_LogConnectionStatusOnce();
#endif

if (foldable_region) // Update frame rate stats
{
_UpdateFrameRateStats();
Expand Down Expand Up @@ -1141,6 +1240,13 @@ void AbstractRunner::TearDown(bool gotException)
if (params.useImGuiTestEngine)
TestEngineCallbacks::TearDown_ImGuiContextDestroyed();
#endif

#ifdef HELLOIMGUI_WITH_NETIMGUI
if (params.remoteParams.enableRemoting)
{
NetImgui::Shutdown();
}
#endif
}


Expand Down
3 changes: 3 additions & 0 deletions src/hello_imgui/internal/backend_impls/abstract_runner.h
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,9 @@ class AbstractRunner
void LayoutSettings_Load();
void LayoutSettings_Save();

bool NetImGui_Connect();
void NetImGui_LogConnectionStatusOnce();

public:
BackendApi::WindowPointer mWindow = nullptr;
protected:
Expand Down
10 changes: 10 additions & 0 deletions src/hello_imgui/internal/backend_impls/runner_factory.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -39,10 +39,20 @@ void ChooseBackendTypesIfSelectedAsFirstAvailable(RunnerParams* runnerParams)
}
}

void ChooseNullBackendsIfUsingRemote(RunnerParams* runnerParams)
{
if (runnerParams->remoteParams.enableRemoting)
{
runnerParams->platformBackendType = PlatformBackendType::Null;
runnerParams->rendererBackendType = RendererBackendType::Null;
}
}


std::unique_ptr<AbstractRunner> FactorRunner(RunnerParams& params)
{
ChooseBackendTypesIfSelectedAsFirstAvailable(&params);
ChooseNullBackendsIfUsingRemote(&params);
if (params.platformBackendType == PlatformBackendType::Glfw)
{
#ifdef HELLOIMGUI_USE_GLFW3
Expand Down
32 changes: 32 additions & 0 deletions src/hello_imgui/runner_params.h
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,36 @@ struct FpsIdling

// --------------------------------------------------------------------------------------------------------------------

// --------------------------------------------------------------------------------------------------------------------

// @@md#NetImGuiParams

// NetImGuiParams is a struct that contains the settings for displaying the application on a remote device.
// using https://github.com/sammyfreg/netImgui
// (This will only work if the application is compiled with the option -DHELLOIMGUI_WITH_NETIMGUI=ON)
struct NetImGuiParams
{
bool enableRemoting = false;

// Name of the client (if empty, will use params.appWindowParams.windowTitle)
// (The client is the app that contains the application logic)
std::string clientName = "";

// The server host (if empty, will use "localhost")
// The server is the app that simply displays the application on a remote device
std::string serverHost = "localhost";
// The server port (default is 8888)
uint32_t serverPort = 8888;

// The client port (default is 8889)
// This is unused since it is the client (i.e. the app logic) that connects to the server,
// using NetImgui::ConnectToApp
uint32_t clientPort = 8889;
};

// @@md

// --------------------------------------------------------------------------------------------------------------------

// @@md#RunnerParams

Expand Down Expand Up @@ -195,6 +225,8 @@ struct RunnerParams
// Only useful when multiple rendering backend are compiled and available.
RendererBackendType rendererBackendType = RendererBackendType::FirstAvailable;

// --------------- RemoteParams -------------------
NetImGuiParams remoteParams;


// --------------- Settings -------------------
Expand Down
18 changes: 18 additions & 0 deletions src/netimgui_remote_display/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
if (NOT HELLOIMGUI_WITH_NETIMGUI)
message(FATAL_ERROR "HELLOIMGUI_WITH_NETIMGUI is not enabled")
endif()
if (NOT HELLOIMGUI_HAS_OPENGL3)
message(FATAL_ERROR "HELLOIMGUI_HAS_OPENGL3 is not enabled, and required for hello_imgui_remote_display")
endif()
if (NOT HELLOIMGUI_USE_GLFW3)
message(FATAL_ERROR "HELLOIMGUI_USE_GLFW3 is not enabled, and required for hello_imgui_remote_display")
endif()


set(hal_gl3_cpp_file ${NETIMGUI_DIR}/Code/ServerApp/Source/GlfwGL3/NetImguiServer_HAL_GL3.cpp)
set(sources ${hal_gl3_cpp_file} netimgui_remote_display.cpp NetImguiServer_HAL_Glfw.cpp)
hello_imgui_add_app(netimgui_remote_display ${sources})
target_compile_definitions(net_imgui_server_lib PUBLIC NETIMGUI_SERVER_APP_BACKEND_GLFW_GL3=1)
target_link_libraries(netimgui_remote_display PRIVATE net_imgui_server_lib)
find_package(OpenGL REQUIRED)
target_link_libraries(netimgui_remote_display PRIVATE OpenGL::GL)
Loading

0 comments on commit 7c4e3b2

Please sign in to comment.