DiffuseGIExampleComponent.cpp 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715
  1. /*
  2. * 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.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0 OR MIT
  5. *
  6. */
  7. #include <DiffuseGIExampleComponent.h>
  8. #include <Atom/Component/DebugCamera/ArcBallControllerComponent.h>
  9. #include <Atom/Component/DebugCamera/NoClipControllerComponent.h>
  10. #include <Atom/Feature/CoreLights/PhotometricValue.h>
  11. #include <Atom/RHI/Device.h>
  12. #include <Atom/RHI/Factory.h>
  13. #include <Atom/RPI.Public/View.h>
  14. #include <Atom/RPI.Reflect/Model/ModelAsset.h>
  15. #include <Atom/RPI.Reflect/Material/MaterialAsset.h>
  16. #include <Atom/RPI.Reflect/Asset/AssetUtils.h>
  17. #include <AzCore/Asset/AssetManagerBus.h>
  18. #include <AzCore/Component/Entity.h>
  19. #include <AzCore/IO/IOUtils.h>
  20. #include <AzCore/Serialization/SerializeContext.h>
  21. #include <AzCore/std/smart_ptr/make_shared.h>
  22. #include <AzCore/std/sort.h>
  23. #include <AzFramework/Components/TransformComponent.h>
  24. #include <AzFramework/Input/Devices/Mouse/InputDeviceMouse.h>
  25. #include <SampleComponentManager.h>
  26. #include <SampleComponentConfig.h>
  27. #include <EntityUtilityFunctions.h>
  28. #include <Automation/ScriptableImGui.h>
  29. #include <Atom/RPI.Public/RPISystemInterface.h>
  30. #include <Atom/RPI.Public/Scene.h>
  31. namespace AtomSampleViewer
  32. {
  33. const char* DiffuseGIExampleComponent::CornellBoxColorNames[] =
  34. {
  35. "Red",
  36. "Green",
  37. "Blue",
  38. "Yellow",
  39. "White"
  40. };
  41. const AZ::Render::ShadowmapSize DiffuseGIExampleComponent::s_shadowmapSizes[] =
  42. {
  43. AZ::Render::ShadowmapSize::Size256,
  44. AZ::Render::ShadowmapSize::Size512,
  45. AZ::Render::ShadowmapSize::Size1024,
  46. AZ::Render::ShadowmapSize::Size2048
  47. };
  48. void DiffuseGIExampleComponent::Reflect(AZ::ReflectContext* context)
  49. {
  50. if (AZ::SerializeContext* serializeContext = azrtti_cast<AZ::SerializeContext*>(context))
  51. {
  52. serializeContext->Class<DiffuseGIExampleComponent, Component>()
  53. ->Version(0)
  54. ;
  55. }
  56. }
  57. void DiffuseGIExampleComponent::Activate()
  58. {
  59. m_imguiSidebar.Activate();
  60. AZ::TickBus::Handler::BusConnect();
  61. // setup the camera
  62. Camera::CameraRequestBus::EventResult(m_originalFarClipDistance, GetCameraEntityId(), &Camera::CameraRequestBus::Events::GetFarClipDistance);
  63. Camera::CameraRequestBus::Event(GetCameraEntityId(), &Camera::CameraRequestBus::Events::SetFarClipDistance, 180.f);
  64. // disable global Ibl in the example scene since we're controlling it separately
  65. DisableGlobalIbl();
  66. LoadSampleSceneAssets();
  67. }
  68. void DiffuseGIExampleComponent::UnloadSampleScene(bool geometryOnly)
  69. {
  70. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  71. // release meshes
  72. AZ::Render::MeshFeatureProcessorInterface* meshFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::MeshFeatureProcessorInterface>();
  73. for (auto& meshHandle : m_meshHandles)
  74. {
  75. meshFeatureProcessor->ReleaseMesh(meshHandle);
  76. }
  77. // unload everything else if necessary
  78. if (!geometryOnly)
  79. {
  80. if (m_diffuseProbeGrid)
  81. {
  82. AZ::Render::DiffuseProbeGridFeatureProcessorInterface* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DiffuseProbeGridFeatureProcessorInterface>();
  83. diffuseProbeGridFeatureProcessor->RemoveProbeGrid(m_diffuseProbeGrid);
  84. }
  85. if (m_directionalLightHandle.IsValid())
  86. {
  87. AZ::Render::DirectionalLightFeatureProcessorInterface* directionalLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DirectionalLightFeatureProcessorInterface>();
  88. directionalLightFeatureProcessor->ReleaseLight(m_directionalLightHandle);
  89. }
  90. if (m_pointLightHandle.IsValid())
  91. {
  92. AZ::Render::PointLightFeatureProcessorInterface* pointLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::PointLightFeatureProcessorInterface>();
  93. pointLightFeatureProcessor->ReleaseLight(m_pointLightHandle);
  94. }
  95. }
  96. }
  97. void DiffuseGIExampleComponent::SetSampleScene(bool geometryOnly /*= false*/)
  98. {
  99. UnloadSampleScene(geometryOnly);
  100. // set new scene
  101. switch (m_sampleScene)
  102. {
  103. case SampleScene::CornellBox:
  104. CreateCornellBoxScene(geometryOnly);
  105. break;
  106. case SampleScene::Sponza:
  107. CreateSponzaScene();
  108. break;
  109. };
  110. }
  111. void DiffuseGIExampleComponent::Deactivate()
  112. {
  113. // disable camera
  114. Camera::CameraRequestBus::Event(GetCameraEntityId(), &Camera::CameraRequestBus::Events::SetFarClipDistance, m_originalFarClipDistance);
  115. AZ::Debug::CameraControllerRequestBus::Event(GetCameraEntityId(), &AZ::Debug::CameraControllerRequestBus::Events::Disable);
  116. AZ::TickBus::Handler::BusDisconnect();
  117. m_imguiSidebar.Deactivate();
  118. m_defaultIbl.Reset();
  119. UnloadSampleScene(false);
  120. }
  121. float DiffuseGIExampleComponent::ComputePointLightAttenuationRadius(AZ::Color color, float intensity)
  122. {
  123. static const float CutoffIntensity = 0.1f;
  124. float luminance = AZ::Render::PhotometricValue::GetPerceptualLuminance(color * intensity);
  125. return sqrt(luminance / CutoffIntensity);
  126. }
  127. void DiffuseGIExampleComponent::LoadSampleSceneAssets()
  128. {
  129. // load plane and cube models
  130. // all geometry in the CornellBox is created from planes and boxes
  131. static constexpr const char PlaneModelPath[] = "objects/plane.azmodel";
  132. AZ::Data::AssetId planeAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(PlaneModelPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  133. m_planeModelAsset.Create(planeAssetId);
  134. static constexpr const char CubeModelPath[] = "objects/cube.azmodel";
  135. AZ::Data::AssetId cubeAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(CubeModelPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  136. m_cubeModelAsset.Create(cubeAssetId);
  137. // materials for the CornellBox
  138. static constexpr const char RedMaterialPath[] = "materials/diffusegiexample/red.azmaterial";
  139. AZ::Data::AssetId redMaterialAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(RedMaterialPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  140. m_redMaterialAsset.Create(redMaterialAssetId);
  141. static constexpr const char GreenMaterialPath[] = "materials/diffusegiexample/green.azmaterial";
  142. AZ::Data::AssetId greenMaterialAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(GreenMaterialPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  143. m_greenMaterialAsset.Create(greenMaterialAssetId);
  144. static constexpr const char BlueMaterialPath[] = "materials/diffusegiexample/blue.azmaterial";
  145. AZ::Data::AssetId blueMaterialAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(BlueMaterialPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  146. m_blueMaterialAsset.Create(blueMaterialAssetId);
  147. static constexpr const char YellowMaterialPath[] = "materials/diffusegiexample/yellow.azmaterial";
  148. AZ::Data::AssetId yellowMaterialAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(YellowMaterialPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  149. m_yellowMaterialAsset.Create(yellowMaterialAssetId);
  150. static constexpr const char WhiteMaterialPath[] = "materials/diffusegiexample/white.azmaterial";
  151. AZ::Data::AssetId whiteMaterialAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(WhiteMaterialPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  152. m_whiteMaterialAsset.Create(whiteMaterialAssetId);
  153. // Sponza models
  154. static constexpr const char InteriorModelPath[] = "objects/sponza.azmodel";
  155. AZ::Data::AssetId interiorAssetId = AZ::RPI::AssetUtils::GetAssetIdForProductPath(InteriorModelPath, AZ::RPI::AssetUtils::TraceLevel::Error);
  156. m_sponzaModelAsset.Create(interiorAssetId);
  157. // diffuse IBL cubemap
  158. const constexpr char* DiffuseAssetPath = "lightingpresets/lowcontrast/palermo_sidewalk_4k_iblskyboxcm_ibldiffuse.exr.streamingimage";
  159. if (!m_diffuseImageAsset.IsReady())
  160. {
  161. m_diffuseImageAsset = AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::StreamingImageAsset>(DiffuseAssetPath, AZ::RPI::AssetUtils::TraceLevel::Assert);
  162. m_diffuseImageAsset.QueueLoad();
  163. }
  164. }
  165. AZ::Data::Asset<AZ::RPI::MaterialAsset>& DiffuseGIExampleComponent::GetCornellBoxMaterialAsset(CornellBoxColors color)
  166. {
  167. switch (color)
  168. {
  169. case CornellBoxColors::Red:
  170. return m_redMaterialAsset;
  171. case CornellBoxColors::Green:
  172. return m_greenMaterialAsset;
  173. case CornellBoxColors::Blue:
  174. return m_blueMaterialAsset;
  175. case CornellBoxColors::Yellow:
  176. return m_yellowMaterialAsset;
  177. case CornellBoxColors::White:
  178. return m_whiteMaterialAsset;
  179. };
  180. AZ_Assert(false, "Unknown material color");
  181. return m_whiteMaterialAsset;
  182. }
  183. void DiffuseGIExampleComponent::CreateCornellBoxScene(bool geometryOnly)
  184. {
  185. m_meshHandles.resize(aznumeric_cast<uint32_t>(CornellBoxMeshes::Count));
  186. // left wall
  187. if (m_leftWallVisible)
  188. {
  189. AZ::Render::MaterialAssignmentMap materialMap;
  190. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = GetCornellBoxMaterialAsset(m_leftWallColor);
  191. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(GetCornellBoxMaterialAsset(m_leftWallColor));
  192. AZ::Transform transform = AZ::Transform::CreateIdentity();
  193. transform.SetTranslation(-5.0f, 0.0f, 0.0f);
  194. transform *= AZ::Transform::CreateRotationY(AZ::Constants::HalfPi);
  195. AZ::Vector3 nonUniformScale(10.05f, 10.05f, 1.0f);
  196. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::LeftWall)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_planeModelAsset }, materialMap);
  197. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::LeftWall)], transform, nonUniformScale);
  198. }
  199. // right wall
  200. if (m_rightWallVisible)
  201. {
  202. AZ::Render::MaterialAssignmentMap materialMap;
  203. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = GetCornellBoxMaterialAsset(m_rightWallColor);
  204. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(GetCornellBoxMaterialAsset(m_rightWallColor));
  205. AZ::Transform transform = AZ::Transform::CreateIdentity();
  206. transform.SetTranslation(5.0f, 0.0f, 0.0f);
  207. transform *= AZ::Transform::CreateRotationY(-AZ::Constants::HalfPi);
  208. AZ::Vector3 nonUniformScale(10.05f, 10.05f, 1.0f);
  209. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::RightWall)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_planeModelAsset }, materialMap);
  210. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::RightWall)], transform, nonUniformScale);
  211. }
  212. // back wall
  213. if (m_backWallVisible)
  214. {
  215. AZ::Render::MaterialAssignmentMap materialMap;
  216. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = GetCornellBoxMaterialAsset(m_backWallColor);
  217. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(GetCornellBoxMaterialAsset(m_backWallColor));
  218. AZ::Transform transform = AZ::Transform::CreateIdentity();
  219. transform.SetTranslation(0.0f, 5.0f, 0.0f);
  220. transform *= AZ::Transform::CreateRotationX(AZ::Constants::HalfPi);
  221. AZ::Vector3 nonUniformScale(10.05f, 10.05f, 1.0f);
  222. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::BackWall)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_planeModelAsset }, materialMap);
  223. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::BackWall)], transform, nonUniformScale);
  224. }
  225. // ceiling
  226. if (m_ceilingVisible)
  227. {
  228. AZ::Render::MaterialAssignmentMap materialMap;
  229. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = GetCornellBoxMaterialAsset(m_ceilingColor);
  230. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(GetCornellBoxMaterialAsset(m_ceilingColor));
  231. AZ::Transform transform = AZ::Transform::CreateIdentity();
  232. transform.SetTranslation(0.0f, 0.0f, 5.0f);
  233. transform *= AZ::Transform::CreateRotationX(AZ::Constants::Pi);
  234. AZ::Vector3 nonUniformScale(10.05f, 10.05f, 1.0f);
  235. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::Ceiling)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_planeModelAsset }, materialMap);
  236. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::Ceiling)], transform, nonUniformScale);
  237. }
  238. // floor
  239. if (m_floorVisible)
  240. {
  241. AZ::Render::MaterialAssignmentMap materialMap;
  242. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = GetCornellBoxMaterialAsset(m_floorColor);
  243. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(GetCornellBoxMaterialAsset(m_floorColor));
  244. AZ::Transform transform = AZ::Transform::CreateIdentity();
  245. transform.SetTranslation(0.0f, 0.0f, -5.0f);
  246. AZ::Vector3 nonUniformScale(10.05f, 10.05f, 1.0f);
  247. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::Floor)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_planeModelAsset }, materialMap);
  248. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::Floor)], transform, nonUniformScale);
  249. }
  250. // large box
  251. {
  252. AZ::Render::MaterialAssignmentMap materialMap;
  253. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = m_whiteMaterialAsset;
  254. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(m_whiteMaterialAsset);
  255. AZ::Transform transform = AZ::Transform::CreateIdentity();
  256. transform.SetTranslation(-2.0f, 0.0f, -2.0f);
  257. transform *= AZ::Transform::CreateRotationZ(AZ::Constants::HalfPi * 0.2f);
  258. AZ::Vector3 nonUniformScale(3.0f, 3.0f, 6.0f);
  259. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::LargeBox)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_cubeModelAsset }, materialMap);
  260. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::LargeBox)], transform, nonUniformScale);
  261. }
  262. // small box
  263. {
  264. AZ::Render::MaterialAssignmentMap materialMap;
  265. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialAsset = m_whiteMaterialAsset;
  266. materialMap[AZ::Render::DefaultMaterialAssignmentId].m_materialInstance = AZ::RPI::Material::FindOrCreate(m_whiteMaterialAsset);
  267. AZ::Transform transform = AZ::Transform::CreateIdentity();
  268. transform.SetTranslation(2.0f, -1.5f, -3.5f);
  269. transform *= AZ::Transform::CreateRotationZ(-AZ::Constants::HalfPi * 0.2f);
  270. AZ::Vector3 nonUniformScale(3.0f, 3.0f, 3.0f);
  271. m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::SmallBox)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_cubeModelAsset }, materialMap);
  272. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(CornellBoxMeshes::SmallBox)], transform, nonUniformScale);
  273. }
  274. // stop now if we were only loading the geometry
  275. if (geometryOnly)
  276. {
  277. return;
  278. }
  279. m_pointLightPos = AZ::Vector3(0.0f, -1.0f, 4.0f);
  280. m_pointLightColor = AZ::Color(1.0f, 1.0f, 1.0f, 1.0f);
  281. m_pointLightIntensity = 20.0f;
  282. // point light
  283. {
  284. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  285. AZ::Render::PointLightFeatureProcessorInterface* pointLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::PointLightFeatureProcessorInterface>();
  286. m_pointLightHandle = pointLightFeatureProcessor->AcquireLight();
  287. pointLightFeatureProcessor->SetPosition(m_pointLightHandle, m_pointLightPos);
  288. pointLightFeatureProcessor->SetRgbIntensity(m_pointLightHandle, AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Candela>(m_pointLightIntensity * m_pointLightColor));
  289. pointLightFeatureProcessor->SetBulbRadius(m_pointLightHandle, 1.0f);
  290. pointLightFeatureProcessor->SetAttenuationRadius(m_pointLightHandle, ComputePointLightAttenuationRadius(m_pointLightColor, m_pointLightIntensity));
  291. }
  292. // diffuse probe grid
  293. {
  294. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  295. AZ::Render::DiffuseProbeGridFeatureProcessorInterface* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DiffuseProbeGridFeatureProcessorInterface>();
  296. AZ::Transform transform = AZ::Transform::CreateIdentity();
  297. m_origin.Set(0.3f, -0.25f, 0.5f);
  298. transform.SetTranslation(m_origin);
  299. m_diffuseProbeGrid = diffuseProbeGridFeatureProcessor->AddProbeGrid(transform, AZ::Vector3(12.0f, 12.0f, 12.0f), AZ::Vector3(1.5f, 1.5f, 2.0f));
  300. diffuseProbeGridFeatureProcessor->SetAmbientMultiplier(m_diffuseProbeGrid, m_ambientMultiplier);
  301. m_viewBias = 0.7f;
  302. diffuseProbeGridFeatureProcessor->SetViewBias(m_diffuseProbeGrid, m_viewBias);
  303. m_normalBias = 0.1f;
  304. diffuseProbeGridFeatureProcessor->SetNormalBias(m_diffuseProbeGrid, m_normalBias);
  305. AZ::Render::DiffuseGlobalIlluminationFeatureProcessorInterface* diffuseGlobalIlluminationFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DiffuseGlobalIlluminationFeatureProcessorInterface>();
  306. diffuseGlobalIlluminationFeatureProcessor->SetQualityLevel(AZ::Render::DiffuseGlobalIlluminationQualityLevel::Medium);
  307. }
  308. // camera
  309. m_cameraTranslation = AZ::Vector3(0.0f, -17.5f, 0.0f);
  310. m_cameraHeading = 0.0f;
  311. // disable diffuse Ibl
  312. DisableGlobalIbl();
  313. }
  314. void DiffuseGIExampleComponent::CreateSponzaScene()
  315. {
  316. m_meshHandles.resize(aznumeric_cast<uint32_t>(SponzaMeshes::Count));
  317. AZ::Transform transform = AZ::Transform::CreateIdentity();
  318. m_meshHandles[aznumeric_cast<uint32_t>(SponzaMeshes::Inside)] = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ m_sponzaModelAsset });
  319. GetMeshFeatureProcessor()->SetTransform(m_meshHandles[aznumeric_cast<uint32_t>(SponzaMeshes::Inside)], transform);
  320. m_directionalLightPitch = AZ::DegToRad(-45.0f);
  321. m_directionalLightYaw = AZ::DegToRad(62.0f);
  322. m_directionalLightColor = AZ::Color(0.92f, 0.78f, 0.35f, 1.0f);
  323. m_directionalLightIntensity = 30.0f;
  324. m_pointLightPos = AZ::Vector3(10.0f, -4.25f, 1.5f);
  325. m_pointLightColor = AZ::Color(1.0f, 0.0f, 0.0f, 1.0f);
  326. m_pointLightIntensity = 10.0f;
  327. m_ambientMultiplier = 1.0f;
  328. // directional light
  329. {
  330. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  331. AZ::Render::DirectionalLightFeatureProcessorInterface* directionalLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DirectionalLightFeatureProcessorInterface>();
  332. m_directionalLightHandle = directionalLightFeatureProcessor->AcquireLight();
  333. const auto lightTransform = AZ::Transform::CreateRotationZ(m_directionalLightYaw) * AZ::Transform::CreateRotationX(m_directionalLightPitch);
  334. directionalLightFeatureProcessor->SetDirection(m_directionalLightHandle, lightTransform.GetBasis(1));
  335. directionalLightFeatureProcessor->SetRgbIntensity(m_directionalLightHandle, AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Lux>(m_directionalLightIntensity * m_directionalLightColor));
  336. directionalLightFeatureProcessor->SetCascadeCount(m_directionalLightHandle, 4);
  337. directionalLightFeatureProcessor->SetShadowmapSize(m_directionalLightHandle, AZ::Render::ShadowmapSize::Size2048);
  338. directionalLightFeatureProcessor->SetViewFrustumCorrectionEnabled(m_directionalLightHandle, false);
  339. directionalLightFeatureProcessor->SetShadowFilterMethod(m_directionalLightHandle, AZ::Render::ShadowFilterMethod::EsmPcf);
  340. directionalLightFeatureProcessor->SetShadowBoundaryWidth(m_directionalLightHandle, 0.03f);
  341. directionalLightFeatureProcessor->SetPredictionSampleCount(m_directionalLightHandle, 4);
  342. directionalLightFeatureProcessor->SetFilteringSampleCount(m_directionalLightHandle, 16);
  343. directionalLightFeatureProcessor->SetGroundHeight(m_directionalLightHandle, 0.0f);
  344. }
  345. // point light
  346. {
  347. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  348. AZ::Render::PointLightFeatureProcessorInterface* pointLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::PointLightFeatureProcessorInterface>();
  349. m_pointLightHandle = pointLightFeatureProcessor->AcquireLight();
  350. pointLightFeatureProcessor->SetPosition(m_pointLightHandle, m_pointLightPos);
  351. pointLightFeatureProcessor->SetRgbIntensity(m_pointLightHandle, AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Candela>(m_pointLightIntensity * m_pointLightColor));
  352. pointLightFeatureProcessor->SetBulbRadius(m_pointLightHandle, 1.0f);
  353. pointLightFeatureProcessor->SetAttenuationRadius(m_pointLightHandle, ComputePointLightAttenuationRadius(m_pointLightColor, m_pointLightIntensity));
  354. }
  355. // diffuse probe grid
  356. {
  357. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  358. AZ::Render::DiffuseProbeGridFeatureProcessorInterface* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DiffuseProbeGridFeatureProcessorInterface>();
  359. transform = AZ::Transform::CreateIdentity();
  360. m_origin.Set(1.4f, -1.25f, 5.0f);
  361. transform.SetTranslation(m_origin);
  362. m_diffuseProbeGrid = diffuseProbeGridFeatureProcessor->AddProbeGrid(transform, AZ::Vector3(35.0f, 45.0f, 25.0f), AZ::Vector3(3.0f, 3.0f, 4.0f));
  363. diffuseProbeGridFeatureProcessor->SetAmbientMultiplier(m_diffuseProbeGrid, m_ambientMultiplier);
  364. m_viewBias = 0.55f;
  365. diffuseProbeGridFeatureProcessor->SetViewBias(m_diffuseProbeGrid, m_viewBias);
  366. m_normalBias = 0.4f;
  367. diffuseProbeGridFeatureProcessor->SetNormalBias(m_diffuseProbeGrid, m_normalBias);
  368. AZ::Render::DiffuseGlobalIlluminationFeatureProcessorInterface* diffuseGlobalIlluminationFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DiffuseGlobalIlluminationFeatureProcessorInterface>();
  369. diffuseGlobalIlluminationFeatureProcessor->SetQualityLevel(AZ::Render::DiffuseGlobalIlluminationQualityLevel::Medium);
  370. }
  371. // camera
  372. m_cameraTranslation = AZ::Vector3(5.0f, 0.0f, 5.0f);
  373. m_cameraHeading = 90.0f;
  374. // diffuse Ibl
  375. EnableDiffuseIbl();
  376. }
  377. void DiffuseGIExampleComponent::OnTick([[maybe_unused]] float deltaTime, [[maybe_unused]] AZ::ScriptTimePoint time)
  378. {
  379. if (m_resetCamera)
  380. {
  381. AZ::Debug::CameraControllerRequestBus::Event(GetCameraEntityId(), &AZ::Debug::CameraControllerRequestBus::Events::Reset);
  382. AZ::TransformBus::Event(GetCameraEntityId(), &AZ::TransformBus::Events::SetWorldTranslation, m_cameraTranslation);
  383. AZ::Debug::CameraControllerRequestBus::Event(GetCameraEntityId(), &AZ::Debug::CameraControllerRequestBus::Events::Enable, azrtti_typeid<AZ::Debug::NoClipControllerComponent>());
  384. AZ::Debug::NoClipControllerRequestBus::Event(GetCameraEntityId(), &AZ::Debug::NoClipControllerRequestBus::Events::SetHeading, AZ::DegToRad(m_cameraHeading));
  385. m_resetCamera = false;
  386. }
  387. // ImGui sidebar
  388. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  389. AZ::Render::DiffuseProbeGridFeatureProcessorInterface* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DiffuseProbeGridFeatureProcessorInterface>();
  390. AZ::Render::PointLightFeatureProcessorInterface* pointLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::PointLightFeatureProcessorInterface>();
  391. AZ::Render::DirectionalLightFeatureProcessorInterface* directionalLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::DirectionalLightFeatureProcessorInterface>();
  392. bool sceneChanged = false;
  393. // initialize the scene in OnTick() in order to delay a frame between samples
  394. if (m_sampleScene == SampleScene::None)
  395. {
  396. m_sampleScene = CornellBox;
  397. sceneChanged = true;
  398. }
  399. if (m_imguiSidebar.Begin())
  400. {
  401. // sample scene
  402. {
  403. ImGui::Text("Scene");
  404. sceneChanged |= ImGui::RadioButton("Cornell Box", (int*)&m_sampleScene, aznumeric_cast<uint32_t>(CornellBox));
  405. sceneChanged |= ImGui::RadioButton("Sponza", (int*)&m_sampleScene, aznumeric_cast<uint32_t>(Sponza));
  406. ImGui::NewLine();
  407. }
  408. // diffuse probe grid settings
  409. {
  410. bool diffuseProbeGridChanged = false;
  411. diffuseProbeGridChanged |= ImGui::Checkbox("Enable Diffuse GI", &m_enableDiffuseGI);
  412. diffuseProbeGridChanged |= ImGui::SliderFloat("##AmbientMultiplier", &m_ambientMultiplier, 0.0f, 10.0f);
  413. ImGui::NewLine();
  414. ImGui::Text("View Bias");
  415. diffuseProbeGridChanged |= ImGui::SliderFloat("##ViewBias", &m_viewBias, 0.01f, 2.0f);
  416. ImGui::Text("Normal Bias");
  417. diffuseProbeGridChanged |= ImGui::SliderFloat("##NormalBias", &m_normalBias, 0.01f, 2.0f);
  418. diffuseProbeGridChanged |= ImGui::Checkbox("GI Shadows", &m_giShadows);
  419. ImGui::NewLine();
  420. ImGui::Text("Grid Origin");
  421. float originX = m_origin.GetX();
  422. diffuseProbeGridChanged |= ImGui::SliderFloat("##OriginX", &originX, -10.0f, 10.0f);
  423. float originY = m_origin.GetY();
  424. diffuseProbeGridChanged |= ImGui::SliderFloat("##OriginY", &originY, -10.0f, 10.0f);
  425. float originZ = m_origin.GetZ();
  426. diffuseProbeGridChanged |= ImGui::SliderFloat("##OriginZ", &originZ, -10.0f, 10.0f);
  427. m_origin.SetX(originX);
  428. m_origin.SetY(originY);
  429. m_origin.SetZ(originZ);
  430. if (diffuseProbeGridChanged && diffuseProbeGridFeatureProcessor->IsValidProbeGridHandle(m_diffuseProbeGrid))
  431. {
  432. diffuseProbeGridFeatureProcessor->Enable(m_diffuseProbeGrid, m_enableDiffuseGI);
  433. diffuseProbeGridFeatureProcessor->SetAmbientMultiplier(m_diffuseProbeGrid, m_ambientMultiplier);
  434. diffuseProbeGridFeatureProcessor->SetViewBias(m_diffuseProbeGrid, m_viewBias);
  435. diffuseProbeGridFeatureProcessor->SetNormalBias(m_diffuseProbeGrid, m_normalBias);
  436. diffuseProbeGridFeatureProcessor->SetGIShadows(m_diffuseProbeGrid, m_giShadows);
  437. AZ::Transform transform = AZ::Transform::CreateIdentity();
  438. transform.SetTranslation(m_origin);
  439. diffuseProbeGridFeatureProcessor->SetTransform(m_diffuseProbeGrid, transform);
  440. }
  441. ImGui::NewLine();
  442. }
  443. // diffuse IBL (Sponza only)
  444. if (m_sampleScene == SampleScene::Sponza)
  445. {
  446. bool diffuseIblChanged = false;
  447. diffuseIblChanged |= ImGui::Checkbox("Diffuse IBL (Sky Light)", &m_useDiffuseIbl);
  448. diffuseIblChanged |= ImGui::SliderFloat("Exposure", &m_diffuseIblExposure, -10.0f, 10.0f);
  449. if (diffuseIblChanged && diffuseProbeGridFeatureProcessor->IsValidProbeGridHandle(m_diffuseProbeGrid))
  450. {
  451. diffuseProbeGridFeatureProcessor->SetUseDiffuseIbl(m_diffuseProbeGrid, m_useDiffuseIbl);
  452. if (m_useDiffuseIbl)
  453. {
  454. EnableDiffuseIbl();
  455. }
  456. else
  457. {
  458. DisableGlobalIbl();
  459. }
  460. }
  461. ImGui::NewLine();
  462. }
  463. // directional light
  464. if (m_directionalLightHandle.IsValid())
  465. {
  466. bool directionalLightChanged = false;
  467. ImGui::Text("Directional Light");
  468. directionalLightChanged |= ImGui::SliderAngle("Pitch", &m_directionalLightPitch, -90.0f, 0.f);
  469. directionalLightChanged |= ImGui::SliderAngle("Yaw", &m_directionalLightYaw, 0.f, 360.f);
  470. ImGui::Text("Color");
  471. float directionalLightColor[4];
  472. m_directionalLightColor.StoreToFloat4(&directionalLightColor[0]);
  473. directionalLightChanged |= ImGui::ColorEdit3("", &directionalLightColor[0]);
  474. ImGui::Text("Intensity");
  475. directionalLightChanged |= ImGui::SliderFloat("##DirectionalLightIntensity", &m_directionalLightIntensity, 0, 100);
  476. if (directionalLightChanged)
  477. {
  478. const auto lightTransform = AZ::Transform::CreateRotationZ(m_directionalLightYaw) * AZ::Transform::CreateRotationX(m_directionalLightPitch);
  479. directionalLightFeatureProcessor->SetDirection(m_directionalLightHandle, lightTransform.GetBasis(1));
  480. m_directionalLightColor = AZ::Color(directionalLightColor[0], directionalLightColor[1], directionalLightColor[2], 1.0f);
  481. directionalLightFeatureProcessor->SetRgbIntensity(m_directionalLightHandle, AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Lux>(m_directionalLightColor * m_directionalLightIntensity));
  482. }
  483. // Camera Configuration
  484. {
  485. Camera::Configuration config;
  486. Camera::CameraRequestBus::EventResult(
  487. config,
  488. GetCameraEntityId(),
  489. &Camera::CameraRequestBus::Events::GetCameraConfiguration);
  490. directionalLightFeatureProcessor->SetCameraConfiguration(
  491. m_directionalLightHandle,
  492. config);
  493. }
  494. // Camera Transform
  495. {
  496. AZ::Transform transform = AZ::Transform::CreateIdentity();
  497. AZ::TransformBus::EventResult(
  498. transform,
  499. GetCameraEntityId(),
  500. &AZ::TransformBus::Events::GetWorldTM);
  501. directionalLightFeatureProcessor->SetCameraTransform(
  502. m_directionalLightHandle, transform);
  503. }
  504. ImGui::NewLine();
  505. }
  506. // point light
  507. if (m_pointLightHandle.IsValid())
  508. {
  509. bool pointLightChanged = false;
  510. ImGui::Text("Point Light");
  511. float lightX = m_pointLightPos.GetX();
  512. pointLightChanged |= ImGui::SliderFloat("##PointLightX", &lightX, -10.0f, 10.0f);
  513. float lightY = m_pointLightPos.GetY();
  514. pointLightChanged |= ImGui::SliderFloat("##PointLightY", &lightY, -10.0f, 10.0f);
  515. float lightZ = m_pointLightPos.GetZ();
  516. pointLightChanged |= ImGui::SliderFloat("##PointLightZ", &lightZ, -10.0f, 10.0f);
  517. m_pointLightPos.SetX(lightX);
  518. m_pointLightPos.SetY(lightY);
  519. m_pointLightPos.SetZ(lightZ);
  520. float pointLightColor[4];
  521. m_pointLightColor.StoreToFloat4(&pointLightColor[0]);
  522. pointLightChanged |= ImGui::ColorEdit3("Point Light Color", &pointLightColor[0]);
  523. ImGui::Text("Intensity");
  524. pointLightChanged |= ImGui::SliderFloat("##PointLightIntensity", &m_pointLightIntensity, 0, 100);
  525. if (pointLightChanged)
  526. {
  527. m_pointLightColor = AZ::Color(pointLightColor[0], pointLightColor[1], pointLightColor[2], 1.0f);
  528. pointLightFeatureProcessor->SetPosition(m_pointLightHandle, m_pointLightPos);
  529. pointLightFeatureProcessor->SetRgbIntensity(m_pointLightHandle, AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Candela>(m_pointLightColor * m_pointLightIntensity));
  530. pointLightFeatureProcessor->SetAttenuationRadius(m_pointLightHandle, ComputePointLightAttenuationRadius(m_pointLightColor, m_pointLightIntensity));
  531. }
  532. ImGui::NewLine();
  533. }
  534. // Cornell Box specific
  535. if (m_sampleScene == CornellBox)
  536. {
  537. bool geometryChanged = false;
  538. geometryChanged |= ImGui::Checkbox("Left Wall Visible", &m_leftWallVisible);
  539. geometryChanged |= ImGui::Combo("Left Wall Color", (int*)&m_leftWallColor, CornellBoxColorNames, CornellBoxColorNamesCount);
  540. geometryChanged |= ImGui::Checkbox("Right Wall Visible", &m_rightWallVisible);
  541. geometryChanged |= ImGui::Combo("Right Wall Color", (int*)&m_rightWallColor, CornellBoxColorNames, CornellBoxColorNamesCount);
  542. geometryChanged |= ImGui::Checkbox("Back Wall Visible", &m_backWallVisible);
  543. geometryChanged |= ImGui::Combo("Back Wall Color", (int*)&m_backWallColor, CornellBoxColorNames, CornellBoxColorNamesCount);
  544. geometryChanged |= ImGui::Checkbox("Floor Visible", &m_floorVisible);
  545. geometryChanged |= ImGui::Combo("Floor Color", (int*)&m_floorColor, CornellBoxColorNames, CornellBoxColorNamesCount);
  546. geometryChanged |= ImGui::Checkbox("Ceiling Visible", &m_ceilingVisible);
  547. geometryChanged |= ImGui::Combo("Ceiling Color", (int*)&m_ceilingColor, CornellBoxColorNames, CornellBoxColorNamesCount);
  548. if (geometryChanged)
  549. {
  550. SetSampleScene(true);
  551. }
  552. }
  553. m_imguiSidebar.End();
  554. }
  555. if (sceneChanged)
  556. {
  557. m_resetCamera = true;
  558. SetSampleScene();
  559. }
  560. }
  561. void DiffuseGIExampleComponent::OnEntityDestruction(const AZ::EntityId& entityId)
  562. {
  563. AZ::EntityBus::MultiHandler::BusDisconnect(entityId);
  564. }
  565. void DiffuseGIExampleComponent::DisableGlobalIbl()
  566. {
  567. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  568. // disable Ibl by setting the empty cubemap
  569. const constexpr char* DiffuseAssetPath = "textures/default/default_iblglobalcm_ibldiffuse.dds.streamingimage";
  570. const constexpr char* SpecularAssetPath = "textures/default/default_iblglobalcm_iblspecular.dds.streamingimage";
  571. auto assertTraceLevel = AZ::RPI::AssetUtils::TraceLevel::Assert;
  572. AZ::Data::Asset<AZ::RPI::StreamingImageAsset> diffuseImageAsset =
  573. AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::StreamingImageAsset>
  574. (DiffuseAssetPath, assertTraceLevel);
  575. AZ::Data::Asset<AZ::RPI::StreamingImageAsset> specularImageAsset =
  576. AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::StreamingImageAsset>
  577. (SpecularAssetPath, assertTraceLevel);
  578. auto featureProcessor = scene->GetFeatureProcessor<AZ::Render::ImageBasedLightFeatureProcessorInterface>();
  579. AZ_Assert(featureProcessor, "Unable to find ImageBasedLightFeatureProcessorInterface on scene.");
  580. featureProcessor->SetDiffuseImage(diffuseImageAsset);
  581. featureProcessor->SetSpecularImage(specularImageAsset);
  582. }
  583. void DiffuseGIExampleComponent::EnableDiffuseIbl()
  584. {
  585. AZ::RPI::ScenePtr scene = AZ::RPI::RPISystemInterface::Get()->GetDefaultScene();
  586. AZ::Render::ImageBasedLightFeatureProcessorInterface* imageBaseLightFeatureProcessor = scene->GetFeatureProcessor<AZ::Render::ImageBasedLightFeatureProcessorInterface>();
  587. AZ_Assert(imageBaseLightFeatureProcessor, "Unable to find ImageBasedLightFeatureProcessorInterface on scene.");
  588. imageBaseLightFeatureProcessor->SetDiffuseImage(m_diffuseImageAsset);
  589. imageBaseLightFeatureProcessor->SetExposure(m_diffuseIblExposure);
  590. }
  591. } // namespace AtomSampleViewer