123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293 |
- /*
- * Copyright (c) Contributors to the Open 3D Engine Project.
- * For complete copyright and license terms please see the LICENSE at the root of this distribution.
- *
- * SPDX-License-Identifier: Apache-2.0 OR MIT
- *
- */
- #include "SurfaceDataMeshComponent.h"
- #include <AzCore/Debug/Profiler.h>
- #include <AzCore/Serialization/EditContext.h>
- #include <AzCore/Serialization/SerializeContext.h>
- #include <Atom/RPI.Reflect/Model/ModelAssetCreator.h>
- #include <SurfaceData/SurfaceDataSystemRequestBus.h>
- #include <SurfaceData/Utility/SurfaceDataUtility.h>
- namespace SurfaceData
- {
- void SurfaceDataMeshConfig::Reflect(AZ::ReflectContext* context)
- {
- AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
- if (serialize)
- {
- serialize->Class<SurfaceDataMeshConfig, AZ::ComponentConfig>()
- ->Version(0)
- ->Field("SurfaceTags", &SurfaceDataMeshConfig::m_tags)
- ;
- AZ::EditContext* edit = serialize->GetEditContext();
- if (edit)
- {
- edit->Class<SurfaceDataMeshConfig>(
- "Mesh Surface Tag Emitter", "")
- ->ClassElement(AZ::Edit::ClassElements::EditorData, "")
- ->Attribute(AZ::Edit::Attributes::Visibility, AZ::Edit::PropertyVisibility::ShowChildrenOnly)
- ->Attribute(AZ::Edit::Attributes::AutoExpand, true)
- ->DataElement(0, &SurfaceDataMeshConfig::m_tags, "Generated Tags", "")
- ;
- }
- }
- }
- void SurfaceDataMeshComponent::GetProvidedServices(AZ::ComponentDescriptor::DependencyArrayType& services)
- {
- services.push_back(AZ_CRC_CE("SurfaceDataProviderService"));
- }
- void SurfaceDataMeshComponent::GetIncompatibleServices(AZ::ComponentDescriptor::DependencyArrayType& services)
- {
- services.push_back(AZ_CRC_CE("SurfaceDataProviderService"));
- }
- void SurfaceDataMeshComponent::GetRequiredServices(AZ::ComponentDescriptor::DependencyArrayType& services)
- {
- services.push_back(AZ_CRC_CE("MeshService"));
- }
- void SurfaceDataMeshComponent::Reflect(AZ::ReflectContext* context)
- {
- SurfaceDataMeshConfig::Reflect(context);
- AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
- if (serialize)
- {
- serialize->Class<SurfaceDataMeshComponent, AZ::Component>()
- ->Version(0)
- ->Field("Configuration", &SurfaceDataMeshComponent::m_configuration)
- ;
- }
- }
- SurfaceDataMeshComponent::SurfaceDataMeshComponent()
- : m_nonUniformScaleChangedHandler([this]([[maybe_unused]] const AZ::Vector3& scale) { this->OnCompositionChanged(); })
- {
- }
- SurfaceDataMeshComponent::SurfaceDataMeshComponent(const SurfaceDataMeshConfig& configuration)
- : m_configuration(configuration)
- , m_nonUniformScaleChangedHandler([this]([[maybe_unused]] const AZ::Vector3& scale) { this->OnCompositionChanged(); })
- {
- }
- void SurfaceDataMeshComponent::Activate()
- {
- AZ::TransformNotificationBus::Handler::BusConnect(GetEntityId());
- AZ::Render::MeshComponentNotificationBus::Handler::BusConnect(GetEntityId());
- AZ::NonUniformScaleRequestBus::Event(
- GetEntityId(), &AZ::NonUniformScaleRequests::RegisterScaleChangedEvent, m_nonUniformScaleChangedHandler);
- m_providerHandle = InvalidSurfaceDataRegistryHandle;
- m_refresh = false;
- // Update the cached mesh data and bounds, then register the surface data provider
- m_newPointWeights.AssignSurfaceTagWeights(m_configuration.m_tags, 1.0f);
- UpdateMeshData();
- }
- void SurfaceDataMeshComponent::Deactivate()
- {
- if (m_providerHandle != InvalidSurfaceDataRegistryHandle)
- {
- AZ::Interface<SurfaceData::SurfaceDataSystem>::Get()->UnregisterSurfaceDataProvider(m_providerHandle);
- m_providerHandle = InvalidSurfaceDataRegistryHandle;
- }
- m_nonUniformScaleChangedHandler.Disconnect();
- SurfaceDataProviderRequestBus::Handler::BusDisconnect();
- AZ::TickBus::Handler::BusDisconnect();
- AZ::TransformNotificationBus::Handler::BusDisconnect();
- AZ::Render::MeshComponentNotificationBus::Handler::BusDisconnect();
- m_refresh = false;
- // Clear the cached mesh data
- {
- AZStd::unique_lock<decltype(m_cacheMutex)> lock(m_cacheMutex);
- m_meshAssetData = {};
- m_meshBounds = AZ::Aabb::CreateNull();
- m_meshWorldTM = AZ::Transform::CreateIdentity();
- m_meshWorldTMInverse = AZ::Transform::CreateIdentity();
- }
- }
- bool SurfaceDataMeshComponent::ReadInConfig(const AZ::ComponentConfig* baseConfig)
- {
- if (auto config = azrtti_cast<const SurfaceDataMeshConfig*>(baseConfig))
- {
- m_configuration = *config;
- return true;
- }
- return false;
- }
- bool SurfaceDataMeshComponent::WriteOutConfig(AZ::ComponentConfig* outBaseConfig) const
- {
- if (auto config = azrtti_cast<SurfaceDataMeshConfig*>(outBaseConfig))
- {
- *config = m_configuration;
- return true;
- }
- return false;
- }
- void SurfaceDataMeshComponent::GetSurfacePoints(const AZ::Vector3& inPosition, SurfacePointList& surfacePointList) const
- {
- GetSurfacePointsFromList(AZStd::span<const AZ::Vector3>(&inPosition, 1), surfacePointList);
- }
- void SurfaceDataMeshComponent::GetSurfacePointsFromList(
- AZStd::span<const AZ::Vector3> inPositions, SurfacePointList& surfacePointList) const
- {
- AZ::Vector3 hitPosition;
- AZ::Vector3 hitNormal;
- AZStd::shared_lock<decltype(m_cacheMutex)> lock(m_cacheMutex);
- AZ::RPI::ModelAsset* mesh = m_meshAssetData.GetAs<AZ::RPI::ModelAsset>();
- if (!mesh)
- {
- return;
- }
- for (auto& inPosition : inPositions)
- {
- // test AABB as first pass to claim the point
- if (SurfaceData::AabbContains2D(m_meshBounds, inPosition))
- {
- const AZ::Vector3 rayStart =
- AZ::Vector3(inPosition.GetX(), inPosition.GetY(), m_meshBounds.GetMax().GetZ() + s_rayAABBHeightPadding);
- const AZ::Vector3 rayEnd =
- AZ::Vector3(inPosition.GetX(), inPosition.GetY(), m_meshBounds.GetMin().GetZ() - s_rayAABBHeightPadding);
- bool rayHit = GetMeshRayIntersection(
- *mesh, m_meshWorldTM, m_meshWorldTMInverse, m_meshNonUniformScale, rayStart, rayEnd, hitPosition, hitNormal);
- if (rayHit)
- {
- surfacePointList.AddSurfacePoint(GetEntityId(), inPosition, hitPosition, hitNormal, m_newPointWeights);
- }
- }
- }
- }
- AZ::Aabb SurfaceDataMeshComponent::GetSurfaceAabb() const
- {
- return m_meshBounds;
- }
- SurfaceTagVector SurfaceDataMeshComponent::GetSurfaceTags() const
- {
- return m_configuration.m_tags;
- }
- void SurfaceDataMeshComponent::OnCompositionChanged()
- {
- if (!m_refresh)
- {
- m_refresh = true;
- AZ::TickBus::Handler::BusConnect();
- }
- }
- void SurfaceDataMeshComponent::OnModelReady([[maybe_unused]] const AZ::Data::Asset<AZ::RPI::ModelAsset>& modelAsset, [[maybe_unused]] const AZ::Data::Instance<AZ::RPI::Model>& model)
- {
- OnCompositionChanged();
- }
- void SurfaceDataMeshComponent::OnTransformChanged(const AZ::Transform & local, const AZ::Transform & world)
- {
- (void)local;
- (void)world;
- OnCompositionChanged();
- }
- void SurfaceDataMeshComponent::OnTick(float /*deltaTime*/, AZ::ScriptTimePoint /*time*/)
- {
- if (m_refresh)
- {
- UpdateMeshData();
- m_refresh = false;
- }
- AZ::TickBus::Handler::BusDisconnect();
- }
- void SurfaceDataMeshComponent::UpdateMeshData()
- {
- AZ_PROFILE_SCOPE(Entity, "SurfaceDataMeshComponent: UpdateMeshData");
- bool meshValidBeforeUpdate = false;
- bool meshValidAfterUpdate = false;
- {
- AZStd::unique_lock<decltype(m_cacheMutex)> lock(m_cacheMutex);
- meshValidBeforeUpdate = (m_meshAssetData.GetAs<AZ::RPI::ModelAsset>() != nullptr) && (m_meshBounds.IsValid());
- m_meshAssetData = {};
- AZ::Render::MeshComponentRequestBus::EventResult(m_meshAssetData, GetEntityId(), &AZ::Render::MeshComponentRequestBus::Events::GetModelAsset);
- m_meshBounds = AZ::Aabb::CreateNull();
- AZ::Render::MeshComponentRequestBus::EventResult(m_meshBounds, GetEntityId(), &AZ::Render::MeshComponentRequestBus::Events::GetWorldBounds);
- m_meshWorldTM = AZ::Transform::CreateIdentity();
- AZ::TransformBus::EventResult(m_meshWorldTM, GetEntityId(), &AZ::TransformBus::Events::GetWorldTM);
- m_meshWorldTMInverse = m_meshWorldTM.GetInverse();
- m_meshNonUniformScale = AZ::Vector3::CreateOne();
- AZ::NonUniformScaleRequestBus::EventResult(m_meshNonUniformScale, GetEntityId(), &AZ::NonUniformScaleRequests::GetScale);
- meshValidAfterUpdate = (m_meshAssetData.GetAs<AZ::RPI::ModelAsset>() != nullptr) && (m_meshBounds.IsValid());
- }
- SurfaceDataRegistryEntry registryEntry;
- registryEntry.m_entityId = GetEntityId();
- registryEntry.m_bounds = GetSurfaceAabb();
- registryEntry.m_tags = GetSurfaceTags();
- registryEntry.m_maxPointsCreatedPerInput = 1;
- if (!meshValidBeforeUpdate && !meshValidAfterUpdate)
- {
- // We didn't have a valid mesh asset before or after running this, so do nothing.
- }
- else if (!meshValidBeforeUpdate && meshValidAfterUpdate)
- {
- // Our mesh has become valid, so register as a provider and save off the provider handle
- AZ_Assert((m_providerHandle == InvalidSurfaceDataRegistryHandle), "Surface data handle is initialized before our mesh became active");
- AZ_Assert(m_meshBounds.IsValid(), "Mesh Geometry isn't correctly initialized.");
- m_providerHandle = AZ::Interface<SurfaceData::SurfaceDataSystem>::Get()->RegisterSurfaceDataProvider(registryEntry);
- // Start listening for surface data events
- AZ_Assert((m_providerHandle != InvalidSurfaceDataRegistryHandle), "Invalid surface data handle");
- SurfaceDataProviderRequestBus::Handler::BusConnect(m_providerHandle);
- }
- else if (meshValidBeforeUpdate && !meshValidAfterUpdate)
- {
- // Our mesh has stopped being valid, so unregister and stop listening for surface data events
- AZ_Assert((m_providerHandle != InvalidSurfaceDataRegistryHandle), "Invalid surface data handle");
- AZ::Interface<SurfaceData::SurfaceDataSystem>::Get()->UnregisterSurfaceDataProvider(m_providerHandle);
- m_providerHandle = InvalidSurfaceDataRegistryHandle;
- SurfaceDataProviderRequestBus::Handler::BusDisconnect();
- }
- else if (meshValidBeforeUpdate && meshValidAfterUpdate)
- {
- // Our mesh was valid before and after, it just changed in some way, so update our registry entry.
- AZ_Assert((m_providerHandle != InvalidSurfaceDataRegistryHandle), "Invalid surface data handle");
- AZ::Interface<SurfaceData::SurfaceDataSystem>::Get()->UpdateSurfaceDataProvider(m_providerHandle, registryEntry);
- }
- }
- }
|