Files
UnrealEngineUWP/Engine/Plugins/Runtime/SmartObjects/Source/SmartObjectsModule/Private/SmartObjectDefinition.cpp
yoan stamant 9116dc7413 [SmartObject] access to activity tags should go through SmartObjectSlotView
+ Tags Filtering Policy now only affect TagQueries from the definitions and applied to UserTags provided by requests
+ Tags Merging Policy added for Activity Tags
#rnx
#rb josselin.francois
#preflight 621fab91257fd6e0996de3b2

#ROBOMERGE-AUTHOR: yoan.stamant
#ROBOMERGE-SOURCE: CL 19225863 via CL 19236663 via CL 19236832 via CL 19236900
#ROBOMERGE-BOT: UE5 (Release-Engine-Staging -> Main) (v921-19075845)

[CL 19237559 by yoan stamant in ue5-main branch]
2022-03-02 22:33:53 -05:00

141 lines
4.6 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#include "SmartObjectDefinition.h"
#include "SmartObjectSettings.h"
#include "SmartObjectTypes.h"
namespace UE::SmartObject
{
const FVector DefaultSlotSize(40, 40, 90);
}
USmartObjectDefinition::USmartObjectDefinition(const FObjectInitializer& ObjectInitializer): UDataAsset(ObjectInitializer)
{
UserTagsFilteringPolicy = GetDefault<USmartObjectSettings>()->DefaultUserTagsFilteringPolicy;
ActivityTagsMergingPolicy = GetDefault<USmartObjectSettings>()->DefaultActivityTagsMergingPolicy;
}
bool USmartObjectDefinition::Validate() const
{
bValid = false;
if (Slots.Num() == 0)
{
UE_LOG(LogSmartObject, Error, TEXT("Need to provide at least one slot definition"));
return false;
}
// Detect null entries in default definitions
int32 NullEntryIndex;
if (DefaultBehaviorDefinitions.Find(nullptr, NullEntryIndex))
{
UE_LOG(LogSmartObject, Error, TEXT("Null entry found at index %d in default behavior definition list"), NullEntryIndex);
return false;
}
// Detect null entries in slot definitions
for (int i = 0; i < Slots.Num(); ++i)
{
const FSmartObjectSlotDefinition& Slot = Slots[i];
if (Slot.BehaviorDefinitions.Find(nullptr, NullEntryIndex))
{
UE_LOG(LogSmartObject, Error, TEXT("Null definition entry found at index %d in behavior list of slot %d"), i, NullEntryIndex);
return false;
}
}
// Detect missing definitions in slots if no default one are provided
if (DefaultBehaviorDefinitions.Num() == 0)
{
for (int i = 0; i < Slots.Num(); ++i)
{
const FSmartObjectSlotDefinition& Slot = Slots[i];
if (Slot.BehaviorDefinitions.Num() == 0)
{
UE_LOG(LogSmartObject, Error, TEXT("Slot at index %d needs to provide a behavior definition since there is no default one in the SmartObject definition"), i);
return false;
}
}
}
bValid = true;
return true;
}
FBox USmartObjectDefinition::GetBounds() const
{
FBox BoundingBox(ForceInitToZero);
for (const FSmartObjectSlotDefinition& Slot : GetSlots())
{
BoundingBox += Slot.Offset + UE::SmartObject::DefaultSlotSize;
BoundingBox += Slot.Offset - UE::SmartObject::DefaultSlotSize;
}
return BoundingBox;
}
void USmartObjectDefinition::GetSlotActivityTags(const FSmartObjectSlotIndex& SlotIndex, FGameplayTagContainer& OutActivityTags) const
{
if (ensureMsgf(Slots.IsValidIndex(SlotIndex), TEXT("Requesting activity tags for an out of range slot index: %s"), *LexToString(SlotIndex)))
{
GetSlotActivityTags(Slots[SlotIndex], OutActivityTags);
}
}
void USmartObjectDefinition::GetSlotActivityTags(const FSmartObjectSlotDefinition& SlotDefinition, FGameplayTagContainer& OutActivityTags) const
{
OutActivityTags = ActivityTags;
if (ActivityTagsMergingPolicy == ESmartObjectTagMergingPolicy::Combine)
{
OutActivityTags.AppendTags(SlotDefinition.ActivityTags);
}
else if (ActivityTagsMergingPolicy == ESmartObjectTagMergingPolicy::Override && !SlotDefinition.ActivityTags.IsEmpty())
{
OutActivityTags = SlotDefinition.ActivityTags;
}
}
TOptional<FTransform> USmartObjectDefinition::GetSlotTransform(const FTransform& OwnerTransform, const FSmartObjectSlotIndex SlotIndex) const
{
TOptional<FTransform> Transform;
if (ensureMsgf(Slots.IsValidIndex(SlotIndex), TEXT("Requesting slot transform for an out of range index: %s"), *LexToString(SlotIndex)))
{
const FSmartObjectSlotDefinition& Slot = Slots[SlotIndex];
Transform = FTransform(Slot.Rotation, Slot.Offset) * OwnerTransform;
}
return Transform;
}
const USmartObjectBehaviorDefinition* USmartObjectDefinition::GetBehaviorDefinition(const FSmartObjectSlotIndex& SlotIndex,
const TSubclassOf<USmartObjectBehaviorDefinition>& DefinitionClass) const
{
const USmartObjectBehaviorDefinition* Definition = nullptr;
if (Slots.IsValidIndex(SlotIndex))
{
Definition = GetBehaviorDefinitionByType(Slots[SlotIndex].BehaviorDefinitions, DefinitionClass);
}
if (Definition == nullptr)
{
Definition = GetBehaviorDefinitionByType(DefaultBehaviorDefinitions, DefinitionClass);
}
return Definition;
}
const USmartObjectBehaviorDefinition* USmartObjectDefinition::GetBehaviorDefinitionByType(const TArray<USmartObjectBehaviorDefinition*>& BehaviorDefinitions,
const TSubclassOf<USmartObjectBehaviorDefinition>& DefinitionClass)
{
USmartObjectBehaviorDefinition* const* BehaviorDefinition = BehaviorDefinitions.FindByPredicate([&DefinitionClass](USmartObjectBehaviorDefinition* SlotBehaviorDefinition)
{
return SlotBehaviorDefinition != nullptr && SlotBehaviorDefinition->GetClass()->IsChildOf(*DefinitionClass);
});
return BehaviorDefinition != nullptr ? *BehaviorDefinition : nullptr;
}