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.
o3de/Gems/GradientSignal/Code/Source/Components/ShapeAreaFalloffGradientCom...

244 lines
12 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/ShapeAreaFalloffGradientComponent.h>
#include <AzCore/RTTI/BehaviorContext.h>
#include <AzCore/Serialization/EditContext.h>
#include <AzCore/Serialization/SerializeContext.h>
#include <LmbrCentral/Shape/ShapeComponentBus.h>
#include <AzCore/Debug/Profiler.h>
namespace GradientSignal
{
void ShapeAreaFalloffGradientConfig::Reflect(AZ::ReflectContext* context)
{
AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
if (serialize)
{
serialize->Class<ShapeAreaFalloffGradientConfig, AZ::ComponentConfig>()
->Version(0)
->Field("ShapeEntityId", &ShapeAreaFalloffGradientConfig::m_shapeEntityId)
->Field("FalloffWidth", &ShapeAreaFalloffGradientConfig::m_falloffWidth)
->Field("FalloffType", &ShapeAreaFalloffGradientConfig::m_falloffType)
;
AZ::EditContext* edit = serialize->GetEditContext();
if (edit)
{
edit->Class<ShapeAreaFalloffGradientConfig>(
"Shape Falloff Gradient", "")
->ClassElement(AZ::Edit::ClassElements::EditorData, "")
->Attribute(AZ::Edit::Attributes::Visibility, AZ::Edit::PropertyVisibility::ShowChildrenOnly)
->Attribute(AZ::Edit::Attributes::AutoExpand, true)
->DataElement(0, &ShapeAreaFalloffGradientConfig::m_shapeEntityId, "Shape Entity Id", "Entity with shape component to test distance against.")
->Attribute(AZ::Edit::Attributes::RequiredService, AZ_CRC("ShapeService", 0xe86aa5fe))
->DataElement(AZ::Edit::UIHandlers::Slider, &ShapeAreaFalloffGradientConfig::m_falloffWidth, "Falloff Width", "Maximum distance used to calculate gradient in meters.")
->Attribute(AZ::Edit::Attributes::Min, 0.0f)
->Attribute(AZ::Edit::Attributes::Max, 100.0f)
->DataElement(0, &ShapeAreaFalloffGradientConfig::m_falloffType, "Falloff Type", "Function for calculating falloff")
// Inner falloff isn't supported yet.
->Attribute(AZ::Edit::Attributes::Visibility, AZ::Edit::PropertyVisibility::Hide)
//->EnumAttribute(ShapeAreaFalloffGradientConfig::FalloffType::Inner, "Inner")
->EnumAttribute(FalloffType::Outer, "Outer")
//->EnumAttribute(ShapeAreaFalloffGradientConfig::FalloffType::InnerOuter, "InnerOuter")
;
}
}
if (auto behaviorContext = azrtti_cast<AZ::BehaviorContext*>(context))
{
behaviorContext->Class<ShapeAreaFalloffGradientConfig>()
->Attribute(AZ::Script::Attributes::Category, "Vegetation")
->Constructor()
->Property("shapeEntityId", BehaviorValueProperty(&ShapeAreaFalloffGradientConfig::m_shapeEntityId))
->Property("falloffWidth", BehaviorValueProperty(&ShapeAreaFalloffGradientConfig::m_falloffWidth))
->Property("falloffType",
[](ShapeAreaFalloffGradientConfig* config) { return (AZ::u8&)(config->m_falloffType); },
[](ShapeAreaFalloffGradientConfig* config, const AZ::u8& i) { config->m_falloffType = (FalloffType)i; })
;
}
}
void ShapeAreaFalloffGradientComponent::GetProvidedServices(AZ::ComponentDescriptor::DependencyArrayType& services)
{
services.push_back(AZ_CRC("GradientService", 0x21c18d23));
}
void ShapeAreaFalloffGradientComponent::GetIncompatibleServices(AZ::ComponentDescriptor::DependencyArrayType& services)
{
services.push_back(AZ_CRC("GradientService", 0x21c18d23));
services.push_back(AZ_CRC("GradientTransformService", 0x8c8c5ecc));
}
void ShapeAreaFalloffGradientComponent::GetRequiredServices([[maybe_unused]] AZ::ComponentDescriptor::DependencyArrayType& services)
{
}
void ShapeAreaFalloffGradientComponent::Reflect(AZ::ReflectContext* context)
{
ShapeAreaFalloffGradientConfig::Reflect(context);
AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
if (serialize)
{
serialize->Class<ShapeAreaFalloffGradientComponent, AZ::Component>()
->Version(0)
->Field("Configuration", &ShapeAreaFalloffGradientComponent::m_configuration)
;
}
if (auto behaviorContext = azrtti_cast<AZ::BehaviorContext*>(context))
{
behaviorContext->Constant("ShapeAreaFalloffGradientComponentTypeId", BehaviorConstant(ShapeAreaFalloffGradientComponentTypeId));
behaviorContext->Class<ShapeAreaFalloffGradientComponent>()->RequestBus("ShapeAreaFalloffGradientRequestBus");
behaviorContext->EBus<ShapeAreaFalloffGradientRequestBus>("ShapeAreaFalloffGradientRequestBus")
->Attribute(AZ::Script::Attributes::Category, "Vegetation")
->Event("GetShapeEntityId", &ShapeAreaFalloffGradientRequestBus::Events::GetShapeEntityId)
->Event("SetShapeEntityId", &ShapeAreaFalloffGradientRequestBus::Events::SetShapeEntityId)
->VirtualProperty("ShapeEntityId", "GetShapeEntityId", "SetShapeEntityId")
->Event("GetFalloffWidth", &ShapeAreaFalloffGradientRequestBus::Events::GetFalloffWidth)
->Event("SetFalloffWidth", &ShapeAreaFalloffGradientRequestBus::Events::SetFalloffWidth)
->VirtualProperty("FalloffWidth", "GetFalloffWidth", "SetFalloffWidth")
->Event("GetFalloffType", &ShapeAreaFalloffGradientRequestBus::Events::GetFalloffType)
->Event("SetFalloffType", &ShapeAreaFalloffGradientRequestBus::Events::SetFalloffType)
->VirtualProperty("FalloffType", "GetFalloffType", "SetFalloffType")
;
}
}
ShapeAreaFalloffGradientComponent::ShapeAreaFalloffGradientComponent(const ShapeAreaFalloffGradientConfig& configuration)
: m_configuration(configuration)
{
}
void ShapeAreaFalloffGradientComponent::Activate()
{
m_dependencyMonitor.Reset();
m_dependencyMonitor.ConnectOwner(GetEntityId());
m_dependencyMonitor.ConnectDependency(m_configuration.m_shapeEntityId);
GradientRequestBus::Handler::BusConnect(GetEntityId());
ShapeAreaFalloffGradientRequestBus::Handler::BusConnect(GetEntityId());
}
void ShapeAreaFalloffGradientComponent::Deactivate()
{
m_dependencyMonitor.Reset();
GradientRequestBus::Handler::BusDisconnect();
ShapeAreaFalloffGradientRequestBus::Handler::BusDisconnect();
}
bool ShapeAreaFalloffGradientComponent::ReadInConfig(const AZ::ComponentConfig* baseConfig)
{
if (auto config = azrtti_cast<const ShapeAreaFalloffGradientConfig*>(baseConfig))
{
m_configuration = *config;
return true;
}
return false;
}
bool ShapeAreaFalloffGradientComponent::WriteOutConfig(AZ::ComponentConfig* outBaseConfig) const
{
if (auto config = azrtti_cast<ShapeAreaFalloffGradientConfig*>(outBaseConfig))
{
*config = m_configuration;
return true;
}
return false;
}
float ShapeAreaFalloffGradientComponent::GetValue(const GradientSampleParams& sampleParams) const
{
float distance = 0.0f;
LmbrCentral::ShapeComponentRequestsBus::EventResult(distance, m_configuration.m_shapeEntityId, &LmbrCentral::ShapeComponentRequestsBus::Events::DistanceFromPoint, sampleParams.m_position);
// Since this is outer falloff, distance should give us values from 1.0 at the minimum distance to 0.0 at the maximum distance.
// The statement is written specifically to handle the 0 falloff case as well. For 0 falloff, all points inside the shape
// (0 distance) return 1.0, and all points outside the shape return 0. This works because division by 0 gives infinity, which gets
// clamped by the GetMax() to 0. However, if distance == 0, it would give us NaN, so we have the separate conditional check to
// handle that case and clamp to 1.0.
return (distance <= 0.0f) ? 1.0f : AZ::GetMax(1.0f - (distance / m_configuration.m_falloffWidth), 0.0f);
}
void ShapeAreaFalloffGradientComponent::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;
}
bool shapeConnected = false;
const float falloffWidth = m_configuration.m_falloffWidth;
LmbrCentral::ShapeComponentRequestsBus::Event(
m_configuration.m_shapeEntityId,
[falloffWidth, positions, &outValues, &shapeConnected](LmbrCentral::ShapeComponentRequestsBus::Events* shapeRequests)
{
shapeConnected = true;
for (size_t index = 0; index < positions.size(); index++)
{
float distance = shapeRequests->DistanceFromPoint(positions[index]);
// Since this is outer falloff, distance should give us values from 1.0 at the minimum distance to 0.0 at the maximum
// distance. The statement is written specifically to handle the 0 falloff case as well. For 0 falloff, all points
// inside the shape (0 distance) return 1.0, and all points outside the shape return 0. This works because division by 0
// gives infinity, which gets clamped by the GetMax() to 0. However, if distance == 0, it would give us NaN, so we have
// the separate conditional check to handle that case and clamp to 1.0.
outValues[index] = (distance <= 0.0f) ? 1.0f : AZ::GetMax(1.0f - (distance / falloffWidth), 0.0f);
}
});
// If there's no shape, there's no falloff.
if (!shapeConnected)
{
for (auto& outValue : outValues)
{
outValue = 1.0f;
}
}
}
AZ::EntityId ShapeAreaFalloffGradientComponent::GetShapeEntityId() const
{
return m_configuration.m_shapeEntityId;
}
void ShapeAreaFalloffGradientComponent::SetShapeEntityId(AZ::EntityId entityId)
{
m_configuration.m_shapeEntityId = entityId;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
float ShapeAreaFalloffGradientComponent::GetFalloffWidth() const
{
return m_configuration.m_falloffWidth;
}
void ShapeAreaFalloffGradientComponent::SetFalloffWidth(float falloffWidth)
{
m_configuration.m_falloffWidth = falloffWidth;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
FalloffType ShapeAreaFalloffGradientComponent::GetFalloffType() const
{
return m_configuration.m_falloffType;
}
void ShapeAreaFalloffGradientComponent::SetFalloffType(FalloffType type)
{
m_configuration.m_falloffType = type;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
}