// Fill out your copyright notice in the Description page of Project Settings. #include "LagCompensationComponent.h" #include "Blaster/Weapon/Weapon.h" #include "Components/BoxComponent.h" #include "Kismet/GameplayStatics.h" ULagCompensationComponent::ULagCompensationComponent() { PrimaryComponentTick.bCanEverTick = true; } void ULagCompensationComponent::BeginPlay() { Super::BeginPlay(); } void ULagCompensationComponent::ServerScoreRequest_Implementation(ABlasterCharacter* HitCharacter, const FVector_NetQuantize& TraceStart, const FVector_NetQuantize& HitLocation, float HitTime, AWeapon* DamageCauser) { const FServerSideRewindResult Confirm = ServerSideRewind(HitCharacter, TraceStart, HitLocation, HitTime); if (Character && HitCharacter && DamageCauser && Confirm.bHitConfirmed) { UGameplayStatics::ApplyDamage( HitCharacter, DamageCauser->GetDamage(), Character->Controller, DamageCauser, UDamageType::StaticClass() ); } } void ULagCompensationComponent::SaveFramePackage(FFramePackage& Package) { Character = Character == nullptr ? Cast(GetOwner()) : Character; if (Character) { Package.Time = GetWorld()->GetTimeSeconds(); for (auto& BoxPair : Character->HitCollisionBoxes) { FBoxInformation BoxInformation; BoxInformation.Location = BoxPair.Value->GetComponentLocation(); BoxInformation.Rotation = BoxPair.Value->GetComponentRotation(); BoxInformation.BoxExtend = BoxPair.Value->GetScaledBoxExtent(); Package.HitBoxInfo.Add(BoxPair.Key, BoxInformation); } } } FFramePackage ULagCompensationComponent::InterpBetweenFrames(const FFramePackage& OlderFrame, const FFramePackage& YoungerFrame, float HitTime) { const float Distance = YoungerFrame.Time - OlderFrame.Time; const float InterpFraction = FMath::Clamp((HitTime - OlderFrame.Time) / Distance, 0.f, 1.f); FFramePackage InterpFramePackage; InterpFramePackage.Time = HitTime; for (auto& YoungerPair : YoungerFrame.HitBoxInfo) { const FName& BoxInfoName = YoungerPair.Key; const FBoxInformation& OlderBox = OlderFrame.HitBoxInfo[BoxInfoName]; const FBoxInformation& YoungerBox = YoungerFrame.HitBoxInfo[BoxInfoName]; FBoxInformation InterpBoxInfo; InterpBoxInfo.Location = FMath::VInterpTo(OlderBox.Location, YoungerBox.Location, 1.f, InterpFraction); InterpBoxInfo.Rotation = FMath::RInterpTo(OlderBox.Rotation, YoungerBox.Rotation, 1.f, InterpFraction); InterpBoxInfo.BoxExtend = OlderBox.BoxExtend; InterpFramePackage.HitBoxInfo.Add(BoxInfoName, InterpBoxInfo); } return InterpFramePackage; } FServerSideRewindResult ULagCompensationComponent::ConfirmHit(const FFramePackage& Package, ABlasterCharacter* HitCharacter, const FVector_NetQuantize& TraceStart, const FVector_NetQuantize HitLocation) { if (HitCharacter == nullptr) return FServerSideRewindResult(); FFramePackage CurrentFrame; CacheBoxPositions(HitCharacter, CurrentFrame); MoveBoxes(HitCharacter, Package); EnableCharacterMeshCollision(HitCharacter, ECollisionEnabled::NoCollision); // Enable collision for the head first UBoxComponent* HeadBox = HitCharacter->HitCollisionBoxes[FName("head")]; HeadBox->SetCollisionEnabled(ECollisionEnabled::QueryOnly); HeadBox->SetCollisionResponseToChannel(ECC_Visibility, ECR_Block); FHitResult ConfirmHitResult; const FVector TraceEnd = TraceStart + (HitLocation - TraceStart) * 1.25f; UWorld* World = GetWorld(); if (World) { World->LineTraceSingleByChannel( ConfirmHitResult, TraceStart, TraceEnd, ECC_Visibility ); if (ConfirmHitResult.bBlockingHit) // we hit the head, return early { ResetHitBoxes(HitCharacter, CurrentFrame); EnableCharacterMeshCollision(HitCharacter, ECollisionEnabled::QueryAndPhysics); return FServerSideRewindResult{ true, true }; } // Didn't hit the head, check the rest of the boxes for (auto& HitBoxPair : HitCharacter->HitCollisionBoxes) { if (HitBoxPair.Value != nullptr) { HitBoxPair.Value->SetCollisionEnabled(ECollisionEnabled::QueryOnly); HitBoxPair.Value->SetCollisionResponseToChannel(ECC_Visibility, ECR_Block); } } World->LineTraceSingleByChannel( ConfirmHitResult, TraceStart, TraceEnd, ECC_Visibility ); if (ConfirmHitResult.bBlockingHit) { ResetHitBoxes(HitCharacter, CurrentFrame); EnableCharacterMeshCollision(HitCharacter, ECollisionEnabled::QueryAndPhysics); return FServerSideRewindResult{ true, false }; } } ResetHitBoxes(HitCharacter, CurrentFrame); EnableCharacterMeshCollision(HitCharacter, ECollisionEnabled::QueryAndPhysics); return FServerSideRewindResult{ false, false }; } void ULagCompensationComponent::CacheBoxPositions(ABlasterCharacter* HitCharacter, FFramePackage& OutFramePackage) { if (HitCharacter == nullptr) return; for (auto& HitBoxPair : HitCharacter->HitCollisionBoxes) { if (HitBoxPair.Value != nullptr) { FBoxInformation BoxInfo; BoxInfo.Location = HitBoxPair.Value->GetComponentLocation(); BoxInfo.Rotation = HitBoxPair.Value->GetComponentRotation(); BoxInfo.BoxExtend = HitBoxPair.Value->GetScaledBoxExtent(); OutFramePackage.HitBoxInfo.Add(HitBoxPair.Key, BoxInfo); } } } void ULagCompensationComponent::MoveBoxes(ABlasterCharacter* HitCharacter, const FFramePackage& Package) { if (HitCharacter == nullptr) return; for (auto& HitBoxPair : HitCharacter->HitCollisionBoxes) { if (HitBoxPair.Value != nullptr) { HitBoxPair.Value->SetWorldLocation(Package.HitBoxInfo[HitBoxPair.Key].Location); HitBoxPair.Value->SetWorldRotation(Package.HitBoxInfo[HitBoxPair.Key].Rotation); HitBoxPair.Value->SetBoxExtent(Package.HitBoxInfo[HitBoxPair.Key].BoxExtend); } } } void ULagCompensationComponent::ResetHitBoxes(ABlasterCharacter* HitCharacter, const FFramePackage& Package) { if (HitCharacter == nullptr) return; for (auto& HitBoxPair : HitCharacter->HitCollisionBoxes) { if (HitBoxPair.Value != nullptr) { HitBoxPair.Value->SetWorldLocation(Package.HitBoxInfo[HitBoxPair.Key].Location); HitBoxPair.Value->SetWorldRotation(Package.HitBoxInfo[HitBoxPair.Key].Rotation); HitBoxPair.Value->SetBoxExtent(Package.HitBoxInfo[HitBoxPair.Key].BoxExtend); HitBoxPair.Value->SetCollisionEnabled(ECollisionEnabled::NoCollision); } } } void ULagCompensationComponent::ShowFramePackage(const FFramePackage& Package, const FColor Color) { for (auto& BoxInfo : Package.HitBoxInfo) { DrawDebugBox( GetWorld(), BoxInfo.Value.Location, BoxInfo.Value.BoxExtend, FQuat(BoxInfo.Value.Rotation), Color, false, 4.f ); } } FServerSideRewindResult ULagCompensationComponent::ServerSideRewind(ABlasterCharacter* HitCharacter, const FVector_NetQuantize& TraceStart, const FVector_NetQuantize& HitLocation, float HitTime) { bool bReturn = HitCharacter == nullptr || HitCharacter->GetLagCompensation() == nullptr || HitCharacter->GetLagCompensation()->FrameHistory.GetHead() == nullptr || HitCharacter->GetLagCompensation()->FrameHistory.GetTail() == nullptr; if (bReturn) FServerSideRewindResult(); // Frame package that we check to verify a hit FFramePackage FrameToCheck; bool bShouldInterpolate = true; const TDoubleLinkedList& History = HitCharacter->GetLagCompensation()->FrameHistory; const float OldestHistoryTime = History.GetTail()->GetValue().Time; const float NewestHistoryTime = History.GetHead()->GetValue().Time; if (OldestHistoryTime > HitTime) { // Too far back, too laggy to do SSR FServerSideRewindResult(); } if (OldestHistoryTime == HitTime) { FrameToCheck = History.GetTail()->GetValue(); bShouldInterpolate = false; } if (NewestHistoryTime <= HitTime) { FrameToCheck = History.GetHead()->GetValue(); bShouldInterpolate = false; } auto Younger = History.GetHead(); auto Older = History.GetHead(); while (Older->GetValue().Time > HitTime) { if (Older->GetNextNode() == nullptr) break; Older = Older->GetNextNode(); if (Older->GetValue().Time > HitTime) { Younger = Older; } } if (Older->GetValue().Time == HitTime) { FrameToCheck = Older->GetValue(); bShouldInterpolate = false; } if (bShouldInterpolate) // Why not just check if FrameToCheck == nullptr? { // Interpolate between Younger and Older FrameToCheck = InterpBetweenFrames(Older->GetValue(), Younger->GetValue(), HitTime); } return ConfirmHit(FrameToCheck, HitCharacter, TraceStart, HitLocation); } void ULagCompensationComponent::EnableCharacterMeshCollision(ABlasterCharacter* HitCharacter, ECollisionEnabled::Type CollisionEnabled) { if (HitCharacter && HitCharacter->GetMesh()) { HitCharacter->GetMesh()->SetCollisionEnabled(CollisionEnabled); } } void ULagCompensationComponent::TickComponent(float DeltaTime, ELevelTick TickType, FActorComponentTickFunction* ThisTickFunction) { Super::TickComponent(DeltaTime, TickType, ThisTickFunction); SaveFramePackage(); } void ULagCompensationComponent::SaveFramePackage() { if (Character == nullptr || !Character->HasAuthority()) return; if (FrameHistory.Num() <= 1) { FFramePackage ThisFrame; SaveFramePackage(ThisFrame); FrameHistory.AddHead(ThisFrame); } else { float HistoryLength = FrameHistory.GetHead()->GetValue().Time - FrameHistory.GetTail()->GetValue().Time; while (HistoryLength > MaxRecordTime) { FrameHistory.RemoveNode(FrameHistory.GetTail()); HistoryLength = FrameHistory.GetHead()->GetValue().Time - FrameHistory.GetTail()->GetValue().Time; } FFramePackage ThisFrame; SaveFramePackage(ThisFrame); FrameHistory.AddHead(ThisFrame); // ShowFramePackage(ThisFrame, FColor::Red); } }