ParallaxMappingExampleComponent.cpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. /*
  2. * Copyright (c) Contributors to the Open 3D Engine Project.
  3. * For complete copyright and license terms please see the LICENSE at the root of this distribution.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0 OR MIT
  6. *
  7. */
  8. #include <ParallaxMappingExampleComponent.h>
  9. #include <Atom/Component/DebugCamera/ArcBallControllerComponent.h>
  10. #include <Atom/RPI.Public/RPISystemInterface.h>
  11. #include <Atom/RPI.Public/Scene.h>
  12. #include <Atom/RPI.Reflect/Asset/AssetUtils.h>
  13. #include <Automation/ScriptableImGui.h>
  14. #include <AzCore/Component/TransformBus.h>
  15. #include <RHI/BasicRHIComponent.h>
  16. namespace AtomSampleViewer
  17. {
  18. static const char* ParallaxEnableName = "parallax.useTexture";
  19. static const char* PdoEnableName = "parallax.pdo";
  20. static const char* ParallaxFactorName = "parallax.factor";
  21. static const char* ParallaxHeightOffsetName = "parallax.offset";
  22. static const char* ParallaxShowClippingName = "parallax.showClipping";
  23. static const char* ParallaxAlgorithmName = "parallax.algorithm";
  24. static const char* ParallaxQualityName = "parallax.quality";
  25. static const char* ParallaxUvIndexName = "parallax.textureMapUv";
  26. static const char* AmbientOcclusionUvIndexName = "occlusion.diffuseTextureMapUv";
  27. static const char* BaseColorUvIndexName = "baseColor.textureMapUv";
  28. static const char* NormalUvIndexName = "normal.textureMapUv";
  29. static const char* RoughnessUvIndexName = "roughness.textureMapUv";
  30. static const char* CenterUVName = "uv.center";
  31. static const char* TileUName = "uv.tileU";
  32. static const char* TileVName = "uv.tileV";
  33. static const char* OffsetUName = "uv.offsetU";
  34. static const char* OffsetVName = "uv.offsetV";
  35. static const char* RotationUVName = "uv.rotateDegrees";
  36. static const char* ScaleUVName = "uv.scale";
  37. // Must align with enum value in StandardPbr.materialtype
  38. static const char* ParallaxAlgorithmList[] =
  39. {
  40. "Basic", "Steep", "POM", "Relief", "ContactRefinement"
  41. };
  42. static const char* ParallaxQualityList[] =
  43. {
  44. "Low", "Medium", "High", "Ultra"
  45. };
  46. static const char* ParallaxUvSetList[] =
  47. {
  48. "UV0", "UV1"
  49. };
  50. void ParallaxMappingExampleComponent::Reflect(AZ::ReflectContext* context)
  51. {
  52. if (AZ::SerializeContext* serializeContext = azrtti_cast<AZ::SerializeContext*>(context))
  53. {
  54. serializeContext->Class < ParallaxMappingExampleComponent, AZ::Component>()->Version(0);
  55. }
  56. }
  57. ParallaxMappingExampleComponent::ParallaxMappingExampleComponent()
  58. : m_imguiSidebar("@user@/ParallaxMappingExampleComponent/sidebar.xml")
  59. {
  60. }
  61. void ParallaxMappingExampleComponent::Activate()
  62. {
  63. // Asset
  64. m_planeAsset = AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::ModelAsset>("objects/plane.azmodel", AZ::RPI::AssetUtils::TraceLevel::Assert);
  65. m_boxAsset = AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::ModelAsset>("objects/cube.azmodel", AZ::RPI::AssetUtils::TraceLevel::Assert);
  66. m_parallaxMaterialAsset = AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::MaterialAsset>("testdata/materials/parallaxrock.azmaterial", AZ::RPI::AssetUtils::TraceLevel::Assert);
  67. m_defaultMaterialAsset = AZ::RPI::AssetUtils::GetAssetByProductPath<AZ::RPI::MaterialAsset>("materials/defaultpbr.azmaterial", AZ::RPI::AssetUtils::TraceLevel::Assert);
  68. m_parallaxMaterial = AZ::RPI::Material::Create(m_parallaxMaterialAsset);
  69. m_defaultMaterial = AZ::RPI::Material::Create(m_defaultMaterialAsset);
  70. m_planeTransform = AZ::Transform::CreateUniformScale(5);
  71. m_planeHandle = LoadMesh(m_planeAsset, m_parallaxMaterial, m_planeTransform);
  72. m_boxHandle = LoadMesh(m_boxAsset, m_defaultMaterial, AZ::Transform::CreateIdentity());
  73. // Material index
  74. m_parallaxEnableIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxEnableName));
  75. m_parallaxFactorIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxFactorName));
  76. m_parallaxOffsetIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxHeightOffsetName));
  77. m_parallaxShowClippingIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxShowClippingName));
  78. m_parallaxAlgorithmIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxAlgorithmName));
  79. m_parallaxQualityIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxQualityName));
  80. m_parallaxUvIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ParallaxUvIndexName));
  81. m_pdoEnableIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(PdoEnableName));
  82. m_ambientOcclusionUvIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(AmbientOcclusionUvIndexName));
  83. m_baseColorUvIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(BaseColorUvIndexName));
  84. m_normalUvIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(NormalUvIndexName));
  85. m_roughnessUvIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(RoughnessUvIndexName));
  86. m_centerUVIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(CenterUVName));
  87. m_tileUIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(TileUName));
  88. m_tileVIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(TileVName));
  89. m_offsetUIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(OffsetUName));
  90. m_offsetVIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(OffsetVName));
  91. m_rotationUVIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(RotationUVName));
  92. m_scaleUVIndex = m_parallaxMaterial->FindPropertyIndex(AZ::Name(ScaleUVName));
  93. SaveCameraConfiguration();
  94. // Camera
  95. AZ::Debug::CameraControllerRequestBus::Event(
  96. GetCameraEntityId(),
  97. &AZ::Debug::CameraControllerRequestBus::Events::Enable,
  98. azrtti_typeid<AZ::Debug::ArcBallControllerComponent>());
  99. ConfigureCameraToLookDown();
  100. SetCameraConfiguration();
  101. // Light
  102. m_directionalLightFeatureProcessor = m_scene->GetFeatureProcessor<AZ::Render::DirectionalLightFeatureProcessorInterface>();
  103. CreateDirectionalLight();
  104. m_diskLightFeatureProcessor = m_scene->GetFeatureProcessor<AZ::Render::DiskLightFeatureProcessorInterface>();
  105. CreateDiskLight();
  106. m_imguiSidebar.Activate();
  107. AZ::TickBus::Handler::BusConnect();
  108. }
  109. void ParallaxMappingExampleComponent::Deactivate()
  110. {
  111. GetMeshFeatureProcessor()->ReleaseMesh(m_planeHandle);
  112. GetMeshFeatureProcessor()->ReleaseMesh(m_boxHandle);
  113. RestoreCameraConfiguration();
  114. AZ::Debug::CameraControllerRequestBus::Event(
  115. GetCameraEntityId(),
  116. &AZ::Debug::CameraControllerRequestBus::Events::Disable);
  117. m_directionalLightFeatureProcessor->ReleaseLight(m_directionalLightHandle);
  118. m_diskLightFeatureProcessor->ReleaseLight(m_diskLightHandle);
  119. m_imguiSidebar.Deactivate();
  120. AZ::TickBus::Handler::BusDisconnect();
  121. }
  122. void ParallaxMappingExampleComponent::ConfigureCameraToLookDown()
  123. {
  124. const float CameraDistance = 5.0f;
  125. const float CameraPitch = -0.8f;
  126. AZ::Debug::ArcBallControllerRequestBus::Event(GetCameraEntityId(), &AZ::Debug::ArcBallControllerRequestBus::Events::SetPitch, CameraPitch);
  127. AZ::Debug::ArcBallControllerRequestBus::Event(GetCameraEntityId(), &AZ::Debug::ArcBallControllerRequestBus::Events::SetDistance, CameraDistance);
  128. }
  129. AZ::Render::MeshFeatureProcessorInterface::MeshHandle ParallaxMappingExampleComponent::LoadMesh(
  130. AZ::Data::Asset<AZ::RPI::ModelAsset> modelAsset,
  131. AZ::Data::Instance<AZ::RPI::Material> material,
  132. AZ::Transform transform)
  133. {
  134. AZ::Render::MeshFeatureProcessorInterface::MeshHandle meshHandle = GetMeshFeatureProcessor()->AcquireMesh(AZ::Render::MeshHandleDescriptor{ modelAsset }, material);
  135. GetMeshFeatureProcessor()->SetTransform(meshHandle, transform);
  136. return meshHandle;
  137. }
  138. void ParallaxMappingExampleComponent::CreateDirectionalLight()
  139. {
  140. const AZ::Render::DirectionalLightFeatureProcessorInterface::LightHandle handle = m_directionalLightFeatureProcessor->AcquireLight();
  141. m_directionalLightFeatureProcessor->SetShadowmapSize(handle, AZ::Render::ShadowmapSize::Size2048);
  142. m_directionalLightFeatureProcessor->SetCascadeCount(handle, 4);
  143. m_directionalLightFeatureProcessor->SetShadowmapFrustumSplitSchemeRatio(handle, 0.5f);
  144. m_directionalLightFeatureProcessor->SetViewFrustumCorrectionEnabled(handle, true);
  145. m_directionalLightFeatureProcessor->SetShadowFilterMethod(handle, AZ::Render::ShadowFilterMethod::Esm);
  146. m_directionalLightFeatureProcessor->SetFilteringSampleCount(handle, 32);
  147. m_directionalLightFeatureProcessor->SetGroundHeight(handle, 0.f);
  148. m_directionalLightHandle = handle;
  149. }
  150. void ParallaxMappingExampleComponent::CreateDiskLight()
  151. {
  152. AZ::Render::DiskLightFeatureProcessorInterface* const featureProcessor = m_diskLightFeatureProcessor;
  153. const AZ::Render::DiskLightFeatureProcessorInterface::LightHandle handle = featureProcessor->AcquireLight();
  154. featureProcessor->SetAttenuationRadius(handle, sqrtf(500.f / CutoffIntensity));
  155. featureProcessor->SetConeAngles(handle, AZ::DegToRad(22.5f) * ConeAngleInnerRatio, AZ::DegToRad(22.5f));
  156. featureProcessor->SetShadowsEnabled(handle, true);
  157. featureProcessor->SetShadowmapMaxResolution(handle, AZ::Render::ShadowmapSize::Size2048);
  158. m_diskLightHandle = handle;
  159. }
  160. void ParallaxMappingExampleComponent::OnTick(float deltaTime, [[maybe_unused]] AZ::ScriptTimePoint time)
  161. {
  162. if (m_lightAutoRotate)
  163. {
  164. m_lightRotationAngle = fmodf(m_lightRotationAngle + deltaTime, AZ::Constants::TwoPi);
  165. }
  166. const auto location = AZ::Vector3(
  167. 5 * sinf(m_lightRotationAngle),
  168. 5 * cosf(m_lightRotationAngle),
  169. 5);
  170. auto transform = AZ::Transform::CreateLookAt(
  171. location,
  172. AZ::Vector3::CreateZero());
  173. if (m_lightType)
  174. {
  175. AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Lux> directionalLightColor(AZ::Color::CreateZero());
  176. AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Candela> diskLightColor(AZ::Color::CreateOne() * 500.f);
  177. m_directionalLightFeatureProcessor->SetRgbIntensity(m_directionalLightHandle, directionalLightColor);
  178. m_diskLightFeatureProcessor->SetRgbIntensity(m_diskLightHandle, diskLightColor);
  179. }
  180. else
  181. {
  182. AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Lux> directionalLightColor(AZ::Color::CreateOne() * 5.f);
  183. AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Candela> diskLightColor(AZ::Color::CreateZero());
  184. m_directionalLightFeatureProcessor->SetRgbIntensity(m_directionalLightHandle, directionalLightColor);
  185. m_diskLightFeatureProcessor->SetRgbIntensity(m_diskLightHandle, diskLightColor);
  186. }
  187. m_diskLightFeatureProcessor->SetPosition(m_diskLightHandle, location);
  188. m_diskLightFeatureProcessor->SetDirection(m_diskLightHandle, transform.GetBasis(1));
  189. m_directionalLightFeatureProcessor->SetDirection(m_directionalLightHandle, transform.GetBasis(1));
  190. // Camera Configuration
  191. {
  192. Camera::Configuration config;
  193. Camera::CameraRequestBus::EventResult(
  194. config,
  195. GetCameraEntityId(),
  196. &Camera::CameraRequestBus::Events::GetCameraConfiguration);
  197. m_directionalLightFeatureProcessor->SetCameraConfiguration(
  198. m_directionalLightHandle,
  199. config);
  200. }
  201. // Camera Transform
  202. {
  203. transform = AZ::Transform::CreateIdentity();
  204. AZ::TransformBus::EventResult(
  205. transform,
  206. GetCameraEntityId(),
  207. &AZ::TransformBus::Events::GetWorldTM);
  208. m_directionalLightFeatureProcessor->SetCameraTransform(
  209. m_directionalLightHandle, transform);
  210. }
  211. // Plane Transform
  212. {
  213. m_planeTransform.SetRotation(AZ::Quaternion::CreateRotationZ(m_planeRotationAngle));
  214. GetMeshFeatureProcessor()->SetTransform(m_planeHandle, m_planeTransform);
  215. }
  216. DrawSidebar();
  217. }
  218. void ParallaxMappingExampleComponent::SetCameraConfiguration()
  219. {
  220. Camera::CameraRequestBus::Event(
  221. GetCameraEntityId(),
  222. &Camera::CameraRequestBus::Events::SetFarClipDistance,
  223. 20.f);
  224. Camera::CameraRequestBus::Event(
  225. GetCameraEntityId(),
  226. &Camera::CameraRequestBus::Events::SetFovRadians,
  227. AZ::Constants::QuarterPi);
  228. }
  229. void ParallaxMappingExampleComponent::SaveCameraConfiguration()
  230. {
  231. Camera::CameraRequestBus::EventResult(
  232. m_originalFarClipDistance,
  233. GetCameraEntityId(),
  234. &Camera::CameraRequestBus::Events::GetFarClipDistance);
  235. Camera::CameraRequestBus::EventResult(
  236. m_originalCameraFovRadians,
  237. GetCameraEntityId(),
  238. &Camera::CameraRequestBus::Events::GetFovRadians);
  239. }
  240. void ParallaxMappingExampleComponent::RestoreCameraConfiguration()
  241. {
  242. Camera::CameraRequestBus::Event(
  243. GetCameraEntityId(),
  244. &Camera::CameraRequestBus::Events::SetFarClipDistance,
  245. m_originalFarClipDistance);
  246. Camera::CameraRequestBus::Event(
  247. GetCameraEntityId(),
  248. &Camera::CameraRequestBus::Events::SetFovRadians,
  249. m_originalCameraFovRadians);
  250. }
  251. void ParallaxMappingExampleComponent::DrawSidebar()
  252. {
  253. if (m_imguiSidebar.Begin())
  254. {
  255. bool parallaxSettingChanged = false;
  256. bool planeUVChanged = false;
  257. ImGui::Spacing();
  258. {
  259. ScriptableImGui::ScopedNameContext context{ "Lighting" };
  260. ImGui::Text("Lighting");
  261. ImGui::Indent();
  262. {
  263. ScriptableImGui::RadioButton("Directional Light", &m_lightType, 0);
  264. ScriptableImGui::RadioButton("Spot Light", &m_lightType, 1);
  265. ScriptableImGui::Checkbox("Auto Rotation", &m_lightAutoRotate);
  266. ScriptableImGui::SliderAngle("Direction", &m_lightRotationAngle, 0, 360);
  267. }
  268. ImGui::Unindent();
  269. }
  270. ImGui::Separator();
  271. {
  272. ScriptableImGui::ScopedNameContext context{ "Parallax Setting" };
  273. ImGui::Text("Parallax Setting");
  274. ImGui::Indent();
  275. {
  276. if (ScriptableImGui::Checkbox("Enable Parallax", &m_parallaxEnable))
  277. {
  278. parallaxSettingChanged = true;
  279. m_parallaxMaterial->SetPropertyValue(m_parallaxEnableIndex, m_parallaxEnable);
  280. }
  281. if (m_parallaxEnable)
  282. {
  283. if (ScriptableImGui::Checkbox("Enable Pdo", &m_pdoEnable))
  284. {
  285. parallaxSettingChanged = true;
  286. m_parallaxMaterial->SetPropertyValue(m_pdoEnableIndex, m_pdoEnable);
  287. }
  288. if (ScriptableImGui::SliderFloat("Heightmap Scale", &m_parallaxFactor, 0.0f, 0.1f))
  289. {
  290. parallaxSettingChanged = true;
  291. m_parallaxMaterial->SetPropertyValue(m_parallaxFactorIndex, m_parallaxFactor);
  292. }
  293. if (ScriptableImGui::SliderFloat("Offset", &m_parallaxOffset, -0.1f, 0.1f))
  294. {
  295. parallaxSettingChanged = true;
  296. m_parallaxMaterial->SetPropertyValue(m_parallaxOffsetIndex, m_parallaxOffset);
  297. }
  298. if (ScriptableImGui::Checkbox("Show Clipping", &m_parallaxShowClipping))
  299. {
  300. parallaxSettingChanged = true;
  301. m_parallaxMaterial->SetPropertyValue(m_parallaxShowClippingIndex, m_parallaxShowClipping);
  302. }
  303. if (ScriptableImGui::Combo("Algorithm", &m_parallaxAlgorithm, ParallaxAlgorithmList, AZ_ARRAY_SIZE(ParallaxAlgorithmList)))
  304. {
  305. parallaxSettingChanged = true;
  306. m_parallaxMaterial->SetPropertyValue(m_parallaxAlgorithmIndex, static_cast<uint32_t>(m_parallaxAlgorithm));
  307. }
  308. if (ScriptableImGui::Combo("Quality", &m_parallaxQuality, ParallaxQualityList, AZ_ARRAY_SIZE(ParallaxQualityList)))
  309. {
  310. parallaxSettingChanged = true;
  311. m_parallaxMaterial->SetPropertyValue(m_parallaxQualityIndex, static_cast<uint32_t>(m_parallaxQuality));
  312. }
  313. if (ScriptableImGui::Combo("UV", &m_parallaxUv, ParallaxUvSetList, AZ_ARRAY_SIZE(ParallaxUvSetList)))
  314. {
  315. parallaxSettingChanged = true;
  316. m_parallaxMaterial->SetPropertyValue(m_parallaxUvIndex, static_cast<uint32_t>(m_parallaxUv));
  317. m_parallaxMaterial->SetPropertyValue(m_ambientOcclusionUvIndex, static_cast<uint32_t>(m_parallaxUv));
  318. m_parallaxMaterial->SetPropertyValue(m_baseColorUvIndex, static_cast<uint32_t>(m_parallaxUv));
  319. m_parallaxMaterial->SetPropertyValue(m_normalUvIndex, static_cast<uint32_t>(m_parallaxUv));
  320. m_parallaxMaterial->SetPropertyValue(m_roughnessUvIndex, static_cast<uint32_t>(m_parallaxUv));
  321. }
  322. }
  323. }
  324. ImGui::Unindent();
  325. }
  326. ImGui::Separator();
  327. {
  328. ScriptableImGui::ScopedNameContext context{ "Plane Setting" };
  329. ImGui::Text("Plane Setting");
  330. ImGui::Indent();
  331. {
  332. ScriptableImGui::SliderAngle("Rotation", &m_planeRotationAngle, 0, 360);
  333. bool centerUChanged = false;
  334. bool centerVChanged = false;
  335. bool tileUChanged = false;
  336. bool tileVChanged = false;
  337. bool offsetUChanged = false;
  338. bool offsetVChanged = false;
  339. bool rotationUVChanged = false;
  340. bool scaleChanged = false;
  341. centerUChanged = ScriptableImGui::SliderFloat("Center U", &m_planeCenterU, -1.f, 1.f);
  342. centerVChanged = ScriptableImGui::SliderFloat("Center V", &m_planeCenterV, -1.f, 1.f);
  343. if (centerUChanged || centerVChanged)
  344. {
  345. m_parallaxMaterial->SetPropertyValue(m_centerUVIndex, AZ::Vector2(m_planeCenterU, m_planeCenterV));
  346. }
  347. tileUChanged = ScriptableImGui::SliderFloat("Tile U", &m_planeTileU, 0.f, 2.f);
  348. if (tileUChanged)
  349. {
  350. m_parallaxMaterial->SetPropertyValue(m_tileUIndex, m_planeTileU);
  351. }
  352. tileVChanged = ScriptableImGui::SliderFloat("Tile V", &m_planeTileV, 0.f, 2.f);
  353. if (tileVChanged)
  354. {
  355. m_parallaxMaterial->SetPropertyValue(m_tileVIndex, m_planeTileV);
  356. }
  357. offsetUChanged = ScriptableImGui::SliderFloat("Offset U", &m_planeOffsetU, -1.f, 1.f);
  358. if (offsetUChanged)
  359. {
  360. m_parallaxMaterial->SetPropertyValue(m_offsetUIndex, m_planeOffsetU);
  361. }
  362. offsetVChanged = ScriptableImGui::SliderFloat("Offset V", &m_planeOffsetV, -1.f, 1.f);
  363. if (offsetVChanged)
  364. {
  365. m_parallaxMaterial->SetPropertyValue(m_offsetVIndex, m_planeOffsetV);
  366. }
  367. rotationUVChanged = ScriptableImGui::SliderFloat("Rotation UV", &m_planeRotateUV, -180.f, 180.f);
  368. if (rotationUVChanged)
  369. {
  370. m_parallaxMaterial->SetPropertyValue(m_rotationUVIndex, m_planeRotateUV);
  371. }
  372. scaleChanged = ScriptableImGui::SliderFloat("Scale UV", &m_planeScaleUV, 0.f, 2.f);
  373. if (scaleChanged)
  374. {
  375. m_parallaxMaterial->SetPropertyValue(m_scaleUVIndex, m_planeScaleUV);
  376. }
  377. planeUVChanged = centerUChanged || centerVChanged || tileUChanged || tileVChanged || offsetUChanged || offsetVChanged || rotationUVChanged || scaleChanged;
  378. }
  379. ImGui::Unindent();
  380. }
  381. m_imguiSidebar.End();
  382. if (parallaxSettingChanged || planeUVChanged)
  383. {
  384. m_parallaxMaterial->Compile();
  385. }
  386. }
  387. }
  388. }