2017-08-28 19:29:07 +00:00
|
|
|
// Distributed under the MIT License (MIT) (see accompanying LICENSE file)
|
|
|
|
|
|
|
|
#include "ImGuiPrivatePCH.h"
|
|
|
|
|
|
|
|
#include "ImGuiContextManager.h"
|
|
|
|
|
2019-04-10 19:19:11 +00:00
|
|
|
#include "ImGuiDelegatesContainer.h"
|
2017-09-27 20:16:54 +00:00
|
|
|
#include "ImGuiImplementation.h"
|
2018-01-09 19:36:52 +00:00
|
|
|
#include "Utilities/ScopeGuards.h"
|
2017-09-27 20:16:54 +00:00
|
|
|
#include "Utilities/WorldContext.h"
|
|
|
|
#include "Utilities/WorldContextIndex.h"
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
#include <imgui.h>
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
2017-08-28 19:29:07 +00:00
|
|
|
#if WITH_EDITOR
|
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
// Name for editor ImGui context.
|
|
|
|
FORCEINLINE FString GetEditorContextName()
|
|
|
|
{
|
|
|
|
return TEXT("Editor");
|
|
|
|
}
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
// Name for world ImGui context.
|
|
|
|
FORCEINLINE FString GetWorldContextName(const UWorld& World)
|
2017-08-28 19:29:07 +00:00
|
|
|
{
|
2017-09-27 20:16:54 +00:00
|
|
|
using namespace Utilities;
|
|
|
|
|
|
|
|
const FWorldContext* WorldContext = GetWorldContext(World);
|
|
|
|
switch (WorldContext->WorldType)
|
|
|
|
{
|
|
|
|
case EWorldType::PIE:
|
|
|
|
return FString::Printf(TEXT("PIEContext%d"), GetWorldContextIndex(*WorldContext));
|
|
|
|
case EWorldType::Game:
|
|
|
|
return TEXT("Game");
|
|
|
|
case EWorldType::Editor:
|
|
|
|
return TEXT("Editor");
|
|
|
|
default:
|
|
|
|
return TEXT("Other");
|
|
|
|
}
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
2017-09-27 20:16:54 +00:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
FORCEINLINE FString GetWorldContextName()
|
2017-08-28 19:29:07 +00:00
|
|
|
{
|
2017-09-27 20:16:54 +00:00
|
|
|
return TEXT("Game");
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
FORCEINLINE FString GetWorldContextName(const UWorld&)
|
|
|
|
{
|
|
|
|
return TEXT("Game");
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // WITH_EDITOR
|
|
|
|
}
|
|
|
|
|
2017-10-28 21:25:44 +00:00
|
|
|
FImGuiContextManager::FImGuiContextManager()
|
|
|
|
{
|
2018-05-24 20:55:25 +00:00
|
|
|
unsigned char* Pixels;
|
|
|
|
int Width, Height, Bpp;
|
|
|
|
FontAtlas.GetTexDataAsRGBA32(&Pixels, &Width, &Height, &Bpp);
|
|
|
|
|
2017-10-28 21:25:44 +00:00
|
|
|
FWorldDelegates::OnWorldTickStart.AddRaw(this, &FImGuiContextManager::OnWorldTickStart);
|
2019-04-14 11:16:15 +00:00
|
|
|
#if ENGINE_COMPATIBILITY_WITH_WORLD_POST_ACTOR_TICK
|
2018-10-28 19:54:18 +00:00
|
|
|
FWorldDelegates::OnWorldPostActorTick.AddRaw(this, &FImGuiContextManager::OnWorldPostActorTick);
|
|
|
|
#endif
|
2017-10-28 21:25:44 +00:00
|
|
|
}
|
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
FImGuiContextManager::~FImGuiContextManager()
|
|
|
|
{
|
2017-10-28 21:25:44 +00:00
|
|
|
// Order matters because contexts can be created during World Tick Start events.
|
|
|
|
FWorldDelegates::OnWorldTickStart.RemoveAll(this);
|
2019-04-14 11:16:15 +00:00
|
|
|
#if ENGINE_COMPATIBILITY_WITH_WORLD_POST_ACTOR_TICK
|
2018-10-28 19:54:18 +00:00
|
|
|
FWorldDelegates::OnWorldPostActorTick.RemoveAll(this);
|
|
|
|
#endif
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FImGuiContextManager::Tick(float DeltaSeconds)
|
|
|
|
{
|
2017-09-27 20:16:54 +00:00
|
|
|
// In editor, worlds can get invalid. We could remove corresponding entries, but that would mean resetting ImGui
|
|
|
|
// context every time when PIE session is restarted. Instead we freeze contexts until their worlds are re-created.
|
|
|
|
|
|
|
|
for (auto& Pair : Contexts)
|
2017-08-28 19:29:07 +00:00
|
|
|
{
|
2017-09-27 20:16:54 +00:00
|
|
|
auto& ContextData = Pair.Value;
|
|
|
|
if (ContextData.CanTick())
|
|
|
|
{
|
2019-08-03 16:45:53 +00:00
|
|
|
ContextData.ContextProxy->Tick(DeltaSeconds);
|
2017-09-27 20:16:54 +00:00
|
|
|
}
|
2019-04-10 19:19:11 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// Clear to make sure that we don't store objects registered for world that is no longer valid.
|
|
|
|
FImGuiDelegatesContainer::Get().OnWorldDebug(Pair.Key).Clear();
|
|
|
|
}
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
2017-09-27 20:16:54 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 21:51:42 +00:00
|
|
|
#if ENGINE_COMPATIBILITY_LEGACY_WORLD_ACTOR_TICK
|
2017-10-28 21:25:44 +00:00
|
|
|
void FImGuiContextManager::OnWorldTickStart(ELevelTick TickType, float DeltaSeconds)
|
|
|
|
{
|
2020-01-27 21:51:42 +00:00
|
|
|
OnWorldTickStart(GWorld, TickType, DeltaSeconds);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void FImGuiContextManager::OnWorldTickStart(UWorld* World, ELevelTick TickType, float DeltaSeconds)
|
|
|
|
{
|
|
|
|
if (World)
|
2017-10-28 21:25:44 +00:00
|
|
|
{
|
2020-01-27 21:51:42 +00:00
|
|
|
FImGuiContextProxy& ContextProxy = GetWorldContextProxy(*World);
|
2019-04-14 11:16:15 +00:00
|
|
|
|
|
|
|
// Set as current, so we have right context ready when updating world objects.
|
2018-03-20 23:24:03 +00:00
|
|
|
ContextProxy.SetAsCurrent();
|
2018-10-28 19:54:18 +00:00
|
|
|
|
2019-04-14 11:16:15 +00:00
|
|
|
ContextProxy.DrawEarlyDebug();
|
|
|
|
#if !ENGINE_COMPATIBILITY_WITH_WORLD_POST_ACTOR_TICK
|
|
|
|
ContextProxy.DrawDebug();
|
2018-10-28 19:54:18 +00:00
|
|
|
#endif
|
2017-10-28 21:25:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-14 11:16:15 +00:00
|
|
|
#if ENGINE_COMPATIBILITY_WITH_WORLD_POST_ACTOR_TICK
|
2018-10-28 19:54:18 +00:00
|
|
|
void FImGuiContextManager::OnWorldPostActorTick(UWorld* World, ELevelTick TickType, float DeltaSeconds)
|
|
|
|
{
|
2020-01-27 21:51:42 +00:00
|
|
|
GetWorldContextProxy(*World).DrawDebug();
|
2018-10-28 19:54:18 +00:00
|
|
|
}
|
2019-04-14 11:16:15 +00:00
|
|
|
#endif // ENGINE_COMPATIBILITY_WITH_WORLD_POST_ACTOR_TICK
|
2018-10-28 19:54:18 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
#if WITH_EDITOR
|
|
|
|
FImGuiContextManager::FContextData& FImGuiContextManager::GetEditorContextData()
|
|
|
|
{
|
|
|
|
FContextData* Data = Contexts.Find(Utilities::EDITOR_CONTEXT_INDEX);
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
if (UNLIKELY(!Data))
|
2017-08-28 19:29:07 +00:00
|
|
|
{
|
2018-11-25 20:36:55 +00:00
|
|
|
Data = &Contexts.Emplace(Utilities::EDITOR_CONTEXT_INDEX, FContextData{ GetEditorContextName(), Utilities::EDITOR_CONTEXT_INDEX, DrawMultiContextEvent, FontAtlas, -1 });
|
2019-08-03 16:45:53 +00:00
|
|
|
ContextProxyCreatedEvent.Broadcast(Utilities::EDITOR_CONTEXT_INDEX, *Data->ContextProxy);
|
2017-09-27 20:16:54 +00:00
|
|
|
}
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
return *Data;
|
|
|
|
}
|
|
|
|
#endif // WITH_EDITOR
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
#if !WITH_EDITOR
|
|
|
|
FImGuiContextManager::FContextData& FImGuiContextManager::GetStandaloneWorldContextData()
|
|
|
|
{
|
|
|
|
FContextData* Data = Contexts.Find(Utilities::STANDALONE_GAME_CONTEXT_INDEX);
|
|
|
|
|
|
|
|
if (UNLIKELY(!Data))
|
|
|
|
{
|
2018-11-25 20:36:55 +00:00
|
|
|
Data = &Contexts.Emplace(Utilities::STANDALONE_GAME_CONTEXT_INDEX, FContextData{ GetWorldContextName(), Utilities::STANDALONE_GAME_CONTEXT_INDEX, DrawMultiContextEvent, FontAtlas });
|
2019-08-03 16:45:53 +00:00
|
|
|
ContextProxyCreatedEvent.Broadcast(Utilities::STANDALONE_GAME_CONTEXT_INDEX, *Data->ContextProxy);
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
2017-09-27 20:16:54 +00:00
|
|
|
|
|
|
|
return *Data;
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
2017-09-27 20:16:54 +00:00
|
|
|
#endif // !WITH_EDITOR
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-10-03 20:16:17 +00:00
|
|
|
FImGuiContextManager::FContextData& FImGuiContextManager::GetWorldContextData(const UWorld& World, int32* OutIndex)
|
2017-08-28 19:29:07 +00:00
|
|
|
{
|
2017-09-27 20:16:54 +00:00
|
|
|
using namespace Utilities;
|
|
|
|
|
2017-10-28 21:25:44 +00:00
|
|
|
#if WITH_EDITOR
|
|
|
|
if (World.WorldType == EWorldType::Editor)
|
|
|
|
{
|
|
|
|
if (OutIndex)
|
|
|
|
{
|
|
|
|
*OutIndex = Utilities::EDITOR_CONTEXT_INDEX;
|
|
|
|
}
|
|
|
|
|
|
|
|
return GetEditorContextData();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
const FWorldContext* WorldContext = GetWorldContext(World);
|
|
|
|
const int32 Index = GetWorldContextIndex(*WorldContext);
|
|
|
|
|
|
|
|
checkf(Index != Utilities::INVALID_CONTEXT_INDEX, TEXT("Couldn't find context index for world %s: WorldType = %d"),
|
2017-11-05 22:36:39 +00:00
|
|
|
*World.GetName(), static_cast<int32>(World.WorldType));
|
2017-09-27 20:16:54 +00:00
|
|
|
|
|
|
|
#if WITH_EDITOR
|
|
|
|
checkf(!GEngine->IsEditor() || Index != Utilities::EDITOR_CONTEXT_INDEX,
|
|
|
|
TEXT("Context index %d is reserved for editor and cannot be used for world %s: WorldType = %d, NetMode = %d"),
|
2017-11-05 22:36:39 +00:00
|
|
|
Index, *World.GetName(), static_cast<int32>(World.WorldType), static_cast<int32>(World.GetNetMode()));
|
2017-09-27 20:16:54 +00:00
|
|
|
#endif
|
|
|
|
|
2017-08-28 19:29:07 +00:00
|
|
|
FContextData* Data = Contexts.Find(Index);
|
|
|
|
|
2017-09-27 20:16:54 +00:00
|
|
|
#if WITH_EDITOR
|
|
|
|
if (UNLIKELY(!Data))
|
2017-08-28 19:29:07 +00:00
|
|
|
{
|
2018-11-25 20:36:55 +00:00
|
|
|
Data = &Contexts.Emplace(Index, FContextData{ GetWorldContextName(World), Index, DrawMultiContextEvent, FontAtlas, WorldContext->PIEInstance });
|
2019-08-03 16:45:53 +00:00
|
|
|
ContextProxyCreatedEvent.Broadcast(Index, *Data->ContextProxy);
|
2017-09-27 20:16:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Because we allow (for the sake of continuity) to map different PIE instances to the same index.
|
|
|
|
Data->PIEInstance = WorldContext->PIEInstance;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (UNLIKELY(!Data))
|
|
|
|
{
|
2018-11-25 20:36:55 +00:00
|
|
|
Data = &Contexts.Emplace(Index, FContextData{ GetWorldContextName(World), Index, DrawMultiContextEvent, FontAtlas });
|
2019-08-03 16:45:53 +00:00
|
|
|
ContextProxyCreatedEvent.Broadcast(Index, *Data->ContextProxy);
|
2017-08-28 19:29:07 +00:00
|
|
|
}
|
2017-09-27 20:16:54 +00:00
|
|
|
#endif
|
2017-08-28 19:29:07 +00:00
|
|
|
|
2017-10-03 20:16:17 +00:00
|
|
|
if (OutIndex)
|
|
|
|
{
|
|
|
|
*OutIndex = Index;
|
|
|
|
}
|
2017-08-28 19:29:07 +00:00
|
|
|
return *Data;
|
|
|
|
}
|