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/DitherGradientComponent.cpp

354 lines
17 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/DitherGradientComponent.h>
#include <AzCore/Debug/Profiler.h>
#include <AzCore/Math/MathUtils.h>
#include <AzCore/RTTI/BehaviorContext.h>
#include <AzCore/Serialization/EditContext.h>
#include <AzCore/Serialization/SerializeContext.h>
#include <GradientSignal/Util.h>
#include <GradientSignal/Ebuses/SectorDataRequestBus.h>
#include <LmbrCentral/Dependency/DependencyNotificationBus.h>
namespace GradientSignal
{
void DitherGradientConfig::Reflect(AZ::ReflectContext* context)
{
AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
if (serialize)
{
serialize->Class<DitherGradientConfig, AZ::ComponentConfig>()
->Version(1)
->Field("PatternOffset", &DitherGradientConfig::m_patternOffset)
->Field("PatternType", &DitherGradientConfig::m_patternType)
->Field("UseSystemPointsPerUnit", &DitherGradientConfig::m_useSystemPointsPerUnit)
->Field("PointsPerUnit", &DitherGradientConfig::m_pointsPerUnit)
->Field("Gradient", &DitherGradientConfig::m_gradientSampler)
;
AZ::EditContext* edit = serialize->GetEditContext();
if (edit)
{
edit->Class<DitherGradientConfig>(
"Dither Gradient", "")
->ClassElement(AZ::Edit::ClassElements::EditorData, "")
->Attribute(AZ::Edit::Attributes::Visibility, AZ::Edit::PropertyVisibility::ShowChildrenOnly)
->Attribute(AZ::Edit::Attributes::AutoExpand, true)
->DataElement(0, &DitherGradientConfig::m_patternOffset, "Pattern Offset", "Shift pattern lookup indices")
->DataElement(AZ::Edit::UIHandlers::ComboBox, &DitherGradientConfig::m_patternType, "Pattern Type", "")
->EnumAttribute(DitherGradientConfig::BayerPatternType::PATTERN_SIZE_4x4, "4x4")
->EnumAttribute(DitherGradientConfig::BayerPatternType::PATTERN_SIZE_8x8, "8x8")
->ClassElement(AZ::Edit::ClassElements::Group, "Sample Settings")
->DataElement(AZ::Edit::UIHandlers::CheckBox, &DitherGradientConfig::m_useSystemPointsPerUnit, "Use System Points Per Unit", "Automatically sets points per unit. Value is equal to Sector Density / Sector Size")
->DataElement(AZ::Edit::UIHandlers::Slider, &DitherGradientConfig::m_pointsPerUnit, "Points Per Unit", "Scales input position before sampling")
->Attribute(AZ::Edit::Attributes::ReadOnly, &DitherGradientConfig::IsPointsPerUnitResdOnly)
->Attribute(AZ::Edit::Attributes::Min, 0.001f)
->Attribute(AZ::Edit::Attributes::Max, std::numeric_limits<float>::max())
->Attribute(AZ::Edit::Attributes::SoftMax, 100.0f)
->ClassElement(AZ::Edit::ClassElements::Group, "")
->Attribute(AZ::Edit::Attributes::AutoExpand, true)
->DataElement(0, &DitherGradientConfig::m_gradientSampler, "Gradient", "Input gradient whose values will be dithered.")
;
}
}
if (auto behaviorContext = azrtti_cast<AZ::BehaviorContext*>(context))
{
behaviorContext->Class<DitherGradientConfig>()
->Constructor()
->Attribute(AZ::Script::Attributes::Category, "Vegetation")
->Property("useSystemPointsPerUnit", BehaviorValueProperty(&DitherGradientConfig::m_useSystemPointsPerUnit))
->Property("pointsPerUnit", BehaviorValueProperty(&DitherGradientConfig::m_pointsPerUnit))
->Property("patternOffset", BehaviorValueProperty(&DitherGradientConfig::m_patternOffset))
->Property("patternType",
[](DitherGradientConfig* config) { return (AZ::u8&)(config->m_patternType); },
[](DitherGradientConfig* config, const AZ::u8& i) { config->m_patternType = (DitherGradientConfig::BayerPatternType)i; })
->Property("gradientSampler", BehaviorValueProperty(&DitherGradientConfig::m_gradientSampler))
;
}
}
bool DitherGradientConfig::IsPointsPerUnitResdOnly() const
{
return m_useSystemPointsPerUnit;
}
void DitherGradientComponent::GetProvidedServices(AZ::ComponentDescriptor::DependencyArrayType& services)
{
services.push_back(AZ_CRC("GradientService", 0x21c18d23));
}
void DitherGradientComponent::GetIncompatibleServices(AZ::ComponentDescriptor::DependencyArrayType& services)
{
services.push_back(AZ_CRC("GradientService", 0x21c18d23));
}
void DitherGradientComponent::GetRequiredServices([[maybe_unused]] AZ::ComponentDescriptor::DependencyArrayType& services)
{
}
void DitherGradientComponent::Reflect(AZ::ReflectContext* context)
{
DitherGradientConfig::Reflect(context);
AZ::SerializeContext* serialize = azrtti_cast<AZ::SerializeContext*>(context);
if (serialize)
{
serialize->Class<DitherGradientComponent, AZ::Component>()
->Version(0)
->Field("Configuration", &DitherGradientComponent::m_configuration)
;
}
if (auto behaviorContext = azrtti_cast<AZ::BehaviorContext*>(context))
{
behaviorContext->Constant("DitherGradientComponentTypeId", BehaviorConstant(DitherGradientComponentTypeId));
behaviorContext->Class<DitherGradientComponent>()->RequestBus("DitherGradientRequestBus");
behaviorContext->EBus<DitherGradientRequestBus>("DitherGradientRequestBus")
->Attribute(AZ::Script::Attributes::Category, "Vegetation")
->Event("GetUseSystemPointsPerUnit", &DitherGradientRequestBus::Events::GetUseSystemPointsPerUnit)
->Event("SetUseSystemPointsPerUnit", &DitherGradientRequestBus::Events::SetUseSystemPointsPerUnit)
->VirtualProperty("UseSystemPointsPerUnit", "GetUseSystemPointsPerUnit", "SetUseSystemPointsPerUnit")
->Event("GetPointsPerUnit", &DitherGradientRequestBus::Events::GetPointsPerUnit)
->Event("SetPointsPerUnit", &DitherGradientRequestBus::Events::SetPointsPerUnit)
->VirtualProperty("PointsPerUnit", "GetPointsPerUnit", "SetPointsPerUnit")
->Event("GetPatternOffset", &DitherGradientRequestBus::Events::GetPatternOffset)
->Event("SetPatternOffset", &DitherGradientRequestBus::Events::SetPatternOffset)
->VirtualProperty("PatternOffset", "GetPatternOffset", "SetPatternOffset")
->Event("GetPatternType", &DitherGradientRequestBus::Events::GetPatternType)
->Event("SetPatternType", &DitherGradientRequestBus::Events::SetPatternType)
->VirtualProperty("PatternType", "GetPatternType", "SetPatternType")
->Event("GetGradientSampler", &DitherGradientRequestBus::Events::GetGradientSampler)
;
}
}
DitherGradientComponent::DitherGradientComponent(const DitherGradientConfig& configuration)
: m_configuration(configuration)
{
}
void DitherGradientComponent::Activate()
{
m_dependencyMonitor.Reset();
m_dependencyMonitor.ConnectOwner(GetEntityId());
m_dependencyMonitor.ConnectDependency(m_configuration.m_gradientSampler.m_gradientId);
GradientRequestBus::Handler::BusConnect(GetEntityId());
DitherGradientRequestBus::Handler::BusConnect(GetEntityId());
SectorDataNotificationBus::Handler::BusConnect();
}
void DitherGradientComponent::Deactivate()
{
m_dependencyMonitor.Reset();
GradientRequestBus::Handler::BusDisconnect();
DitherGradientRequestBus::Handler::BusDisconnect();
SectorDataNotificationBus::Handler::BusDisconnect();
}
bool DitherGradientComponent::ReadInConfig(const AZ::ComponentConfig* baseConfig)
{
if (auto config = azrtti_cast<const DitherGradientConfig*>(baseConfig))
{
m_configuration = *config;
return true;
}
return false;
}
bool DitherGradientComponent::WriteOutConfig(AZ::ComponentConfig* outBaseConfig) const
{
if (auto config = azrtti_cast<DitherGradientConfig*>(outBaseConfig))
{
*config = m_configuration;
return true;
}
return false;
}
int DitherGradientComponent::ScaledPositionToPatternIndex(const AZ::Vector3& scaledPosition, int patternSize)
{
// The input position is expected to be scaled up so that each integer value is a unique point in our dither pattern, and
// the fractional value is just the amount within the point. The output is the specific index into an NxN pattern to use
// for the dither comparison value.
// Get the floor before casting to int because we want fractional negative values to go "down" to the next negative value.
AZ::Vector3 flooredScaledPosition = scaledPosition.GetFloor();
// For a pattern of 4, we want our indices to go 0, 1, 2, 3, 0, 1, 2, 3, etc. However, we want it continuous across
// negative and positive positions so we can't just use mod with abs(). Instead, we use a double-mod which gives us
// a result that's continuous across all coordinate space.
const int x = ((static_cast<int>(flooredScaledPosition.GetX()) % patternSize) + patternSize) % patternSize;
const int y = ((static_cast<int>(flooredScaledPosition.GetY()) % patternSize) + patternSize) % patternSize;
return (patternSize * y + x);
}
float DitherGradientComponent::GetDitherValue4x4(const AZ::Vector3& scaledPosition)
{
constexpr int patternSize = 4;
constexpr float indexMatrix[] = {
0.0f / 16.0f, 8.0f / 16.0f, 2.0f / 16.0f, 10.0f / 16.0f,
12.0f / 16.0f, 4.0f / 16.0f, 14.0f / 16.0f, 6.0f / 16.0f,
3.0f / 16.0f, 11.0f / 16.0f, 1.0f / 16.0f, 9.0f / 16.0f,
15.0f / 16.0f, 7.0f / 16.0f, 13.0f / 16.0f, 5.0f / 16.0f };
return indexMatrix[ScaledPositionToPatternIndex(scaledPosition, patternSize)];
}
float DitherGradientComponent::GetDitherValue8x8(const AZ::Vector3& scaledPosition)
{
constexpr int patternSize = 8;
constexpr float indexMatrix[] = {
0.0f / 64.0f, 32.0f / 64.0f, 8.0f / 64.0f, 40.0f / 64.0f, 2.0f / 64.0f, 34.0f / 64.0f, 10.0f / 64.0f, 42.0f / 64.0f,
48.0f / 64.0f, 16.0f / 64.0f, 56.0f / 64.0f, 24.0f / 64.0f, 50.0f / 64.0f, 18.0f / 64.0f, 58.0f / 64.0f, 26.0f / 64.0f,
12.0f / 64.0f, 44.0f / 64.0f, 4.0f / 64.0f, 36.0f / 64.0f, 14.0f / 64.0f, 46.0f / 64.0f, 6.0f / 64.0f, 38.0f / 64.0f,
60.0f / 64.0f, 28.0f / 64.0f, 52.0f / 64.0f, 20.0f / 64.0f, 62.0f / 64.0f, 30.0f / 64.0f, 54.0f / 64.0f, 22.0f / 64.0f,
3.0f / 64.0f, 35.0f / 64.0f, 11.0f / 64.0f, 43.0f / 64.0f, 1.0f / 64.0f, 33.0f / 64.0f, 9.0f / 64.0f, 41.0f / 64.0f,
51.0f / 64.0f, 19.0f / 64.0f, 59.0f / 64.0f, 27.0f / 64.0f, 49.0f / 64.0f, 17.0f / 64.0f, 57.0f / 64.0f, 25.0f / 64.0f,
15.0f / 64.0f, 47.0f / 64.0f, 7.0f / 64.0f, 39.0f / 64.0f, 13.0f / 64.0f, 45.0f / 64.0f, 5.0f / 64.0f, 37.0f / 64.0f,
63.0f / 64.0f, 31.0f / 64.0f, 55.0f / 64.0f, 23.0f / 64.0f, 61.0f / 64.0f, 29.0f / 64.0f, 53.0f / 64.0f, 21.0f / 64.0f
};
return indexMatrix[ScaledPositionToPatternIndex(scaledPosition, patternSize)];
}
float DitherGradientComponent::GetCalculatedPointsPerUnit() const
{
float pointsPerUnit = m_configuration.m_pointsPerUnit;
if (m_configuration.m_useSystemPointsPerUnit)
{
SectorDataRequestBus::Broadcast(&SectorDataRequestBus::Events::GetPointsPerMeter, pointsPerUnit);
}
return AZ::GetMax(pointsPerUnit, 0.0001f);
}
float DitherGradientComponent::GetDitherValue(const AZ::Vector3& scaledPosition, float value) const
{
float d = 0.0f;
switch (m_configuration.m_patternType)
{
default:
case DitherGradientConfig::BayerPatternType::PATTERN_SIZE_4x4:
d = GetDitherValue4x4(scaledPosition + m_configuration.m_patternOffset);
break;
case DitherGradientConfig::BayerPatternType::PATTERN_SIZE_8x8:
d = GetDitherValue8x8(scaledPosition + m_configuration.m_patternOffset);
break;
}
return (value > d) ? 1.0f : 0.0f;
}
float DitherGradientComponent::GetValue(const GradientSampleParams& sampleParams) const
{
const AZ::Vector3& coordinate = sampleParams.m_position;
const float pointsPerUnit = GetCalculatedPointsPerUnit();
AZ::Vector3 scaledCoordinate = coordinate * pointsPerUnit;
AZ::Vector3 flooredCoordinate = scaledCoordinate.GetFloor() / pointsPerUnit;
GradientSampleParams adjustedSampleParams = sampleParams;
adjustedSampleParams.m_position = flooredCoordinate;
float value = m_configuration.m_gradientSampler.GetValue(adjustedSampleParams);
return GetDitherValue(scaledCoordinate, value);
}
void DitherGradientComponent::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;
}
const float pointsPerUnit = GetCalculatedPointsPerUnit();
// Create the entire set of floored coordinates to use in the gradient value lookups.
AZStd::vector<AZ::Vector3> flooredCoordinates(positions.size());
for (size_t index = 0; index < positions.size(); index++)
{
AZ::Vector3 scaledCoordinate = positions[index] * pointsPerUnit;
flooredCoordinates[index] = scaledCoordinate.GetFloor() / pointsPerUnit;
}
m_configuration.m_gradientSampler.GetValues(flooredCoordinates, outValues);
// For each gradient value, turn it into a 0 or 1 based on the location and the dither pattern.
for (size_t index = 0; index < positions.size(); index++)
{
AZ::Vector3 scaledCoordinate = positions[index] * pointsPerUnit;
outValues[index] = GetDitherValue(scaledCoordinate, outValues[index]);
}
}
bool DitherGradientComponent::IsEntityInHierarchy(const AZ::EntityId& entityId) const
{
return m_configuration.m_gradientSampler.IsEntityInHierarchy(entityId);
}
void DitherGradientComponent::OnSectorDataConfigurationUpdated() const
{
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
bool DitherGradientComponent::GetUseSystemPointsPerUnit() const
{
return m_configuration.m_useSystemPointsPerUnit;
}
void DitherGradientComponent::SetUseSystemPointsPerUnit(bool value)
{
m_configuration.m_useSystemPointsPerUnit = value;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
float DitherGradientComponent::GetPointsPerUnit() const
{
return m_configuration.m_pointsPerUnit;
}
void DitherGradientComponent::SetPointsPerUnit(float points)
{
m_configuration.m_pointsPerUnit = points;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
AZ::Vector3 DitherGradientComponent::GetPatternOffset() const
{
return m_configuration.m_patternOffset;
}
void DitherGradientComponent::SetPatternOffset(AZ::Vector3 offset)
{
m_configuration.m_patternOffset = offset;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
AZ::u8 DitherGradientComponent::GetPatternType() const
{
return (AZ::u8)m_configuration.m_patternType;
}
void DitherGradientComponent::SetPatternType(AZ::u8 type)
{
m_configuration.m_patternType = (DitherGradientConfig::BayerPatternType)type;
LmbrCentral::DependencyNotificationBus::Event(GetEntityId(), &LmbrCentral::DependencyNotificationBus::Events::OnCompositionChanged);
}
GradientSampler& DitherGradientComponent::GetGradientSampler()
{
return m_configuration.m_gradientSampler;
}
}