Hi everyone !
I’m working on a host/client multiplayer game using gravity fields on small planets and I’m encountering some issues with the character rotation replication. I followed this tutorial to get the gravity to work locally and slightly adpated it to my needs. On the host everything is fine, player rotation is correct for all player characters but for the client only the local client is correctly rotated, all non-local clients are rotated in world space. Also, something strange is when the character jumps, it is correctly replicated while in air, even when I make the character rotate, but suddenly rotate when it lands. Here are some videos to be clearer:
From Host:
From Client:
The character is set like this:
I might be missing something but the host characters rotation should be replicted to not owning client, is it a correct assumption? I feel like it’s a Controller problem since it works only on characters which have a controller locally.
I’m working in UE5.5. I gathered all the gravity management in a UActorComponent child component:
The header file:
#pragma once
#include "Components/ActorComponent.h"
#include "Gameplay/Gravity/GravityInfluencedInterface.h"
#include "GravityInfluencedComponent.generated.h"
class ADeliveryShipCharacter;
class ADeliveryShipPlayerController;
UCLASS( ClassGroup=(Custom), meta=(BlueprintSpawnableComponent) )
class DELIVERYSHIP_API UGravityInfluencedComponent : public UActorComponent, public IGravityInfluencedInterface
{
GENERATED_BODY()
public:
UGravityInfluencedComponent();
protected:
virtual void BeginPlay() override;
/** Sets the current gravity field */
virtual void SetCurrentGravityField(UPrimitiveComponent* NewCurrentGravityField) override;
/** Gets the current gravity field */
virtual UPrimitiveComponent* GetCurrentGravityField() override;
/** Gets the list of all the currently overlapped gravity fields */
virtual TArray<UPrimitiveComponent*>* GetOverlappingGravityFields() override;
public:
virtual void TickComponent(float DeltaTime, ELevelTick TickType, FActorComponentTickFunction* ThisTickFunction) override;
UFUNCTION()
void UpdateRotation(float DeltaTime);
// Converts a rotation from world space to gravity relative space.
UFUNCTION(BlueprintPure)
static FRotator GetGravityRelativeRotation(FRotator Rotation, FVector GravityDirection);
// Converts a rotation from gravity relative space to world space.
UFUNCTION(BlueprintPure)
static FRotator GetGravityWorldRotation(FRotator Rotation, FVector GravityDirection);
private:
UFUNCTION(Server, Reliable)
void Server_SetUseGravity(bool bNewUseGravity);
UFUNCTION(Server, Unreliable)
void Server_SetCurrentGravityDirection(FVector NewCurrentGravityDirection);
UFUNCTION()
void OnRep_UseGravity();
UFUNCTION()
void OnRep_CurrentGravityDirection();
ADeliveryShipCharacter* PlayerCharacter = nullptr;
ADeliveryShipPlayerController* PlayerController = nullptr;
FVector LastFrameGravity = FVector::ZeroVector;
UPrimitiveComponent* CurrentGravityField;
bool bUpdateGravityFieldInTick = false;
TArray<UPrimitiveComponent*> OverlappingGravityFields;
UPROPERTY(ReplicatedUsing(OnRep_CurrentGravityDirection))
FVector CurrentGravityDirection = FVector::DownVector;
UPROPERTY(ReplicatedUsing(OnRep_UseGravity))
bool bUseGravity = true;
};
The cpp file:
#include "Gameplay/Gravity/GravityInfluencedComponent.h"
#include "Player/DeliveryShipCharacter.h"
#include "Player/DeliveryShipPlayerController.h"
#include "Gameplay/Gravity/GravityInterface.h"
#include "Components/CapsuleComponent.h"
#include "GameFramework/CharacterMovementComponent.h"
#include "Net/UnrealNetwork.h"
// Sets default values for this component's properties
UGravityInfluencedComponent::UGravityInfluencedComponent()
{
// Set this component to be initialized when the game starts, and to be ticked every frame. You can turn these features
// off to improve performance if you don't need them.
PrimaryComponentTick.bCanEverTick = true;
SetIsReplicatedByDefault(true);
}
void UGravityInfluencedComponent::GetLifetimeReplicatedProps(TArray<FLifetimeProperty>& OutLifetimeProps) const
{
Super::GetLifetimeReplicatedProps(OutLifetimeProps);
DOREPLIFETIME(UGravityInfluencedComponent, CurrentGravityDirection);
DOREPLIFETIME(UGravityInfluencedComponent, bUseGravity);
}
// Called when the game starts
void UGravityInfluencedComponent::BeginPlay()
{
Super::BeginPlay();
PlayerCharacter = Cast<ADeliveryShipCharacter>(GetOwner());
check(PlayerCharacter);
if (PlayerCharacter->IsLocallyControlled())
{
PlayerCharacter->GetCapsuleComponent()->OnComponentBeginOverlap.AddDynamic(this, &UGravityInfluencedComponent::OnBeginOverlap);
PlayerCharacter->GetCapsuleComponent()->OnComponentEndOverlap.AddDynamic(this, &UGravityInfluencedComponent::OnEndOverlap);
}
}
// Called every frame
void UGravityInfluencedComponent::TickComponent(float DeltaTime, ELevelTick TickType, FActorComponentTickFunction* ThisTickFunction)
{
Super::TickComponent(DeltaTime, TickType, ThisTickFunction);
if (!PlayerController && (PlayerCharacter->IsLocallyControlled() || PlayerCharacter->HasAuthority()))
{
PlayerController = Cast<ADeliveryShipPlayerController>(PlayerCharacter->GetController());
if (!PlayerController)
{
return;
}
PlayerController->OnUpdateRotation.AddDynamic(this, &UGravityInfluencedComponent::UpdateRotation);
}
if (PlayerCharacter->IsLocallyControlled())
{
if (CurrentGravityField && bUpdateGravityFieldInTick)
{
CurrentGravityDirection = IGravityInterface::Execute_GetGravityDirection(CurrentGravityField, PlayerCharacter->GetRootComponent());
PlayerCharacter->GetCharacterMovement()->SetGravityDirection(CurrentGravityDirection);
if (GetNetMode() != ENetMode::NM_ListenServer)
{
Server_SetCurrentGravityDirection(CurrentGravityDirection);
}
}
}
}
void UGravityInfluencedComponent::UpdateRotation(float DeltaTime)
{
FVector GravityDirection = FVector::DownVector;
if (PlayerCharacter)
{
UCharacterMovementComponent* MoveComp = PlayerCharacter->GetCharacterMovement();
if (MoveComp)
{
GravityDirection = MoveComp->GetGravityDirection();
}
}
// Get the current control rotation in world space
FRotator ViewRotation = PlayerController->GetControlRotation();
// Add any rotation from the gravity changes, if any happened.
// Delete this code block if you don't want the camera to automatically compensate for gravity rotation.
if (!LastFrameGravity.Equals(FVector::ZeroVector))
{
const FQuat DeltaGravityRotation = FQuat::FindBetweenNormals(LastFrameGravity, GravityDirection);
const FQuat WarpedCameraRotation = DeltaGravityRotation * FQuat(ViewRotation);
ViewRotation = WarpedCameraRotation.Rotator();
}
LastFrameGravity = GravityDirection;
// Convert the view rotation from world space to gravity relative space.
// Now we can work with the rotation as if no custom gravity was affecting it.
ViewRotation = GetGravityRelativeRotation(ViewRotation, GravityDirection);
// Calculate Delta to be applied on ViewRotation
FRotator DeltaRot(PlayerController->RotationInput);
if (PlayerController->PlayerCameraManager)
{
PlayerController->PlayerCameraManager->ProcessViewRotation(DeltaTime, ViewRotation, DeltaRot);
// Zero the roll of the camera as we always want it horizontal in relation to the gravity.
ViewRotation.Roll = 0;
// Convert the rotation back to world space, and set it as the current control rotation.
PlayerController->SetControlRotation(GetGravityWorldRotation(ViewRotation, GravityDirection));
}
}
FRotator UGravityInfluencedComponent::GetGravityRelativeRotation(FRotator Rotation, FVector GravityDirection)
{
if (!GravityDirection.Equals(FVector::DownVector))
{
FQuat GravityRotation = FQuat::FindBetweenNormals(GravityDirection, FVector::DownVector);
return (GravityRotation * Rotation.Quaternion()).Rotator();
}
return Rotation;
}
FRotator UGravityInfluencedComponent::GetGravityWorldRotation(FRotator Rotation, FVector GravityDirection)
{
if (!GravityDirection.Equals(FVector::DownVector))
{
FQuat GravityRotation = FQuat::FindBetweenNormals(FVector::DownVector, GravityDirection);
return (GravityRotation * Rotation.Quaternion()).Rotator();
}
return Rotation;
}
void UGravityInfluencedComponent::SetCurrentGravityField(UPrimitiveComponent* NewCurrentGravityField)
{
if (!CurrentGravityField)
{
PlayerCharacter->GetCapsuleComponent()->SetEnableGravity(true);
bUseGravity = true;
}
CurrentGravityField = NewCurrentGravityField;
if (CurrentGravityField)
{
bUpdateGravityFieldInTick = IGravityInterface::Execute_ShouldBeUpdatedInTick(CurrentGravityField);
}
else
{
PlayerCharacter->GetCapsuleComponent()->SetEnableGravity(false);
bUseGravity = false;
bUpdateGravityFieldInTick = false;
}
Server_SetUseGravity(bUseGravity);
}
UPrimitiveComponent* UGravityInfluencedComponent::GetCurrentGravityField()
{
return CurrentGravityField;
}
TArray<UPrimitiveComponent*>* UGravityInfluencedComponent::GetOverlappingGravityFields()
{
return &OverlappingGravityFields;
}
void UGravityInfluencedComponent::Server_SetUseGravity_Implementation(bool bNewUseGravity)
{
bUseGravity = bNewUseGravity;
}
void UGravityInfluencedComponent::Server_SetCurrentGravityDirection_Implementation(FVector NewCurrentGravityDirection)
{
CurrentGravityDirection = NewCurrentGravityDirection;
PlayerCharacter->GetCharacterMovement()->SetGravityDirection(CurrentGravityDirection);
}
void UGravityInfluencedComponent::OnRep_UseGravity()
{
if (!PlayerCharacter->IsLocallyControlled())
{
PlayerCharacter->GetCapsuleComponent()->SetEnableGravity(bUseGravity);
}
}
void UGravityInfluencedComponent::OnRep_CurrentGravityDirection()
{
if (!PlayerCharacter->IsLocallyControlled())
{
PlayerCharacter->GetCharacterMovement()->SetGravityDirection(CurrentGravityDirection);
}
}
Any idea about what I’m doing wrong or missing ?