You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
350 lines
15 KiB
C++
350 lines
15 KiB
C++
/*
|
|
* 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 <GradientSignal/Components/SurfaceAltitudeGradientComponent.h>
|
|
#include <AzCore/Component/Entity.h>
|
|
#include <AzCore/RTTI/BehaviorContext.h>
|
|
#include <AzCore/Serialization/EditContext.h>
|
|
#include <AzCore/Serialization/SerializeContext.h>
|
|
#include <LmbrCentral/Shape/ShapeComponentBus.h>
|
|
#include <SurfaceData/SurfaceDataSystemRequestBus.h>
|
|
#include <GradientSignal/Util.h>
|
|
#include <LmbrCentral/Dependency/DependencyMonitor.h>
|
|
|
|
namespace GradientSignal
|
|
{
|
|
void SurfaceAltitudeGradientConfig::Reflect(AZ::ReflectContext* context)
|
|
{
|
|
AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
|
|
if (serialize)
|
|
{
|
|
serialize->Class<SurfaceAltitudeGradientConfig, AZ::ComponentConfig>()
|
|
->Version(0)
|
|
->Field("ShapeEntityId", &SurfaceAltitudeGradientConfig::m_shapeEntityId)
|
|
->Field("AltitudeMin", &SurfaceAltitudeGradientConfig::m_altitudeMin)
|
|
->Field("AltitudeMax", &SurfaceAltitudeGradientConfig::m_altitudeMax)
|
|
->Field("SurfaceTagsToSample", &SurfaceAltitudeGradientConfig::m_surfaceTagsToSample)
|
|
;
|
|
|
|
AZ::EditContext* edit = serialize->GetEditContext();
|
|
if (edit)
|
|
{
|
|
edit->Class<SurfaceAltitudeGradientConfig>(
|
|
"Altitude Gradient", "altitude Gradient")
|
|
->ClassElement(AZ::Edit::ClassElements::EditorData, "")
|
|
->Attribute(AZ::Edit::Attributes::Visibility, AZ::Edit::PropertyVisibility::ShowChildrenOnly)
|
|
->Attribute(AZ::Edit::Attributes::AutoExpand, true)
|
|
->DataElement(0, &SurfaceAltitudeGradientConfig::m_shapeEntityId, "Pin To Shape Entity Id", "Shape bounds override min/max altitude if specified.")
|
|
->Attribute(AZ::Edit::Attributes::RequiredService, AZ_CRC("ShapeService", 0xe86aa5fe))
|
|
->DataElement(0, &SurfaceAltitudeGradientConfig::m_altitudeMin, "Altitude Min", "Minimum acceptable surface altitude.")
|
|
->Attribute(AZ::Edit::Attributes::ReadOnly, &SurfaceAltitudeGradientConfig::IsShapeValid)
|
|
->DataElement(0, &SurfaceAltitudeGradientConfig::m_altitudeMax, "Altitude Max", "Maximum acceptable surface altitude.")
|
|
->Attribute(AZ::Edit::Attributes::ReadOnly, &SurfaceAltitudeGradientConfig::IsShapeValid)
|
|
->DataElement(0, &SurfaceAltitudeGradientConfig::m_surfaceTagsToSample, "Surface Tags to track", "")
|
|
;
|
|
}
|
|
}
|
|
|
|
if (auto behaviorContext = azrtti_cast<AZ::BehaviorContext*>(context))
|
|
{
|
|
behaviorContext->Class<SurfaceAltitudeGradientConfig>()
|
|
->Attribute(AZ::Script::Attributes::Category, "Vegetation")
|
|
->Constructor()
|
|
->Property("shapeEntityId", BehaviorValueProperty(&SurfaceAltitudeGradientConfig::m_shapeEntityId))
|
|
->Property("altitudeMin", BehaviorValueProperty(&SurfaceAltitudeGradientConfig::m_altitudeMin))
|
|
->Property("altitudeMax", BehaviorValueProperty(&SurfaceAltitudeGradientConfig::m_altitudeMax))
|
|
->Method("GetNumTags", &SurfaceAltitudeGradientConfig::GetNumTags)
|
|
->Method("GetTag", &SurfaceAltitudeGradientConfig::GetTag)
|
|
->Method("RemoveTag", &SurfaceAltitudeGradientConfig::RemoveTag)
|
|
->Method("AddTag", &SurfaceAltitudeGradientConfig::AddTag)
|
|
;
|
|
}
|
|
}
|
|
|
|
bool SurfaceAltitudeGradientConfig::IsShapeValid() const
|
|
{
|
|
return m_shapeEntityId.IsValid();
|
|
}
|
|
|
|
size_t SurfaceAltitudeGradientConfig::GetNumTags() const
|
|
{
|
|
return m_surfaceTagsToSample.size();
|
|
}
|
|
|
|
AZ::Crc32 SurfaceAltitudeGradientConfig::GetTag(int tagIndex) const
|
|
{
|
|
if (tagIndex < m_surfaceTagsToSample.size() && tagIndex >= 0)
|
|
{
|
|
return m_surfaceTagsToSample[tagIndex];
|
|
}
|
|
|
|
return AZ::Crc32();
|
|
}
|
|
|
|
void SurfaceAltitudeGradientConfig::RemoveTag(int tagIndex)
|
|
{
|
|
if (tagIndex < m_surfaceTagsToSample.size() && tagIndex >= 0)
|
|
{
|
|
m_surfaceTagsToSample.erase(m_surfaceTagsToSample.begin() + tagIndex);
|
|
}
|
|
}
|
|
|
|
void SurfaceAltitudeGradientConfig::AddTag(AZStd::string tag)
|
|
{
|
|
m_surfaceTagsToSample.push_back(SurfaceData::SurfaceTag(tag));
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::GetProvidedServices(AZ::ComponentDescriptor::DependencyArrayType& services)
|
|
{
|
|
services.push_back(AZ_CRC("GradientService", 0x21c18d23));
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::GetIncompatibleServices(AZ::ComponentDescriptor::DependencyArrayType& services)
|
|
{
|
|
services.push_back(AZ_CRC("GradientService", 0x21c18d23));
|
|
services.push_back(AZ_CRC("GradientTransformService", 0x8c8c5ecc));
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::GetRequiredServices([[maybe_unused]] AZ::ComponentDescriptor::DependencyArrayType& services)
|
|
{
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::Reflect(AZ::ReflectContext* context)
|
|
{
|
|
SurfaceAltitudeGradientConfig::Reflect(context);
|
|
|
|
AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
|
|
if (serialize)
|
|
{
|
|
serialize->Class<SurfaceAltitudeGradientComponent, AZ::Component>()
|
|
->Version(0)
|
|
->Field("Configuration", &SurfaceAltitudeGradientComponent::m_configuration)
|
|
;
|
|
}
|
|
|
|
if (auto behaviorContext = azrtti_cast<AZ::BehaviorContext*>(context))
|
|
{
|
|
behaviorContext->Constant("SurfaceAltitudeGradientComponentTypeId", BehaviorConstant(SurfaceAltitudeGradientComponentTypeId));
|
|
|
|
behaviorContext->Class<SurfaceAltitudeGradientComponent>()->RequestBus("SurfaceAltitudeGradientRequestBus");
|
|
|
|
behaviorContext->EBus<SurfaceAltitudeGradientRequestBus>("SurfaceAltitudeGradientRequestBus")
|
|
->Attribute(AZ::Script::Attributes::Category, "Vegetation")
|
|
->Event("GetShapeEntityId", &SurfaceAltitudeGradientRequestBus::Events::GetShapeEntityId)
|
|
->Event("SetShapeEntityId", &SurfaceAltitudeGradientRequestBus::Events::SetShapeEntityId)
|
|
->VirtualProperty("ShapeEntityId", "GetShapeEntityId", "SetShapeEntityId")
|
|
->Event("GetAltitudeMin", &SurfaceAltitudeGradientRequestBus::Events::GetAltitudeMin)
|
|
->Event("SetAltitudeMin", &SurfaceAltitudeGradientRequestBus::Events::SetAltitudeMin)
|
|
->VirtualProperty("AltitudeMin", "GetAltitudeMin", "SetAltitudeMin")
|
|
->Event("GetAltitudeMax", &SurfaceAltitudeGradientRequestBus::Events::GetAltitudeMax)
|
|
->Event("SetAltitudeMax", &SurfaceAltitudeGradientRequestBus::Events::SetAltitudeMax)
|
|
->VirtualProperty("AltitudeMax", "GetAltitudeMax", "SetAltitudeMax")
|
|
->Event("GetNumTags", &SurfaceAltitudeGradientRequestBus::Events::GetNumTags)
|
|
->Event("GetTag", &SurfaceAltitudeGradientRequestBus::Events::GetTag)
|
|
->Event("RemoveTag", &SurfaceAltitudeGradientRequestBus::Events::RemoveTag)
|
|
->Event("AddTag", &SurfaceAltitudeGradientRequestBus::Events::AddTag)
|
|
;
|
|
}
|
|
}
|
|
|
|
SurfaceAltitudeGradientComponent::SurfaceAltitudeGradientComponent(const SurfaceAltitudeGradientConfig& configuration)
|
|
: m_configuration(configuration)
|
|
{
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::Activate()
|
|
{
|
|
m_dependencyMonitor.Reset();
|
|
m_dependencyMonitor.ConnectOwner(GetEntityId());
|
|
m_dependencyMonitor.ConnectDependency(m_configuration.m_shapeEntityId);
|
|
LmbrCentral::DependencyNotificationBus::Handler::BusConnect(GetEntityId());
|
|
AZ::TickBus::Handler::BusConnect();
|
|
GradientRequestBus::Handler::BusConnect(GetEntityId());
|
|
SurfaceAltitudeGradientRequestBus::Handler::BusConnect(GetEntityId());
|
|
UpdateFromShape();
|
|
m_dirty = false;
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::Deactivate()
|
|
{
|
|
m_dependencyMonitor.Reset();
|
|
LmbrCentral::DependencyNotificationBus::Handler::BusDisconnect();
|
|
AZ::TickBus::Handler::BusDisconnect();
|
|
GradientRequestBus::Handler::BusDisconnect();
|
|
SurfaceAltitudeGradientRequestBus::Handler::BusDisconnect();
|
|
m_dirty = false;
|
|
}
|
|
|
|
bool SurfaceAltitudeGradientComponent::ReadInConfig(const AZ::ComponentConfig* baseConfig)
|
|
{
|
|
if (auto config = azrtti_cast<const SurfaceAltitudeGradientConfig*>(baseConfig))
|
|
{
|
|
m_configuration = *config;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool SurfaceAltitudeGradientComponent::WriteOutConfig(AZ::ComponentConfig* outBaseConfig) const
|
|
{
|
|
if (auto config = azrtti_cast<SurfaceAltitudeGradientConfig*>(outBaseConfig))
|
|
{
|
|
*config = m_configuration;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
float SurfaceAltitudeGradientComponent::GetValue(const GradientSampleParams& sampleParams) const
|
|
{
|
|
// For GetValue(), we reuse our SurfacePointList for the GetSurfacePoints query to avoid the repeated cost of memory allocation
|
|
// and deallocation across GetValue() calls. However, this also means we can only use it from one thread at a time, so lock a
|
|
// mutex to ensure no other threads call GetValue() at the same time.
|
|
AZStd::unique_lock<decltype(m_surfacePointListMutex)> surfacePointLock(m_surfacePointListMutex);
|
|
|
|
AZStd::shared_lock<decltype(m_cacheMutex)> lock(m_cacheMutex);
|
|
|
|
SurfaceData::SurfaceDataSystemRequestBus::Broadcast(&SurfaceData::SurfaceDataSystemRequestBus::Events::GetSurfacePoints,
|
|
sampleParams.m_position, m_configuration.m_surfaceTagsToSample, m_surfacePointList);
|
|
|
|
constexpr size_t inPositionIndex = 0;
|
|
return CalculateAltitudeRatio(m_surfacePointList, inPositionIndex, m_configuration.m_altitudeMin, m_configuration.m_altitudeMax);
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::GetValues(AZStd::span<const AZ::Vector3> positions, AZStd::span<float> outValues) const
|
|
{
|
|
if (positions.size() != outValues.size())
|
|
{
|
|
AZ_Assert(false, "input and output lists are different sizes (%zu vs %zu).", positions.size(), outValues.size());
|
|
return;
|
|
}
|
|
|
|
AZStd::shared_lock<decltype(m_cacheMutex)> lock(m_cacheMutex);
|
|
|
|
SurfaceData::SurfacePointList points;
|
|
SurfaceData::SurfaceDataSystemRequestBus::Broadcast(
|
|
&SurfaceData::SurfaceDataSystemRequestBus::Events::GetSurfacePointsFromList, positions, m_configuration.m_surfaceTagsToSample,
|
|
points);
|
|
|
|
if (points.IsEmpty())
|
|
{
|
|
// No surface data, so no output values.
|
|
AZStd::fill(outValues.begin(), outValues.end(), 0.0f);
|
|
}
|
|
else
|
|
{
|
|
// For each position, turn the height into a 0-1 value based on our min/max altitudes.
|
|
for (size_t index = 0; index < positions.size(); index++)
|
|
{
|
|
outValues[index] = CalculateAltitudeRatio(points, index, m_configuration.m_altitudeMin, m_configuration.m_altitudeMax);
|
|
}
|
|
}
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::OnCompositionChanged()
|
|
{
|
|
m_dirty = true;
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::OnTick([[maybe_unused]] float deltaTime, [[maybe_unused]] AZ::ScriptTimePoint time)
|
|
{
|
|
if (m_dirty)
|
|
{
|
|
const auto altitudeMinOld = m_configuration.m_altitudeMin;
|
|
const auto altitudeMaxOld = m_configuration.m_altitudeMax;
|
|
|
|
//updating on tick to query shape bus on main thread
|
|
UpdateFromShape();
|
|
|
|
//notify observers if content has changed
|
|
if (altitudeMinOld != m_configuration.m_altitudeMin ||
|
|
altitudeMaxOld != m_configuration.m_altitudeMax)
|
|
{
|
|
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
|
|
}
|
|
m_dirty = false;
|
|
}
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::UpdateFromShape()
|
|
{
|
|
AZ_PROFILE_FUNCTION(Entity);
|
|
|
|
AZStd::unique_lock<decltype(m_cacheMutex)> lock(m_cacheMutex);
|
|
|
|
if (m_configuration.m_shapeEntityId.IsValid())
|
|
{
|
|
AZ::Aabb bounds = AZ::Aabb::CreateFromMinMax(
|
|
AZ::Vector3(-AZ::Constants::FloatMax, -AZ::Constants::FloatMax, AZ::GetMin(m_configuration.m_altitudeMin, m_configuration.m_altitudeMax)),
|
|
AZ::Vector3(AZ::Constants::FloatMax, AZ::Constants::FloatMax, AZ::GetMax(m_configuration.m_altitudeMin, m_configuration.m_altitudeMax)));
|
|
|
|
LmbrCentral::ShapeComponentRequestsBus::EventResult(bounds, m_configuration.m_shapeEntityId, &LmbrCentral::ShapeComponentRequestsBus::Events::GetEncompassingAabb);
|
|
|
|
if (bounds.IsValid())
|
|
{
|
|
m_configuration.m_altitudeMin = bounds.GetMin().GetZ();
|
|
m_configuration.m_altitudeMax = bounds.GetMax().GetZ();
|
|
}
|
|
}
|
|
}
|
|
|
|
AZ::EntityId SurfaceAltitudeGradientComponent::GetShapeEntityId() const
|
|
{
|
|
return m_configuration.m_shapeEntityId;
|
|
}
|
|
void SurfaceAltitudeGradientComponent::SetShapeEntityId(AZ::EntityId entityId)
|
|
{
|
|
m_configuration.m_shapeEntityId = entityId;
|
|
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
|
|
}
|
|
|
|
float SurfaceAltitudeGradientComponent::GetAltitudeMin() const
|
|
{
|
|
return m_configuration.m_altitudeMin;
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::SetAltitudeMin(float altitudeMin)
|
|
{
|
|
m_configuration.m_altitudeMin = altitudeMin;
|
|
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
|
|
}
|
|
|
|
float SurfaceAltitudeGradientComponent::GetAltitudeMax() const
|
|
{
|
|
return m_configuration.m_altitudeMax;
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::SetAltitudeMax(float altitudeMax)
|
|
{
|
|
m_configuration.m_altitudeMax = altitudeMax;
|
|
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
|
|
}
|
|
|
|
size_t SurfaceAltitudeGradientComponent::GetNumTags() const
|
|
{
|
|
return m_configuration.GetNumTags();
|
|
}
|
|
|
|
AZ::Crc32 SurfaceAltitudeGradientComponent::GetTag(int tagIndex) const
|
|
{
|
|
return m_configuration.GetTag(tagIndex);
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::RemoveTag(int tagIndex)
|
|
{
|
|
m_configuration.RemoveTag(tagIndex);
|
|
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
|
|
}
|
|
|
|
void SurfaceAltitudeGradientComponent::AddTag(AZStd::string tag)
|
|
{
|
|
m_configuration.AddTag(tag);
|
|
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
|
|
}
|
|
}
|