// Copyright Epic Games, Inc. All Rights Reserved. #pragma once #include "Engine/DataAsset.h" #include "StateTreeSchema.h" #include "StateTreePropertyBindings.h" #include "StateTreeInstanceData.h" #include "StateTree.generated.h" class UUserDefinedStruct; /** Custom serialization version for StateTree Asset */ struct STATETREEMODULE_API FStateTreeCustomVersion { enum Type { // Before any version changes were made in the plugin BeforeCustomVersionWasAdded = 0, // Separated conditions to shared instance data. SharedInstanceData, // Moved evaluators to be global. GlobalEvaluators, // Moved instance data to arrays. InstanceDataArrays, // Added index types. IndexTypes, // Added events. AddedEvents, // Testing mishap AddedFoo, // Changed transition delay TransitionDelay, // Added external transitions AddedExternalTransitions, // Changed how bindings are represented ChangedBindingsRepresentation, // ------------------------------------------------------ VersionPlusOne, LatestVersion = VersionPlusOne - 1 }; /** The GUID for this custom version number */ const static FGuid GUID; private: FStateTreeCustomVersion() {} }; #if WITH_EDITOR /** Struct containing information about the StateTree runtime memory usage. */ struct FStateTreeMemoryUsage { FStateTreeMemoryUsage() = default; FStateTreeMemoryUsage(const FString InName, const FStateTreeStateHandle InHandle = FStateTreeStateHandle::Invalid) : Name(InName) , Handle(InHandle) { } void AddUsage(FConstStructView View); void AddUsage(const UObject* Object); FString Name; FStateTreeStateHandle Handle; int32 NodeCount = 0; int32 EstimatedMemoryUsage = 0; int32 ChildNodeCount = 0; int32 EstimatedChildMemoryUsage = 0; }; #endif /** * StateTree asset. Contains the StateTree definition in both editor and runtime (baked) formats. */ UCLASS(BlueprintType) class STATETREEMODULE_API UStateTree : public UDataAsset { GENERATED_BODY() public: /** @return Shared instance data. */ TSharedPtr GetSharedInstanceData() const; /** @return Number of data views required for StateTree execution (Evaluators, Tasks, Conditions, External data). */ int32 GetNumDataViews() const { return NumDataViews; } /** @return List of external data required by the state tree */ TConstArrayView GetExternalDataDescs() const { return ExternalDataDescs; } /** @return List of context data enforced by the schema that must be provided through the execution context. */ TConstArrayView GetContextDataDescs() const { return ContextDataDescs; } /** @return List of default parameters of the state tree. Default parameter values can be overridden at runtime by the execution context. */ const FInstancedPropertyBag& GetDefaultParameters() const { return Parameters; } /** @return true if the tree asset can be used at runtime. */ bool IsReadyToRun() const; /** @return schema that was used to compile the StateTree. */ const UStateTreeSchema* GetSchema() const { return Schema; } /** @return Pointer to a state or null if state not found */ const FCompactStateTreeState* GetStateFromHandle(const FStateTreeStateHandle StateHandle) const; /** @return State handle matching a given Id; invalid handle if state not found. */ FStateTreeStateHandle GetStateHandleFromId(const FGuid Id) const; /** @return Id of the state matching a given state handle; invalid Id if state not found. */ FGuid GetStateIdFromHandle(const FStateTreeStateHandle Handle) const; /** @return Struct view of the node matching a given node index; invalid view if state not found. */ FConstStructView GetNode(const int32 NodeIndex) const; /** @return View of all states. */ TConstArrayView GetStates() const { return States; } /** @return Pointer to the transition at a given index; null if not found. */ const FCompactStateTransition* GetTransitionFromIndex(const int16 TransitionIndex) const; #if WITH_EDITOR /** Resets the compiled data to empty. */ void ResetCompiled(); /** Calculates runtime memory usage for different sections of the tree. */ TArray CalculateEstimatedMemoryUsage() const; #endif #if WITH_EDITORONLY_DATA /** Edit time data for the StateTree, instance of UStateTreeEditorData */ UPROPERTY() TObjectPtr EditorData; FDelegateHandle OnObjectsReinstancedHandle; FDelegateHandle OnUserDefinedStructReinstancedHandle; #endif /** Hash of the editor data from last compile. Also used to detect mismatching events from recorded traces. */ UPROPERTY() uint32 LastCompiledEditorDataHash = 0; protected: /** * Resolves references between data in the StateTree. * @return true if all references to internal and external data are resolved properly, false otherwise. */ [[nodiscard]] bool Link(); virtual void PostLoad() override; #if WITH_EDITORONLY_DATA static void DeclareConstructClasses(TArray& OutConstructClasses, const UClass* SpecificSubclass); #endif virtual void Serialize(FStructuredArchiveRecord Record) override; #if WITH_EDITOR using FReplacementObjectMap = TMap; void OnObjectsReinstanced(const FReplacementObjectMap& ObjectMap); void OnUserDefinedStructReinstanced(const UUserDefinedStruct& UserDefinedStruct); virtual void PostInitProperties() override; virtual void BeginDestroy() override; virtual void GetAssetRegistryTags(TArray& OutTags) const override; virtual void PostLoadAssetRegistryTags(const FAssetData& InAssetData, TArray& OutTagsAndValuesToUpdate) const override; virtual EDataValidationResult IsDataValid(class FDataValidationContext& Context) const override; #endif static void AddReferencedObjects(UObject* InThis, FReferenceCollector& Collector); private: /** * Reset the data generated by Link(), this in turn will cause IsReadyToRun() to return false. * Used during linking, or to invalidate the linked data when data version is old (requires recompile). */ void ResetLinked(); // Data created during compilation, source data in EditorData. /** Schema used to compile the StateTree. */ UPROPERTY(Instanced) TObjectPtr Schema = nullptr; /** Runtime states, root state at index 0 */ UPROPERTY() TArray States; /** Runtime transitions. */ UPROPERTY() TArray Transitions; /** Evaluators, Tasks, and Condition nodes. */ UPROPERTY() FInstancedStructContainer Nodes; /** Default node instance data (e.g. evaluators, tasks). */ UPROPERTY() FStateTreeInstanceData DefaultInstanceData; /** Shared node instance data (e.g. conditions). */ UPROPERTY() FStateTreeInstanceData SharedInstanceData; mutable FRWLock PerThreadSharedInstanceDataLock; mutable TArray> PerThreadSharedInstanceData; /** List of names external data enforced by the schema, created at compilation. */ UPROPERTY() TArray ContextDataDescs; UPROPERTY() FStateTreePropertyBindings PropertyBindings; /** Mapping of state guid for the Editor and state handles, created at compilation. */ UPROPERTY() TArray IDToStateMappings; /** * Parameters that could be used for bindings within the Tree. * Default values are stored within the asset but StateTreeReference can be used to parameterized the tree. * @see FStateTreeReference */ UPROPERTY() FInstancedPropertyBag Parameters; /** Data view index of the tree Parameters */ UPROPERTY() FStateTreeIndex8 ParametersDataViewIndex = FStateTreeIndex8::Invalid; /** Index of first evaluator in Nodes. */ UPROPERTY() uint16 EvaluatorsBegin = 0; /** Number of evaluators. */ UPROPERTY() uint16 EvaluatorsNum = 0; /** Index of first global task in Nodes. */ UPROPERTY() uint16 GlobalTasksBegin = 0; /** Number of global tasks. */ UPROPERTY() uint16 GlobalTasksNum = 0; /** True if any global task is a transition task. */ UPROPERTY() bool bHasGlobalTransitionTasks = false; // Data created during linking. /** List of external data required by the state tree, created during linking. */ UPROPERTY(Transient) TArray ExternalDataDescs; /** Base index of external data, created during linking. */ UPROPERTY(Transient) int32 ExternalDataBaseIndex = 0; /** Total number of data views, created during linking. */ UPROPERTY(Transient) int32 NumDataViews = 0; /** True if the StateTree was linked successfully. */ bool bIsLinked = false; friend struct FStateTreeInstance; friend struct FStateTreeExecutionContext; #if WITH_EDITORONLY_DATA friend struct FStateTreeCompiler; #endif }; #if UE_ENABLE_INCLUDE_ORDER_DEPRECATED_IN_5_2 #include "Misc/ScopeRWLock.h" #endif