Files
UnrealEngineUWP/Engine/Plugins/Runtime/StateTree/Source/StateTreeModule/Private/Blueprint/StateTreeTaskBlueprintBase.cpp
mikko mononen f7c48b982d StateTree: Change Task EnterState/ExitState signature
- Removed change type from EnterState/ExitState and moved it to transition result
- Added bShouldStateChangeOnReselect which allows to configure a task to behave more like state
- Updated existing tasks

#jira UE-156544
#rb Mieszko.Zielinski
#preflight 6333f6dca907d7192f5f0ccc

[CL 22221382 by mikko mononen in ue5-main branch]
2022-09-28 09:55:53 -04:00

112 lines
4.5 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#include "Blueprint/StateTreeTaskBlueprintBase.h"
#include "CoreMinimal.h"
#include "StateTreeExecutionContext.h"
#include "BlueprintNodeHelpers.h"
#include UE_INLINE_GENERATED_CPP_BY_NAME(StateTreeTaskBlueprintBase)
//----------------------------------------------------------------------//
// UStateTreeTaskBlueprintBase
//----------------------------------------------------------------------//
UStateTreeTaskBlueprintBase::UStateTreeTaskBlueprintBase(const FObjectInitializer& ObjectInitializer)
: Super(ObjectInitializer)
{
bHasEnterState = BlueprintNodeHelpers::HasBlueprintFunction(TEXT("ReceiveEnterState"), *this, *StaticClass());
bHasExitState = BlueprintNodeHelpers::HasBlueprintFunction(TEXT("ReceiveExitState"), *this, *StaticClass());
bHasStateCompleted = BlueprintNodeHelpers::HasBlueprintFunction(TEXT("ReceiveStateCompleted"), *this, *StaticClass());
bHasTick = BlueprintNodeHelpers::HasBlueprintFunction(TEXT("ReceiveTick"), *this, *StaticClass());
}
EStateTreeRunStatus UStateTreeTaskBlueprintBase::EnterState(FStateTreeExecutionContext& Context, const FStateTreeTransitionResult& Transition)
{
if (bHasEnterState)
{
FScopedCurrentContext(*this, Context);
return ReceiveEnterState(Transition);
}
return EStateTreeRunStatus::Running;
}
void UStateTreeTaskBlueprintBase::ExitState(FStateTreeExecutionContext& Context, const FStateTreeTransitionResult& Transition)
{
if (bHasExitState)
{
FScopedCurrentContext(*this, Context);
ReceiveExitState(Transition);
}
}
void UStateTreeTaskBlueprintBase::StateCompleted(FStateTreeExecutionContext& Context, const EStateTreeRunStatus CompletionStatus, const FStateTreeActiveStates& CompletedActiveStates)
{
if (bHasStateCompleted)
{
FScopedCurrentContext(*this, Context);
ReceiveStateCompleted(CompletionStatus, CompletedActiveStates);
}
}
EStateTreeRunStatus UStateTreeTaskBlueprintBase::Tick(FStateTreeExecutionContext& Context, const float DeltaTime)
{
if (bHasTick)
{
FScopedCurrentContext(*this, Context);
return ReceiveTick(DeltaTime);
}
return EStateTreeRunStatus::Running;
}
//----------------------------------------------------------------------//
// FStateTreeBlueprintTaskWrapper
//----------------------------------------------------------------------//
EStateTreeRunStatus FStateTreeBlueprintTaskWrapper::EnterState(FStateTreeExecutionContext& Context, const FStateTreeTransitionResult& Transition) const
{
UStateTreeTaskBlueprintBase* Instance = Context.GetInstanceDataPtr<UStateTreeTaskBlueprintBase>(*this);
check(Instance);
// @todo: remove this and copy bShouldStateChangeOnReselect from the instance instead.
// This is not currently possible todo. Maybe we should have some kind of pre-compile validation pass where this could be done.
const bool bShouldCallStateChange = Transition.ChangeType == EStateTreeStateChangeType::Changed
|| (Transition.ChangeType == EStateTreeStateChangeType::Sustained && Instance->bShouldStateChangeOnReselect);
if (bShouldCallStateChange)
{
return Instance->EnterState(Context, Transition);
}
return EStateTreeRunStatus::Running;
}
void FStateTreeBlueprintTaskWrapper::ExitState(FStateTreeExecutionContext& Context, const FStateTreeTransitionResult& Transition) const
{
UStateTreeTaskBlueprintBase* Instance = Context.GetInstanceDataPtr<UStateTreeTaskBlueprintBase>(*this);
check(Instance);
// @todo: remove this and copy bShouldStateChangeOnReselect from the instance instead.
// This is not currently possible todo. Maybe we should have some kind of pre-compile validation pass where this could be done.
const bool bShouldCallStateChange = Transition.ChangeType == EStateTreeStateChangeType::Changed
|| (Transition.ChangeType == EStateTreeStateChangeType::Sustained && Instance->bShouldStateChangeOnReselect);
if (bShouldCallStateChange)
{
Instance->ExitState(Context, Transition);
}
}
void FStateTreeBlueprintTaskWrapper::StateCompleted(FStateTreeExecutionContext& Context, const EStateTreeRunStatus CompletionStatus, const FStateTreeActiveStates& CompletedActiveStates) const
{
UStateTreeTaskBlueprintBase* Instance = Context.GetInstanceDataPtr<UStateTreeTaskBlueprintBase>(*this);
check(Instance);
Instance->StateCompleted(Context, CompletionStatus, CompletedActiveStates);
}
EStateTreeRunStatus FStateTreeBlueprintTaskWrapper::Tick(FStateTreeExecutionContext& Context, const float DeltaTime) const
{
UStateTreeTaskBlueprintBase* Instance = Context.GetInstanceDataPtr<UStateTreeTaskBlueprintBase>(*this);
check(Instance);
return Instance->Tick(Context, DeltaTime);
}