ParallaxMappingExampleComponent.cpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  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. using Lux = AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Lux>;
  174. using Candela = AZ::Render::PhotometricColor<AZ::Render::PhotometricUnit::Candela>;
  175. Lux directionalLightColor{AZ::Color::CreateZero()};
  176. Candela diskLightColor{AZ::Color::CreateZero()};
  177. switch (static_cast<LightSelection>(m_lightType))
  178. {
  179. case LightSelection::Directional:
  180. directionalLightColor = Lux{AZ::Color::CreateOne() * 5.f};
  181. break;
  182. case LightSelection::Spot:
  183. diskLightColor = Candela{AZ::Color::CreateOne() * 500.f};
  184. break;
  185. case LightSelection::None:
  186. // Keep initial 0 values
  187. break;
  188. default:
  189. AZ_Assert(false, "Unhandled case");
  190. break;
  191. }
  192. m_diskLightFeatureProcessor->SetRgbIntensity(m_diskLightHandle, diskLightColor);
  193. m_diskLightFeatureProcessor->SetPosition(m_diskLightHandle, location);
  194. m_diskLightFeatureProcessor->SetDirection(m_diskLightHandle, transform.GetBasis(1));
  195. m_directionalLightFeatureProcessor->SetRgbIntensity(m_directionalLightHandle, directionalLightColor);
  196. m_directionalLightFeatureProcessor->SetDirection(m_directionalLightHandle, transform.GetBasis(1));
  197. // Camera Configuration
  198. {
  199. Camera::Configuration config;
  200. Camera::CameraRequestBus::EventResult(
  201. config,
  202. GetCameraEntityId(),
  203. &Camera::CameraRequestBus::Events::GetCameraConfiguration);
  204. m_directionalLightFeatureProcessor->SetCameraConfiguration(
  205. m_directionalLightHandle,
  206. config);
  207. }
  208. // Camera Transform
  209. {
  210. transform = AZ::Transform::CreateIdentity();
  211. AZ::TransformBus::EventResult(
  212. transform,
  213. GetCameraEntityId(),
  214. &AZ::TransformBus::Events::GetWorldTM);
  215. m_directionalLightFeatureProcessor->SetCameraTransform(
  216. m_directionalLightHandle, transform);
  217. }
  218. // Plane Transform
  219. {
  220. m_planeTransform.SetRotation(AZ::Quaternion::CreateRotationZ(m_planeRotationAngle));
  221. GetMeshFeatureProcessor()->SetTransform(m_planeHandle, m_planeTransform);
  222. }
  223. DrawSidebar();
  224. }
  225. void ParallaxMappingExampleComponent::SetCameraConfiguration()
  226. {
  227. Camera::CameraRequestBus::Event(
  228. GetCameraEntityId(),
  229. &Camera::CameraRequestBus::Events::SetFarClipDistance,
  230. 20.f);
  231. Camera::CameraRequestBus::Event(
  232. GetCameraEntityId(),
  233. &Camera::CameraRequestBus::Events::SetFovRadians,
  234. AZ::Constants::QuarterPi);
  235. }
  236. void ParallaxMappingExampleComponent::SaveCameraConfiguration()
  237. {
  238. Camera::CameraRequestBus::EventResult(
  239. m_originalFarClipDistance,
  240. GetCameraEntityId(),
  241. &Camera::CameraRequestBus::Events::GetFarClipDistance);
  242. Camera::CameraRequestBus::EventResult(
  243. m_originalCameraFovRadians,
  244. GetCameraEntityId(),
  245. &Camera::CameraRequestBus::Events::GetFovRadians);
  246. }
  247. void ParallaxMappingExampleComponent::RestoreCameraConfiguration()
  248. {
  249. Camera::CameraRequestBus::Event(
  250. GetCameraEntityId(),
  251. &Camera::CameraRequestBus::Events::SetFarClipDistance,
  252. m_originalFarClipDistance);
  253. Camera::CameraRequestBus::Event(
  254. GetCameraEntityId(),
  255. &Camera::CameraRequestBus::Events::SetFovRadians,
  256. m_originalCameraFovRadians);
  257. }
  258. void ParallaxMappingExampleComponent::DrawSidebar()
  259. {
  260. if (m_imguiSidebar.Begin())
  261. {
  262. bool parallaxSettingChanged = false;
  263. bool planeUVChanged = false;
  264. ImGui::Spacing();
  265. {
  266. ScriptableImGui::ScopedNameContext context{ "Lighting" };
  267. ImGui::Text("Lighting");
  268. ImGui::Indent();
  269. {
  270. ScriptableImGui::RadioButton("No Light", &m_lightType, 0);
  271. ScriptableImGui::RadioButton("Directional Light", &m_lightType, 1);
  272. ScriptableImGui::RadioButton("Spot Light", &m_lightType, 2);
  273. ScriptableImGui::Checkbox("Auto Rotation", &m_lightAutoRotate);
  274. ScriptableImGui::SliderAngle("Direction", &m_lightRotationAngle, 0, 360);
  275. }
  276. ImGui::Unindent();
  277. }
  278. ImGui::Separator();
  279. {
  280. ScriptableImGui::ScopedNameContext context{ "Parallax Setting" };
  281. ImGui::Text("Parallax Setting");
  282. ImGui::Indent();
  283. {
  284. if (ScriptableImGui::Checkbox("Enable Parallax", &m_parallaxEnable))
  285. {
  286. parallaxSettingChanged = true;
  287. m_parallaxMaterial->SetPropertyValue(m_parallaxEnableIndex, m_parallaxEnable);
  288. }
  289. if (m_parallaxEnable)
  290. {
  291. if (ScriptableImGui::Checkbox("Enable Pdo", &m_pdoEnable))
  292. {
  293. parallaxSettingChanged = true;
  294. m_parallaxMaterial->SetPropertyValue(m_pdoEnableIndex, m_pdoEnable);
  295. }
  296. if (ScriptableImGui::SliderFloat("Heightmap Scale", &m_parallaxFactor, 0.0f, 0.1f))
  297. {
  298. parallaxSettingChanged = true;
  299. m_parallaxMaterial->SetPropertyValue(m_parallaxFactorIndex, m_parallaxFactor);
  300. }
  301. if (ScriptableImGui::SliderFloat("Offset", &m_parallaxOffset, -0.1f, 0.1f))
  302. {
  303. parallaxSettingChanged = true;
  304. m_parallaxMaterial->SetPropertyValue(m_parallaxOffsetIndex, m_parallaxOffset);
  305. }
  306. if (ScriptableImGui::Checkbox("Show Clipping", &m_parallaxShowClipping))
  307. {
  308. parallaxSettingChanged = true;
  309. m_parallaxMaterial->SetPropertyValue(m_parallaxShowClippingIndex, m_parallaxShowClipping);
  310. }
  311. if (ScriptableImGui::Combo("Algorithm", &m_parallaxAlgorithm, ParallaxAlgorithmList, AZ_ARRAY_SIZE(ParallaxAlgorithmList)))
  312. {
  313. parallaxSettingChanged = true;
  314. m_parallaxMaterial->SetPropertyValue(m_parallaxAlgorithmIndex, static_cast<uint32_t>(m_parallaxAlgorithm));
  315. }
  316. if (ScriptableImGui::Combo("Quality", &m_parallaxQuality, ParallaxQualityList, AZ_ARRAY_SIZE(ParallaxQualityList)))
  317. {
  318. parallaxSettingChanged = true;
  319. m_parallaxMaterial->SetPropertyValue(m_parallaxQualityIndex, static_cast<uint32_t>(m_parallaxQuality));
  320. }
  321. if (ScriptableImGui::Combo("UV", &m_parallaxUv, ParallaxUvSetList, AZ_ARRAY_SIZE(ParallaxUvSetList)))
  322. {
  323. parallaxSettingChanged = true;
  324. m_parallaxMaterial->SetPropertyValue(m_parallaxUvIndex, static_cast<uint32_t>(m_parallaxUv));
  325. m_parallaxMaterial->SetPropertyValue(m_ambientOcclusionUvIndex, static_cast<uint32_t>(m_parallaxUv));
  326. m_parallaxMaterial->SetPropertyValue(m_baseColorUvIndex, static_cast<uint32_t>(m_parallaxUv));
  327. m_parallaxMaterial->SetPropertyValue(m_normalUvIndex, static_cast<uint32_t>(m_parallaxUv));
  328. m_parallaxMaterial->SetPropertyValue(m_roughnessUvIndex, static_cast<uint32_t>(m_parallaxUv));
  329. }
  330. }
  331. }
  332. ImGui::Unindent();
  333. }
  334. ImGui::Separator();
  335. {
  336. ScriptableImGui::ScopedNameContext context{ "Plane Setting" };
  337. ImGui::Text("Plane Setting");
  338. ImGui::Indent();
  339. {
  340. ScriptableImGui::SliderAngle("Rotation", &m_planeRotationAngle, 0, 360);
  341. bool centerUChanged = false;
  342. bool centerVChanged = false;
  343. bool tileUChanged = false;
  344. bool tileVChanged = false;
  345. bool offsetUChanged = false;
  346. bool offsetVChanged = false;
  347. bool rotationUVChanged = false;
  348. bool scaleChanged = false;
  349. centerUChanged = ScriptableImGui::SliderFloat("Center U", &m_planeCenterU, -1.f, 1.f);
  350. centerVChanged = ScriptableImGui::SliderFloat("Center V", &m_planeCenterV, -1.f, 1.f);
  351. if (centerUChanged || centerVChanged)
  352. {
  353. m_parallaxMaterial->SetPropertyValue(m_centerUVIndex, AZ::Vector2(m_planeCenterU, m_planeCenterV));
  354. }
  355. tileUChanged = ScriptableImGui::SliderFloat("Tile U", &m_planeTileU, 0.f, 2.f);
  356. if (tileUChanged)
  357. {
  358. m_parallaxMaterial->SetPropertyValue(m_tileUIndex, m_planeTileU);
  359. }
  360. tileVChanged = ScriptableImGui::SliderFloat("Tile V", &m_planeTileV, 0.f, 2.f);
  361. if (tileVChanged)
  362. {
  363. m_parallaxMaterial->SetPropertyValue(m_tileVIndex, m_planeTileV);
  364. }
  365. offsetUChanged = ScriptableImGui::SliderFloat("Offset U", &m_planeOffsetU, -1.f, 1.f);
  366. if (offsetUChanged)
  367. {
  368. m_parallaxMaterial->SetPropertyValue(m_offsetUIndex, m_planeOffsetU);
  369. }
  370. offsetVChanged = ScriptableImGui::SliderFloat("Offset V", &m_planeOffsetV, -1.f, 1.f);
  371. if (offsetVChanged)
  372. {
  373. m_parallaxMaterial->SetPropertyValue(m_offsetVIndex, m_planeOffsetV);
  374. }
  375. rotationUVChanged = ScriptableImGui::SliderFloat("Rotation UV", &m_planeRotateUV, -180.f, 180.f);
  376. if (rotationUVChanged)
  377. {
  378. m_parallaxMaterial->SetPropertyValue(m_rotationUVIndex, m_planeRotateUV);
  379. }
  380. scaleChanged = ScriptableImGui::SliderFloat("Scale UV", &m_planeScaleUV, 0.f, 2.f);
  381. if (scaleChanged)
  382. {
  383. m_parallaxMaterial->SetPropertyValue(m_scaleUVIndex, m_planeScaleUV);
  384. }
  385. planeUVChanged = centerUChanged || centerVChanged || tileUChanged || tileVChanged || offsetUChanged || offsetVChanged || rotationUVChanged || scaleChanged;
  386. }
  387. ImGui::Unindent();
  388. }
  389. m_imguiSidebar.End();
  390. if (parallaxSettingChanged || planeUVChanged)
  391. {
  392. m_parallaxMaterial->Compile();
  393. }
  394. }
  395. }
  396. }