// 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.
In order to start using the HTTP module we first need to modify our Build.cs file to include a few new public dependencies. Specifically Http, Json and JsonUtilities.
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
public class Network1 : ModuleRules
{
public Network1(ReadOnlyTargetRules Target):base(Target)
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "HTTP")
UBoxComponent* OverlapComponent;
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "HTTP")
UWidgetComponent* ResponseWidgetComponent;
private:
FHttpModule* Http;
};
UCLASS()
class NETWORK1_API AHTTPGETActor : public AActor
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
AHTTPGETActor();
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
UFUNCTION()
void OnBeginOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OtherComp, int32 OtherBodyIndex, bool bFromSweep, const FHitResult& SweepResult);
/*Assign this function to call when the GET request processes sucessfully*/
void OnGetUsersResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful);
void OnGetUserByUsernameResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful);
UFUNCTION()
void SendHTTPGet(FString Username);
private:
void AddUserToWidget(TSharedPtr<FJsonObject> JsonObject);
public:
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "HTTP")
UBoxComponent* OverlapComponent;
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "HTTP")
UWidgetComponent* ResponseWidgetComponent;
private:
FHttpModule* Http;
};
UCLASS()
class NETWORK1_API AHTTPGETActor : public AActor
{
GENERATED_BODY()
public:
// Sets default values for this actor's properties
AHTTPGETActor();
protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override;
public:
// Called every frame
virtual void Tick(float DeltaTime) override;
UFUNCTION()
void OnBeginOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OtherComp, int32 OtherBodyIndex, bool bFromSweep, const FHitResult& SweepResult);
/*Assign this function to call when the GET request processes sucessfully*/
void OnGetUsersResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful);
void OnGetUserByUsernameResponse(FHttpRequestPtr Request, FHttpResponsePtr Response, bool bWasSuccessful);
UFUNCTION()
void SendHTTPGet(FString Username);
private:
void AddUserToWidget(TSharedPtr<FJsonObject> JsonObject);
public:
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "HTTP")
UBoxComponent* OverlapComponent;
UPROPERTY(VisibleAnywhere, BlueprintReadOnly, meta = (AllowPrivateAccess = "true"), Category = "HTTP")
UWidgetComponent* ResponseWidgetComponent;
private:
FHttpModule* Http;
};
The other property is our reference to the FHttpModule that will be used for all of our network communication with the server. This is available to us by including the “Http.h” header in our Actor.
Let’s step through our functions one by one and see how they all communicate.
First thing is our overlap that is triggered when the character interacts with this Actor as well as the instantiation of our components and the FHttpModule.
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
// Sets default values
AHTTPGETActor::AHTTPGETActor()
{
// 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
AHTTPGETActor::AHTTPGETActor()
{
// 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;
OverlapComponent = CreateDefaultSubobject<UBoxComponent>(TEXT("Overlap Area"));
SetRootComponent(OverlapComponent);
ResponseWidgetComponent = CreateDefaultSubobject<UWidgetComponent>(TEXT("Response Widget"));
ResponseWidgetComponent->SetupAttachment(OverlapComponent);
Http = &FHttpModule::Get();
}
// Called when the game starts or when spawned
void AHTTPGETActor::BeginPlay()
{
Super::BeginPlay();
OverlapComponent->OnComponentBeginOverlap.AddDynamic(this, &ThisClass::OnBeginOverlap);
}
// Sets default values
AHTTPGETActor::AHTTPGETActor()
{
// 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;
OverlapComponent = CreateDefaultSubobject<UBoxComponent>(TEXT("Overlap Area"));
SetRootComponent(OverlapComponent);
ResponseWidgetComponent = CreateDefaultSubobject<UWidgetComponent>(TEXT("Response Widget"));
ResponseWidgetComponent->SetupAttachment(OverlapComponent);
Http = &FHttpModule::Get();
}
// Called when the game starts or when spawned
void AHTTPGETActor::BeginPlay()
{
Super::BeginPlay();
OverlapComponent->OnComponentBeginOverlap.AddDynamic(this, &ThisClass::OnBeginOverlap);
}
With our objects all setup we can now proceed to looking at the OnBeginOverlap method as that is the first thing our character will interact with.
In the begin overlap we do a few things. We update our user widget ( HTTPResponseWidget ) that a loading operation has started and we try to retrieve the username of the player that interacted with our component.
We then send the players username over to SendHTTPGet() which will try to determine if it’s a valid username or not and make it’s HTTP calls out to an external service.
Here is where we start creating our request structure for the HTTP GET call. This means we have to provide a URL and depending on the username we either use it to retrieve a single record or instead get all users available to us.
We also create callbacks via OnProcessRequestComplete to two separate methods: OnGetUsersResponse and OnGetUserByUsernameResponse.
This allows us to handle the response structure per each GET call separately and offload the processing of those requests to different components if required.
We also include some custom headers that can be used for things like the Content-Type definition but also for more complex security and authentication situations.
The OnGetUsersResponse method will be processing this JSON Payload which includes an array of elements as part of the response.
Lastly in our OnGetUsersResponse and OnGetUserByUsernameResponse methods we can use the reference to the FHttpRequestPtr and FHttpResponsePtr parameters to determine if our calls were successful and came back with a status code of 200
Additionally we can observe how the JSON structure is parsed by first using the FJsonObjectin combination with the TJsonReader
That’s it, you successfully processed a JSON response from an external service using your Unreal code.
Hope this helps you guys make your own network requests.
If you would like to see more examples check out the video as well as the GitHub project for examples of child classes that move the player and particles around.
Today we are going to take a look how to use Spline Components and modify their various properties to allow us to specify custom materials as well as determine how meshes are attached to the spline.
The project files for this video & article can be found on our GitHub page.
Looking at our previous tutorial we know that splines are not terribly hard to work with but there is a bit of magic to having something move along a spline.
The key ingredient in this is the FTimeline. This allows us to keep track of an execution of values across a specific timeframe.
The FTimeline combined with a UCurveFloat gives us all that flexibility. So let’s look at the code to see how this all breaks down.
Looking over the code we can see that as part of the FTimeline definition we are specifying a function that will be executed during the runtime of the timeline.
As you can see the logic is not that scary. We are simply taking the delta time by the lenght of the spline and retrieving that points location and rotation.
This is then applied to our mesh and we prevent the pitch from being corrected so our elevator / platform does not tip as it’s traversing along the spline.
Lastly let’s take a look at the function OnEndMovementTimeline() which is called at the .. you guessed it … end of our timeline.
This function is a great spot to put in logic for things like restarting the movement or reversing the movement by injecting some simple properties into our new object.
For further reading about these topics take a look at the following links: