You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
o3de/Gems/Atom/Feature/Common/Code/Source/ImGui/ImGuiSystemComponent.cpp

263 lines
9.2 KiB
C++

/*
* Copyright (c) Contributors to the Open 3D Engine Project.
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
*
* SPDX-License-Identifier: Apache-2.0 OR MIT
*
*/
#include <ImGui/ImGuiSystemComponent.h>
#include <ImGui/ImGuiPass.h>
#include <AzCore/Serialization/SerializeContext.h>
#include <Atom/RPI.Public/Pass/PassSystemInterface.h>
#include <Atom/RPI.Public/Pass/PassFilter.h>
#include <Atom/RPI.Public/ViewportContext.h>
#include <Atom/RPI.Public/ViewportContextBus.h>
namespace AZ
{
namespace Render
{
void ImGuiSystemComponent::Reflect(AZ::ReflectContext* context)
{
if (auto* serializeContext = azrtti_cast<AZ::SerializeContext*>(context))
{
serializeContext->Class<ImGuiSystemComponent, AZ::Component>()
->Version(0)
;
}
}
void ImGuiSystemComponent::GetProvidedServices(AZ::ComponentDescriptor::DependencyArrayType& provided)
{
provided.push_back(AZ_CRC("ImGuiSystemComponent", 0x2f08b9a7));
}
void ImGuiSystemComponent::GetRequiredServices(AZ::ComponentDescriptor::DependencyArrayType& required)
{
required.push_back(AZ_CRC("RPISystem", 0xf2add773));
}
void ImGuiSystemComponent::GetIncompatibleServices(AZ::ComponentDescriptor::DependencyArrayType& incompatbile)
{
incompatbile.push_back(AZ_CRC("ImGuiSystemComponent", 0x2f08b9a7));
}
ImGuiSystemComponent::ImGuiSystemComponent()
{
}
void ImGuiSystemComponent::Activate()
{
ImGuiSystemRequestBus::Handler::BusConnect();
}
void ImGuiSystemComponent::Deactivate()
{
ImGuiSystemRequestBus::Handler::BusDisconnect();
}
void ImGuiSystemComponent::SetGlobalSizeScale(float scale)
{
if (m_sizeScale != scale)
{
m_sizeScale = scale;
ForAllImGuiPasses(
[&]([[maybe_unused]] ImGuiPass* pass)
{
ImGui::GetStyle().ScaleAllSizes(m_sizeScale);
}
);
}
}
void ImGuiSystemComponent::SetGlobalFontScale(float scale)
{
if (m_fontScale != scale)
{
m_fontScale = scale;
ForAllImGuiPasses(
[&]([[maybe_unused]] ImGuiPass* pass)
{
ImGui::GetIO().FontGlobalScale = scale;
}
);
}
}
void ImGuiSystemComponent::HideAllImGuiPasses()
{
ForAllImGuiPasses(
[&](ImGuiPass* pass)
{
pass->SetEnabled(false);
}
);
}
void ImGuiSystemComponent::ShowAllImGuiPasses()
{
ForAllImGuiPasses(
[&](ImGuiPass* pass)
{
pass->SetEnabled(true);
}
);
}
void ImGuiSystemComponent::ForAllImGuiPasses(PassFunction func)
{
ImGuiContext* contextToRestore = ImGui::GetCurrentContext();
RPI::PassFilter passFilter = RPI::PassFilter::CreateWithPassClass<ImGuiPass>();
RPI::PassSystemInterface::Get()->ForEachPass(passFilter, [func](RPI::Pass* pass) -> RPI::PassFilterExecutionFlow
{
ImGuiPass* imguiPass = azrtti_cast<ImGuiPass*>(pass);
ImGui::SetCurrentContext(imguiPass->GetContext());
func(imguiPass);
return RPI::PassFilterExecutionFlow::ContinueVisitingPasses;
});
ImGui::SetCurrentContext(contextToRestore);
}
void ImGuiSystemComponent::PushDefaultImGuiPass(ImGuiPass* imguiPass)
{
ImGuiPass** existingPass = AZStd::find(m_defaultImguiPassStack.begin(), m_defaultImguiPassStack.end(), imguiPass);
if (existingPass != m_defaultImguiPassStack.end())
{
AZ_Assert(false, "This Imgui pass is already registered as a default pass.");
return;
}
m_defaultImguiPassStack.push_back(imguiPass);
}
void ImGuiSystemComponent::RemoveDefaultImGuiPass(ImGuiPass* imguiPass)
{
for (size_t i = 0; i < m_defaultImguiPassStack.size(); ++i)
{
if (imguiPass == m_defaultImguiPassStack.at(i))
{
m_defaultImguiPassStack.erase(m_defaultImguiPassStack.begin() + i);
// If the pass being removed as default is active, assume the current active should be changed
// to whatever's on the top of the default stack as long as it has at least one entry.
if (GetActiveContext() == imguiPass->GetContext())
{
PushActiveContextFromDefaultPass();
}
break;
}
}
}
ImGuiPass* ImGuiSystemComponent::GetDefaultImGuiPass()
{
return m_defaultImguiPassStack.size() > 0 ? m_defaultImguiPassStack.back() : nullptr;
}
bool ImGuiSystemComponent::PushActiveContextFromDefaultPass()
{
if (m_defaultImguiPassStack.size() > 0)
{
ImGuiContext* context = m_defaultImguiPassStack.back()->GetContext();
m_activeContextStack.push_back(context);
ImGuiSystemNotificationBus::Broadcast(&ImGuiSystemNotifications::ActiveImGuiContextChanged, context);
return true;
}
return false;
}
bool ImGuiSystemComponent::PushActiveContextFromPass(const AZStd::vector<AZStd::string>& passHierarchyFilter)
{
if (passHierarchyFilter.size() == 0)
{
AZ_Warning("ImGuiSystemComponent", false, "passHierarchyFilter is empty");
return false;
}
AZStd::vector<ImGuiPass*> foundImGuiPasses;
RPI::PassFilter passFilter = RPI::PassFilter::CreateWithPassHierarchy(passHierarchyFilter);
RPI::PassSystemInterface::Get()->ForEachPass(passFilter, [&foundImGuiPasses](RPI::Pass* pass) -> RPI::PassFilterExecutionFlow
{
ImGuiPass* imGuiPass = azrtti_cast<ImGuiPass*>(pass);
if (imGuiPass)
{
foundImGuiPasses.push_back(imGuiPass);
}
return RPI::PassFilterExecutionFlow::ContinueVisitingPasses;
});
if (foundImGuiPasses.size() == 0)
{
AZ_Warning("ImGuiSystemComponent", false, "Failed to find ImGui pass to activate from %s", passHierarchyFilter[0].c_str());
return false;
}
if (foundImGuiPasses.size() > 1)
{
AZ_Warning("ImGuiSystemComponent", false, "Found more than one ImGui pass to activate from %s, only activating first one.", passHierarchyFilter[0].c_str());
}
ImGuiContext* context = foundImGuiPasses.at(0)->GetContext();
m_activeContextStack.push_back(context);
ImGuiSystemNotificationBus::Broadcast(&ImGuiSystemNotifications::ActiveImGuiContextChanged, context);
return true;
}
bool ImGuiSystemComponent::PopActiveContext()
{
if (m_activeContextStack.size() > 0)
{
m_activeContextStack.pop_back();
ImGuiContext* newContext = GetActiveContext();
ImGuiSystemNotificationBus::Broadcast(&ImGuiSystemNotifications::ActiveImGuiContextChanged, newContext);
return true;
}
AZ_Error("ImGuiSystemComponent", false, "Attempting to pop active ImGui context when there are none on the stack. There must be a Push/Pop mismatch.")
return false;
}
ImGuiContext* ImGuiSystemComponent::GetActiveContext()
{
if (m_activeContextStack.size() > 0)
{
return m_activeContextStack.back();
}
return nullptr;
}
bool ImGuiSystemComponent::RenderImGuiBuffersToCurrentViewport(const ImDrawData& drawData)
{
auto atomViewportRequests = AZ::Interface<AZ::RPI::ViewportContextRequestsInterface>::Get();
RPI::ViewportContextPtr viewportContext = atomViewportRequests->GetDefaultViewportContext();
if (viewportContext == nullptr)
{
return false;
}
auto renderPipeline = viewportContext->GetCurrentPipeline();
if (renderPipeline == nullptr)
{
return false;
}
for (auto imGuiPass : m_defaultImguiPassStack)
{
if (imGuiPass->GetRenderPipeline() == renderPipeline.get())
{
imGuiPass->RenderImguiDrawData(drawData);
return true;
}
}
return false;
}
} // namespace RPI
} // namespace AZ