2026-03-27 16:22:59 +08:00
|
|
|
#include "Rendering/CameraRenderer.h"
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
#include "Rendering/DirectionalShadowSurfaceCache.h"
|
2026-04-04 14:41:01 +08:00
|
|
|
#include "Rendering/Passes/BuiltinDepthOnlyPass.h"
|
2026-04-01 16:44:11 +08:00
|
|
|
#include "Rendering/Passes/BuiltinObjectIdPass.h"
|
2026-04-04 14:41:01 +08:00
|
|
|
#include "Rendering/Passes/BuiltinShadowCasterPass.h"
|
2026-03-27 16:22:59 +08:00
|
|
|
#include "Rendering/Pipelines/BuiltinForwardPipeline.h"
|
2026-04-01 00:56:48 +08:00
|
|
|
#include "Rendering/RenderPipelineAsset.h"
|
2026-03-27 16:22:59 +08:00
|
|
|
#include "Rendering/RenderSurface.h"
|
2026-04-04 20:35:47 +08:00
|
|
|
#include "RHI/RHIResourceView.h"
|
2026-03-27 16:22:59 +08:00
|
|
|
#include "Scene/Scene.h"
|
|
|
|
|
|
|
|
|
|
namespace XCEngine {
|
|
|
|
|
namespace Rendering {
|
|
|
|
|
|
2026-04-05 00:03:22 +08:00
|
|
|
namespace {
|
|
|
|
|
|
2026-04-01 00:56:48 +08:00
|
|
|
std::shared_ptr<const RenderPipelineAsset> CreateDefaultPipelineAsset() {
|
|
|
|
|
static const std::shared_ptr<const RenderPipelineAsset> s_defaultPipelineAsset =
|
|
|
|
|
std::make_shared<Pipelines::BuiltinForwardPipelineAsset>();
|
|
|
|
|
return s_defaultPipelineAsset;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-04 14:41:01 +08:00
|
|
|
std::unique_ptr<RenderPass> CreateDefaultDepthOnlyPass() {
|
|
|
|
|
return std::make_unique<Passes::BuiltinDepthOnlyPass>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<RenderPass> CreateDefaultShadowCasterPass() {
|
|
|
|
|
return std::make_unique<Passes::BuiltinShadowCasterPass>();
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-01 00:56:48 +08:00
|
|
|
std::unique_ptr<RenderPipeline> CreatePipelineFromAsset(
|
|
|
|
|
const std::shared_ptr<const RenderPipelineAsset>& pipelineAsset) {
|
|
|
|
|
if (pipelineAsset != nullptr) {
|
|
|
|
|
std::unique_ptr<RenderPipeline> pipeline = pipelineAsset->CreatePipeline();
|
|
|
|
|
if (pipeline != nullptr) {
|
|
|
|
|
return pipeline;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return std::make_unique<Pipelines::BuiltinForwardPipeline>();
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-04 14:41:01 +08:00
|
|
|
bool InitializeStandalonePass(
|
|
|
|
|
RenderPass* pass,
|
|
|
|
|
const RenderContext& context) {
|
|
|
|
|
if (pass == nullptr) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pass->Initialize(context)) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pass->Shutdown();
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ExecuteScenePassRequest(
|
|
|
|
|
RenderPass* pass,
|
|
|
|
|
const ScenePassRenderRequest& request,
|
|
|
|
|
const RenderContext& context,
|
|
|
|
|
const RenderSceneData& baseSceneData) {
|
|
|
|
|
if (!request.IsRequested()) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!InitializeStandalonePass(pass, context)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RenderSceneData sceneData = baseSceneData;
|
|
|
|
|
if (request.hasCameraDataOverride) {
|
|
|
|
|
sceneData.cameraData = request.cameraDataOverride;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sceneData.cameraData.viewportWidth = request.surface.GetRenderAreaWidth();
|
|
|
|
|
sceneData.cameraData.viewportHeight = request.surface.GetRenderAreaHeight();
|
|
|
|
|
sceneData.cameraData.clearFlags = request.clearFlags;
|
|
|
|
|
if (request.hasClearColorOverride) {
|
|
|
|
|
sceneData.cameraData.clearColor = request.clearColorOverride;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const RenderPassContext passContext = {
|
|
|
|
|
context,
|
|
|
|
|
request.surface,
|
|
|
|
|
sceneData
|
|
|
|
|
};
|
|
|
|
|
return pass->Execute(passContext);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 18:20:19 +08:00
|
|
|
bool ExecuteStandalonePass(
|
|
|
|
|
RenderPass* pass,
|
|
|
|
|
const RenderContext& context,
|
|
|
|
|
const RenderSurface& surface,
|
|
|
|
|
const RenderSceneData& sceneData) {
|
|
|
|
|
if (pass == nullptr) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!InitializeStandalonePass(pass, context)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const RenderPassContext passContext = {
|
|
|
|
|
context,
|
|
|
|
|
surface,
|
|
|
|
|
sceneData
|
|
|
|
|
};
|
|
|
|
|
return pass->Execute(passContext);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
class ScopedInitializedPassSequence {
|
|
|
|
|
public:
|
|
|
|
|
ScopedInitializedPassSequence(
|
|
|
|
|
RenderPassSequence* sequence,
|
|
|
|
|
const RenderContext& context)
|
|
|
|
|
: m_sequence(sequence) {
|
|
|
|
|
if (m_sequence == nullptr) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_initialized = m_sequence->Initialize(context);
|
|
|
|
|
if (!m_initialized) {
|
|
|
|
|
m_failed = true;
|
|
|
|
|
m_sequence->Shutdown();
|
|
|
|
|
m_sequence = nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ScopedInitializedPassSequence(const ScopedInitializedPassSequence&) = delete;
|
|
|
|
|
ScopedInitializedPassSequence& operator=(const ScopedInitializedPassSequence&) = delete;
|
|
|
|
|
|
|
|
|
|
~ScopedInitializedPassSequence() {
|
|
|
|
|
if (m_sequence != nullptr && m_initialized) {
|
|
|
|
|
m_sequence->Shutdown();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IsReady() const {
|
|
|
|
|
return !m_failed;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Execute(const RenderPassContext& context) const {
|
|
|
|
|
return m_sequence == nullptr || m_sequence->Execute(context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
RenderPassSequence* m_sequence = nullptr;
|
|
|
|
|
bool m_initialized = false;
|
|
|
|
|
bool m_failed = false;
|
|
|
|
|
};
|
|
|
|
|
|
2026-04-04 23:01:34 +08:00
|
|
|
RenderDirectionalShadowData BuildDirectionalShadowData(
|
|
|
|
|
const DirectionalShadowRenderPlan& plan,
|
|
|
|
|
RHI::RHIResourceView* shadowMapView) {
|
|
|
|
|
RenderDirectionalShadowData shadowData = {};
|
|
|
|
|
if (!plan.IsValid() || shadowMapView == nullptr) {
|
|
|
|
|
return shadowData;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
shadowData.enabled = true;
|
|
|
|
|
shadowData.viewProjection = plan.cameraData.viewProjection;
|
|
|
|
|
shadowData.shadowMap = shadowMapView;
|
|
|
|
|
shadowData.shadowParams = Math::Vector4(
|
|
|
|
|
0.0015f,
|
|
|
|
|
1.0f / static_cast<float>(plan.mapWidth),
|
|
|
|
|
1.0f / static_cast<float>(plan.mapHeight),
|
|
|
|
|
0.85f);
|
|
|
|
|
return shadowData;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-31 21:54:00 +08:00
|
|
|
} // namespace
|
|
|
|
|
|
2026-03-27 16:22:59 +08:00
|
|
|
CameraRenderer::CameraRenderer()
|
2026-04-01 00:56:48 +08:00
|
|
|
: CameraRenderer(CreateDefaultPipelineAsset()) {
|
2026-03-27 16:22:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CameraRenderer::CameraRenderer(std::unique_ptr<RenderPipeline> pipeline)
|
2026-04-04 14:41:01 +08:00
|
|
|
: CameraRenderer(
|
|
|
|
|
std::move(pipeline),
|
|
|
|
|
std::make_unique<Passes::BuiltinObjectIdPass>(),
|
|
|
|
|
CreateDefaultDepthOnlyPass(),
|
|
|
|
|
CreateDefaultShadowCasterPass()) {
|
2026-04-01 16:44:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CameraRenderer::CameraRenderer(
|
|
|
|
|
std::unique_ptr<RenderPipeline> pipeline,
|
2026-04-05 18:20:19 +08:00
|
|
|
std::unique_ptr<RenderPass> objectIdPass,
|
2026-04-04 14:41:01 +08:00
|
|
|
std::unique_ptr<RenderPass> depthOnlyPass,
|
|
|
|
|
std::unique_ptr<RenderPass> shadowCasterPass)
|
2026-04-01 16:44:11 +08:00
|
|
|
: m_pipelineAsset(nullptr)
|
2026-04-04 14:41:01 +08:00
|
|
|
, m_objectIdPass(std::move(objectIdPass))
|
|
|
|
|
, m_depthOnlyPass(std::move(depthOnlyPass))
|
|
|
|
|
, m_shadowCasterPass(std::move(shadowCasterPass)) {
|
2026-04-01 16:44:11 +08:00
|
|
|
if (m_objectIdPass == nullptr) {
|
|
|
|
|
m_objectIdPass = std::make_unique<Passes::BuiltinObjectIdPass>();
|
|
|
|
|
}
|
2026-04-04 14:41:01 +08:00
|
|
|
if (m_depthOnlyPass == nullptr) {
|
|
|
|
|
m_depthOnlyPass = CreateDefaultDepthOnlyPass();
|
|
|
|
|
}
|
|
|
|
|
if (m_shadowCasterPass == nullptr) {
|
|
|
|
|
m_shadowCasterPass = CreateDefaultShadowCasterPass();
|
|
|
|
|
}
|
2026-04-01 00:56:48 +08:00
|
|
|
ResetPipeline(std::move(pipeline));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CameraRenderer::CameraRenderer(std::shared_ptr<const RenderPipelineAsset> pipelineAsset)
|
2026-04-01 16:44:11 +08:00
|
|
|
: m_pipelineAsset(std::move(pipelineAsset))
|
2026-04-04 14:41:01 +08:00
|
|
|
, m_objectIdPass(std::make_unique<Passes::BuiltinObjectIdPass>())
|
|
|
|
|
, m_depthOnlyPass(CreateDefaultDepthOnlyPass())
|
|
|
|
|
, m_shadowCasterPass(CreateDefaultShadowCasterPass()) {
|
2026-04-01 00:56:48 +08:00
|
|
|
SetPipelineAsset(m_pipelineAsset);
|
2026-03-27 16:22:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CameraRenderer::~CameraRenderer() {
|
|
|
|
|
if (m_pipeline) {
|
|
|
|
|
m_pipeline->Shutdown();
|
|
|
|
|
}
|
2026-04-01 16:44:11 +08:00
|
|
|
if (m_objectIdPass != nullptr) {
|
|
|
|
|
m_objectIdPass->Shutdown();
|
|
|
|
|
}
|
2026-04-04 14:41:01 +08:00
|
|
|
if (m_depthOnlyPass != nullptr) {
|
|
|
|
|
m_depthOnlyPass->Shutdown();
|
|
|
|
|
}
|
|
|
|
|
if (m_shadowCasterPass != nullptr) {
|
|
|
|
|
m_shadowCasterPass->Shutdown();
|
|
|
|
|
}
|
2026-03-27 16:22:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CameraRenderer::SetPipeline(std::unique_ptr<RenderPipeline> pipeline) {
|
2026-04-01 00:56:48 +08:00
|
|
|
m_pipelineAsset.reset();
|
|
|
|
|
ResetPipeline(std::move(pipeline));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CameraRenderer::SetPipelineAsset(std::shared_ptr<const RenderPipelineAsset> pipelineAsset) {
|
|
|
|
|
m_pipelineAsset = pipelineAsset != nullptr ? std::move(pipelineAsset) : CreateDefaultPipelineAsset();
|
|
|
|
|
ResetPipeline(CreatePipelineFromAsset(m_pipelineAsset));
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 18:20:19 +08:00
|
|
|
void CameraRenderer::SetObjectIdPass(std::unique_ptr<RenderPass> objectIdPass) {
|
2026-04-01 16:44:11 +08:00
|
|
|
if (m_objectIdPass != nullptr) {
|
|
|
|
|
m_objectIdPass->Shutdown();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_objectIdPass = std::move(objectIdPass);
|
|
|
|
|
if (m_objectIdPass == nullptr) {
|
|
|
|
|
m_objectIdPass = std::make_unique<Passes::BuiltinObjectIdPass>();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-04 14:41:01 +08:00
|
|
|
void CameraRenderer::SetDepthOnlyPass(std::unique_ptr<RenderPass> depthOnlyPass) {
|
|
|
|
|
if (m_depthOnlyPass != nullptr) {
|
|
|
|
|
m_depthOnlyPass->Shutdown();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_depthOnlyPass = std::move(depthOnlyPass);
|
|
|
|
|
if (m_depthOnlyPass == nullptr) {
|
|
|
|
|
m_depthOnlyPass = CreateDefaultDepthOnlyPass();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CameraRenderer::SetShadowCasterPass(std::unique_ptr<RenderPass> shadowCasterPass) {
|
|
|
|
|
if (m_shadowCasterPass != nullptr) {
|
|
|
|
|
m_shadowCasterPass->Shutdown();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_shadowCasterPass = std::move(shadowCasterPass);
|
|
|
|
|
if (m_shadowCasterPass == nullptr) {
|
|
|
|
|
m_shadowCasterPass = CreateDefaultShadowCasterPass();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-01 00:56:48 +08:00
|
|
|
void CameraRenderer::ResetPipeline(std::unique_ptr<RenderPipeline> pipeline) {
|
|
|
|
|
if (m_pipeline != nullptr) {
|
2026-03-27 16:22:59 +08:00
|
|
|
m_pipeline->Shutdown();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_pipeline = std::move(pipeline);
|
2026-04-01 00:56:48 +08:00
|
|
|
if (m_pipeline == nullptr) {
|
|
|
|
|
m_pipelineAsset = CreateDefaultPipelineAsset();
|
|
|
|
|
m_pipeline = CreatePipelineFromAsset(m_pipelineAsset);
|
2026-03-27 16:22:59 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
bool CameraRenderer::ResolveShadowCasterRequest(
|
|
|
|
|
const CameraRenderRequest& request,
|
|
|
|
|
ShadowCasterRenderRequest& outResolvedShadowCaster,
|
|
|
|
|
RHI::RHIResourceView*& outShadowMapView) {
|
|
|
|
|
outResolvedShadowCaster = request.shadowCaster;
|
|
|
|
|
outShadowMapView = nullptr;
|
2026-04-05 00:03:22 +08:00
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
if (outResolvedShadowCaster.IsRequested()) {
|
|
|
|
|
return outResolvedShadowCaster.IsValid();
|
2026-04-05 00:03:22 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
if (!request.directionalShadow.IsValid()) {
|
|
|
|
|
return true;
|
2026-03-27 16:22:59 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
if (m_directionalShadowSurface == nullptr) {
|
|
|
|
|
m_directionalShadowSurface = std::make_unique<DirectionalShadowSurfaceCache>();
|
2026-04-04 14:41:01 +08:00
|
|
|
}
|
2026-04-05 19:55:37 +08:00
|
|
|
|
|
|
|
|
if (!m_directionalShadowSurface->EnsureSurface(request.context, request.directionalShadow)) {
|
2026-04-01 16:44:11 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
2026-04-04 20:35:47 +08:00
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
outResolvedShadowCaster.surface = m_directionalShadowSurface->GetSurface();
|
|
|
|
|
outResolvedShadowCaster.clearFlags = RenderClearFlags::Depth;
|
|
|
|
|
if (!outResolvedShadowCaster.hasCameraDataOverride) {
|
|
|
|
|
outResolvedShadowCaster.hasCameraDataOverride = true;
|
|
|
|
|
outResolvedShadowCaster.cameraDataOverride = request.directionalShadow.cameraData;
|
2026-04-04 20:35:47 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
outShadowMapView = m_directionalShadowSurface->GetDepthShaderView();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool CameraRenderer::BuildSceneDataForRequest(
|
|
|
|
|
const CameraRenderRequest& request,
|
|
|
|
|
RHI::RHIResourceView* shadowMapView,
|
|
|
|
|
RenderSceneData& outSceneData) {
|
|
|
|
|
outSceneData = m_sceneExtractor.ExtractForCamera(
|
2026-03-27 16:57:04 +08:00
|
|
|
*request.scene,
|
|
|
|
|
*request.camera,
|
2026-04-01 13:01:11 +08:00
|
|
|
request.surface.GetRenderAreaWidth(),
|
|
|
|
|
request.surface.GetRenderAreaHeight());
|
2026-04-05 19:55:37 +08:00
|
|
|
if (!outSceneData.HasCamera()) {
|
2026-03-27 16:22:59 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-04 23:01:34 +08:00
|
|
|
if (request.directionalShadow.IsValid()) {
|
2026-04-05 19:55:37 +08:00
|
|
|
outSceneData.lighting.mainDirectionalShadow =
|
2026-04-04 23:01:34 +08:00
|
|
|
BuildDirectionalShadowData(request.directionalShadow, shadowMapView);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
outSceneData.cameraData.clearFlags = request.clearFlags;
|
2026-04-01 22:49:26 +08:00
|
|
|
if (request.hasClearColorOverride) {
|
2026-04-05 19:55:37 +08:00
|
|
|
outSceneData.cameraData.clearColor = request.clearColorOverride;
|
2026-04-01 22:49:26 +08:00
|
|
|
}
|
2026-04-05 19:55:37 +08:00
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool CameraRenderer::ExecuteRenderPlan(
|
|
|
|
|
const CameraRenderRequest& request,
|
|
|
|
|
const ShadowCasterRenderRequest& resolvedShadowCaster,
|
|
|
|
|
const RenderSceneData& sceneData) {
|
2026-03-30 02:22:17 +08:00
|
|
|
const RenderPassContext passContext = {
|
|
|
|
|
request.context,
|
|
|
|
|
request.surface,
|
|
|
|
|
sceneData
|
|
|
|
|
};
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
ScopedInitializedPassSequence preScenePasses(request.preScenePasses, request.context);
|
|
|
|
|
if (!preScenePasses.IsReady() || !preScenePasses.Execute(passContext)) {
|
2026-03-31 21:54:00 +08:00
|
|
|
return false;
|
2026-03-30 02:22:17 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-04 14:41:01 +08:00
|
|
|
if (!ExecuteScenePassRequest(
|
|
|
|
|
m_shadowCasterPass.get(),
|
2026-04-04 20:35:47 +08:00
|
|
|
resolvedShadowCaster,
|
2026-04-04 14:41:01 +08:00
|
|
|
request.context,
|
|
|
|
|
sceneData)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!ExecuteScenePassRequest(
|
|
|
|
|
m_depthOnlyPass.get(),
|
|
|
|
|
request.depthOnly,
|
|
|
|
|
request.context,
|
|
|
|
|
sceneData)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-30 02:22:17 +08:00
|
|
|
if (!m_pipeline->Render(request.context, request.surface, sceneData)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-01 16:44:11 +08:00
|
|
|
if (request.objectId.IsRequested() &&
|
2026-04-05 18:20:19 +08:00
|
|
|
!ExecuteStandalonePass(
|
|
|
|
|
m_objectIdPass.get(),
|
|
|
|
|
request.context,
|
|
|
|
|
request.objectId.surface,
|
|
|
|
|
sceneData)) {
|
2026-04-01 16:44:11 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
ScopedInitializedPassSequence postScenePasses(request.postScenePasses, request.context);
|
|
|
|
|
if (!postScenePasses.IsReady() || !postScenePasses.Execute(passContext)) {
|
2026-03-31 21:54:00 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
2026-04-05 19:55:37 +08:00
|
|
|
|
|
|
|
|
ScopedInitializedPassSequence overlayPasses(request.overlayPasses, request.context);
|
|
|
|
|
if (!overlayPasses.IsReady() || !overlayPasses.Execute(passContext)) {
|
2026-03-31 21:54:00 +08:00
|
|
|
return false;
|
2026-03-30 02:22:17 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool CameraRenderer::Render(
|
|
|
|
|
const CameraRenderRequest& request) {
|
|
|
|
|
if (!request.IsValid() || m_pipeline == nullptr) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (request.surface.GetRenderAreaWidth() == 0 ||
|
|
|
|
|
request.surface.GetRenderAreaHeight() == 0) {
|
2026-04-03 13:22:30 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
2026-04-05 19:55:37 +08:00
|
|
|
if (request.depthOnly.IsRequested() &&
|
|
|
|
|
!request.depthOnly.IsValid()) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if (request.objectId.IsRequested() &&
|
|
|
|
|
!request.objectId.IsValid()) {
|
2026-04-03 13:22:30 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
ShadowCasterRenderRequest resolvedShadowCaster = {};
|
|
|
|
|
RHI::RHIResourceView* shadowMapView = nullptr;
|
|
|
|
|
if (!ResolveShadowCasterRequest(request, resolvedShadowCaster, shadowMapView)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2026-03-31 21:54:00 +08:00
|
|
|
|
2026-04-05 19:55:37 +08:00
|
|
|
RenderSceneData sceneData = {};
|
|
|
|
|
if (!BuildSceneDataForRequest(request, shadowMapView, sceneData)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ExecuteRenderPlan(request, resolvedShadowCaster, sceneData);
|
2026-03-27 16:22:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Rendering
|
|
|
|
|
} // namespace XCEngine
|