// Sets default values
APressurePlateTrigger::APressurePlateTrigger()
{
// Set this actor to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
BaseMesh = CreateDefaultSubobject<UStaticMeshComponent>(TEXT("Base Mesh"));
SetRootComponent(BaseMesh);
MoveableMesh = CreateDefaultSubobject<UMovableStaticMeshComponent>(TEXT("Movable Mesh"));
MoveableMesh->SetupAttachment(BaseMesh);
OverlapComponent = CreateDefaultSubobject<USphereComponent>(TEXT("Overlap Area"));
OverlapComponent->SetupAttachment(MoveableMesh);
}
// Called when the game starts or when spawned
void APressurePlateTrigger::BeginPlay()
{
Super::BeginPlay();
OverlapComponent->OnComponentBeginOverlap.AddDynamic(this, &ThisClass::OnBeginOverlap);
OverlapComponent->OnComponentEndOverlap.AddDynamic(this, &ThisClass::OnEndOverlap);
}
// Called every frame
void APressurePlateTrigger::Tick(float DeltaTime)
{
Super::Tick(DeltaTime);
}
void APressurePlateTrigger::Interact(bool bIsInteracting)
{
TArray<AActor*> InteractableActors;
UGameplayStatics::GetAllActorsWithInterface(GetWorld(), UInteractableInterface::StaticClass(), InteractableActors);
for(AActor* Actor : InteractableActors)
{
FName InteractTag = IInteractableInterface::Execute_GetInteractTag(Actor);
for(FName Tag : TargetTags)
{
if(InteractTag.IsEqual(Tag, ENameCase::IgnoreCase))
{
IInteractableInterface::Execute_Interact(Actor, bIsInteracting);
}
}
}
}
void APressurePlateTrigger::OnBeginOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OtherComp, int32 OtherBodyIndex, bool bFromSweep, const FHitResult& SweepResult)
{
if(!bIsDisabled && !bIsTriggered)
{
MoveableMesh->Move(true);
Interact(true);
bIsTriggered = true;
}
}
void APressurePlateTrigger::OnEndOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OtherComp, int32 OtherBodyIndex)
{
if(!bIsDisabled && bResetTrigger)
{
MoveableMesh->Move(false);
Interact(false);
bIsTriggered = false;
}
}
// Sets default values
APressurePlateTrigger::APressurePlateTrigger()
{
// Set this actor to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
BaseMesh = CreateDefaultSubobject<UStaticMeshComponent>(TEXT("Base Mesh"));
SetRootComponent(BaseMesh);
MoveableMesh = CreateDefaultSubobject<UMovableStaticMeshComponent>(TEXT("Movable Mesh"));
MoveableMesh->SetupAttachment(BaseMesh);
OverlapComponent = CreateDefaultSubobject<USphereComponent>(TEXT("Overlap Area"));
OverlapComponent->SetupAttachment(MoveableMesh);
}
// Called when the game starts or when spawned
void APressurePlateTrigger::BeginPlay()
{
Super::BeginPlay();
OverlapComponent->OnComponentBeginOverlap.AddDynamic(this, &ThisClass::OnBeginOverlap);
OverlapComponent->OnComponentEndOverlap.AddDynamic(this, &ThisClass::OnEndOverlap);
}
// Called every frame
void APressurePlateTrigger::Tick(float DeltaTime)
{
Super::Tick(DeltaTime);
}
void APressurePlateTrigger::Interact(bool bIsInteracting)
{
TArray<AActor*> InteractableActors;
UGameplayStatics::GetAllActorsWithInterface(GetWorld(), UInteractableInterface::StaticClass(), InteractableActors);
for(AActor* Actor : InteractableActors)
{
FName InteractTag = IInteractableInterface::Execute_GetInteractTag(Actor);
for(FName Tag : TargetTags)
{
if(InteractTag.IsEqual(Tag, ENameCase::IgnoreCase))
{
IInteractableInterface::Execute_Interact(Actor, bIsInteracting);
}
}
}
}
void APressurePlateTrigger::OnBeginOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OtherComp, int32 OtherBodyIndex, bool bFromSweep, const FHitResult& SweepResult)
{
if(!bIsDisabled && !bIsTriggered)
{
MoveableMesh->Move(true);
Interact(true);
bIsTriggered = true;
}
}
void APressurePlateTrigger::OnEndOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OtherComp, int32 OtherBodyIndex)
{
if(!bIsDisabled && bResetTrigger)
{
MoveableMesh->Move(false);
Interact(false);
bIsTriggered = false;
}
}
Since we offloaded the majority of the work to our Movable Static Mesh Component you will notice that this actor is really not doing much of anything.
It handles the Begin and End overlap states to simply notify the movable component that it needs to perform some work.
The only tricky bit is examining the world for our affected components. This is done by using the TargetTags against any actor that implements a specific interface.
Using UGameplayStatics::GetAllActorsWithInterface we can find all the appropriate actors and then use the interface calls directly to call the specific methods on t hose actors.
Let’s go ahead and define our interface as we haven’t done that yet.
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
// This class does not need to be modified.
UINTERFACE(MinimalAPI)
class UInteractableInterface : public UInterface
{
GENERATED_BODY()
};
/**
*
*/
class PUZZLE1_API IInteractableInterface
{
GENERATED_BODY()
// Add interface functions to this class. This is the class that will be inherited to implement this interface.
// This class does not need to be modified.
UINTERFACE(MinimalAPI)
class UInteractableInterface : public UInterface
{
GENERATED_BODY()
};
/**
*
*/
class PUZZLE1_API IInteractableInterface
{
GENERATED_BODY()
// Add interface functions to this class. This is the class that will be inherited to implement this interface.
public:
UFUNCTION(BlueprintNativeEvent, BlueprintCallable, Category = "Interact")
FName GetInteractTag();
UFUNCTION(BlueprintNativeEvent, BlueprintCallable, Category = "Interact")
void Interact(bool bInteracting);
};
// This class does not need to be modified.
UINTERFACE(MinimalAPI)
class UInteractableInterface : public UInterface
{
GENERATED_BODY()
};
/**
*
*/
class PUZZLE1_API IInteractableInterface
{
GENERATED_BODY()
// Add interface functions to this class. This is the class that will be inherited to implement this interface.
public:
UFUNCTION(BlueprintNativeEvent, BlueprintCallable, Category = "Interact")
FName GetInteractTag();
UFUNCTION(BlueprintNativeEvent, BlueprintCallable, Category = "Interact")
void Interact(bool bInteracting);
};
The two methods of GetInteractTag and Interact is what our doors / windows / chests etc will have to implement in order for our pressure plate to work.
We are going to create a basic actor that similar to our pressure plate actor only contains a few meshes but inherits our new interface.
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
UCLASS()
class PUZZLE1_API AMovingStaticMeshActor : public AActor, public IInteractableInterface
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
UCLASS()
class PUZZLE1_API AMovingStaticMeshActor : public AActor, public IInteractableInterface
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
AMovingStaticMeshActor();
virtual FName GetInteractTag_Implementation() override;
virtual void Interact_Implementation(bool bInteracting) override;
bool IsDisabled() const { return bIsDisabled; }
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
public:
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "Moving")
UStaticMeshComponent* BaseMesh;
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "Moving")
UMovableStaticMeshComponent* MovableMesh;
UPROPERTY(EditDefaultsOnly, BlueprintReadOnly, Category = "Moving")
FName InteractTag;
UPROPERTY(EditDefaultsOnly, BlueprintReadOnly, Category = "Moving")
bool bIsDisabled;
};
UCLASS()
class PUZZLE1_API AMovingStaticMeshActor : public AActor, public IInteractableInterface
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
AMovingStaticMeshActor();
virtual FName GetInteractTag_Implementation() override;
virtual void Interact_Implementation(bool bInteracting) override;
bool IsDisabled() const { return bIsDisabled; }
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
public:
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "Moving")
UStaticMeshComponent* BaseMesh;
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "Moving")
UMovableStaticMeshComponent* MovableMesh;
UPROPERTY(EditDefaultsOnly, BlueprintReadOnly, Category = "Moving")
FName InteractTag;
UPROPERTY(EditDefaultsOnly, BlueprintReadOnly, Category = "Moving")
bool bIsDisabled;
};
The other property we need to ensure is available to our new actor is the InteractTag which our GetInteractTag method returns. This is important as we use those values do to a string match.
Now let’s define the body of our new actor.
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
// Sets default values
AMovingStaticMeshActor::AMovingStaticMeshActor()
{
// Set this actor to call Tick() every frame. You can turn this off to improve performance if you don't need it.
// Sets default values
AMovingStaticMeshActor::AMovingStaticMeshActor()
{
// Set this actor to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
BaseMesh = CreateDefaultSubobject<UStaticMeshComponent>(TEXT("Base Mesh"));
SetRootComponent(BaseMesh);
MovableMesh = CreateDefaultSubobject<UMovableStaticMeshComponent>(TEXT("Movable Mesh"));
MovableMesh->SetupAttachment(GetRootComponent());
}
// Called when the game starts or when spawned
void AMovingStaticMeshActor::BeginPlay()
{
Super::BeginPlay();
}
// Called every frame
void AMovingStaticMeshActor::Tick(float DeltaTime)
{
Super::Tick(DeltaTime);
}
FName AMovingStaticMeshActor::GetInteractTag_Implementation()
{
return InteractTag;
}
void AMovingStaticMeshActor::Interact_Implementation(bool bInteracting)
{
if(!bIsDisabled)
{
MovableMesh->Move(bInteracting);
}
}
// Sets default values
AMovingStaticMeshActor::AMovingStaticMeshActor()
{
// Set this actor to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
BaseMesh = CreateDefaultSubobject<UStaticMeshComponent>(TEXT("Base Mesh"));
SetRootComponent(BaseMesh);
MovableMesh = CreateDefaultSubobject<UMovableStaticMeshComponent>(TEXT("Movable Mesh"));
MovableMesh->SetupAttachment(GetRootComponent());
}
// Called when the game starts or when spawned
void AMovingStaticMeshActor::BeginPlay()
{
Super::BeginPlay();
}
// Called every frame
void AMovingStaticMeshActor::Tick(float DeltaTime)
{
Super::Tick(DeltaTime);
}
FName AMovingStaticMeshActor::GetInteractTag_Implementation()
{
return InteractTag;
}
void AMovingStaticMeshActor::Interact_Implementation(bool bInteracting)
{
if(!bIsDisabled)
{
MovableMesh->Move(bInteracting);
}
}
Again very simple logic as all we are doing is triggering the Movable Static Mesh Component when someone interacts with this actor.
This actor can now be used to represent a door, window, trunk of a car, or even an elevator. Basically anything that has a single component that you need to move or rotate this will give you that flexbility.
Please check out the video and the GitHub page for more details.
Today we resume our C++ Fundamentals lessons by looking at interfaces and how they can help simply our life by allowing for common contracts to be shared by different game actors.
Tags are essentially a collection of strings that can be applied to an Actor or an Actors Component and then referenced by other Actors or parts of the game world by those tags.
Very similar to using tags on a blog to include metadata for search engines to query … like this blog !
Tags ?! We don’t need to no stinking tags !
Sure you do, and they are really easy to use which is why this post is going to be super short.
Take a look, in this case we are displaying all tags for a specific actor.
// iterate over all of our actors
for(TActorIterator<AActor> ActorIterator(GetWorld()); ActorIterator; ++ActorIterator)
{
AActor* Actor = *ActorIterator;
// ensure actor is not null
// ignore self if found
// ensure we find actors of a specific interface only
if(Actor && Actor != this && Actor->GetClass()->ImplementsInterface(UInteractiveActor::StaticClass()))
{
// display all available tags for an actor
for(FName Tag : Actor->Tags)
{
GEngine->AddOnScreenDebugMessage(-1, 5.f, FColor::Cyan, Tag.ToString());
}
}
}
// iterate over all of our actors
for(TActorIterator<AActor> ActorIterator(GetWorld()); ActorIterator; ++ActorIterator)
{
AActor* Actor = *ActorIterator;
// ensure actor is not null
// ignore self if found
// ensure we find actors of a specific interface only
if(Actor && Actor != this && Actor->GetClass()->ImplementsInterface(UInteractiveActor::StaticClass()))
{
// display all available tags for an actor
for(FName Tag : Actor->Tags)
{
GEngine->AddOnScreenDebugMessage(-1, 5.f, FColor::Cyan, Tag.ToString());
}
}
}
We can also just check for a specific tag without iterating over all of them by using ActorHasTag method.
Today we resume our C++ Fundamentals lessons by looking at interfaces and how they can help simply our life by allowing for common contracts to be shared by different game actors.
An interface is an abstract definition of a contract. What this means is an interface defines how our class can be interfaced ( hah ! ) with while keeping the implementation details away.
More than that it allows us to create a common method signature between various disparate objects.
Consider I have an interface that has a method on it called “DoStuff” and I have two objects, Dog and Person, that inherit this interface. When I call Dog->DoStuff he may perform a trick while if I call the same method of Person->DoStuff he may tell me to fuck off and to stop bothering them. But from the point of view of the code interacting with those actors it’s the same behavior.
This allows for a lot of re-use and generic implementations while keeping the details of those implementations specific.
This all seems very abstract, show me code !
So let’s take a look at how an interface class is structured
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
UINTERFACE(MinimalAPI)
class UInteractiveActor : public UInterface
{
GENERATED_BODY()
};
/**
*
*/
class UE4FUNDAMENTALS09_API IInteractiveActor
{
GENERATED_BODY()
// Add interface functions to this class. This is the class that will be inherited to implement this interface.
UINTERFACE(MinimalAPI)
class UInteractiveActor : public UInterface
{
GENERATED_BODY()
};
/**
*
*/
class UE4FUNDAMENTALS09_API IInteractiveActor
{
GENERATED_BODY()
// Add interface functions to this class. This is the class that will be inherited to implement this interface.
public:
UFUNCTION(BlueprintCallable, BlueprintNativeEvent, Category = "Interact")
void Interact();
};
UINTERFACE(MinimalAPI)
class UInteractiveActor : public UInterface
{
GENERATED_BODY()
};
/**
*
*/
class UE4FUNDAMENTALS09_API IInteractiveActor
{
GENERATED_BODY()
// Add interface functions to this class. This is the class that will be inherited to implement this interface.
public:
UFUNCTION(BlueprintCallable, BlueprintNativeEvent, Category = "Interact")
void Interact();
};
As you can see it’s pretty thin in definition, we simply have a class that inherits from UInterface and then defines the method signature of “Interact”.
Excellent now we are cooking with fire, now what ?
Once we have the interface in place we now start including it on our objects. Let’s take a look at our InteractiveProp.
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
UCLASS()
class UE4FUNDAMENTALS09_API AInteractiveProp : public AActor, public IInteractiveActor
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
virtualvoidInteract_Implementation() override; // actual implementation of our interact method
private:
bool bIsBig;
};
UCLASS()
class UE4FUNDAMENTALS09_API AInteractiveProp : public AActor, public IInteractiveActor
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
AInteractiveProp();
UPROPERTY(VisibleAnywhere, BlueprintReadWrite, Category = "Interact")
class UStaticMeshComponent* BaseMesh;
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
UFUNCTION(BlueprintCallable, BlueprintNativeEvent, Category = "Interact")
void Interact(); // prototype declaration
virtual void Interact_Implementation() override; // actual implementation of our interact method
private:
bool bIsBig;
};
UCLASS()
class UE4FUNDAMENTALS09_API AInteractiveProp : public AActor, public IInteractiveActor
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
AInteractiveProp();
UPROPERTY(VisibleAnywhere, BlueprintReadWrite, Category = "Interact")
class UStaticMeshComponent* BaseMesh;
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
UFUNCTION(BlueprintCallable, BlueprintNativeEvent, Category = "Interact")
void Interact(); // prototype declaration
virtual void Interact_Implementation() override; // actual implementation of our interact method
private:
bool bIsBig;
};
The major thing to note is that in order to leverage the interface, we modify our class signature to bring in “IInteractiveActor”
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
class UE4FUNDAMENTALS09_API AInteractiveProp : public AActor, public IInteractiveActor
class UE4FUNDAMENTALS09_API AInteractiveProp : public AActor, public IInteractiveActor
class UE4FUNDAMENTALS09_API AInteractiveProp : public AActor, public IInteractiveActor
Then we overwrite the Interact method by first declaring the prototype method signature and then the implementation one. Please note that the “_Implementation” is important and has to be exact for the reflection mechanic in Unreal to process your interface + implementation.