// Fill out your copyright notice in the Description page of Project Settings. #include "BlasterCharacter.h" #include "Blaster/Blaster.h" #include "Blaster/Components/CombatComponent.h" #include "Blaster/GameMode/BlasterGameMode.h" #include "Blaster/PlayerController/BlasterPlayerController.h" #include "Blaster/PlayerState/BlasterPlayerState.h" #include "Blaster/Weapon/Weapon.h" #include "Camera/CameraComponent.h" #include "Components/CapsuleComponent.h" #include "Components/WidgetComponent.h" #include "GameFramework/CharacterMovementComponent.h" #include "GameFramework/SpringArmComponent.h" #include "Kismet/GameplayStatics.h" #include "Kismet/KismetMathLibrary.h" #include "Net/UnrealNetwork.h" #include "Particles/ParticleSystemComponent.h" ABlasterCharacter::ABlasterCharacter() { PrimaryActorTick.bCanEverTick = true; SpawnCollisionHandlingMethod = ESpawnActorCollisionHandlingMethod::AdjustIfPossibleButAlwaysSpawn; CameraBoom = CreateDefaultSubobject(TEXT("CameraBoom")); CameraBoom->SetupAttachment(GetMesh()); CameraBoom->TargetArmLength = 600.f; CameraBoom->bUsePawnControlRotation = true; FollowCamera = CreateDefaultSubobject(TEXT("FollowCamera")); FollowCamera->SetupAttachment(CameraBoom, USpringArmComponent::SocketName); FollowCamera->bUsePawnControlRotation = false; bUseControllerRotationYaw = false; GetCharacterMovement()->bOrientRotationToMovement = true; OverheadWidget = CreateDefaultSubobject(TEXT("OverheadWidget")); OverheadWidget->SetupAttachment(RootComponent); Combat = CreateDefaultSubobject(TEXT("CombatComponent")); Combat->SetIsReplicated(true); GetCharacterMovement()->NavAgentProps.bCanCrouch = true; GetCapsuleComponent()->SetCollisionResponseToChannel(ECollisionChannel::ECC_Camera, ECollisionResponse::ECR_Ignore); GetMesh()->SetCollisionObjectType(ECC_SkeletalMesh); GetMesh()->SetCollisionResponseToChannel(ECollisionChannel::ECC_Camera, ECollisionResponse::ECR_Ignore); GetMesh()->SetCollisionResponseToChannel(ECollisionChannel::ECC_Visibility, ECollisionResponse::ECR_Block); GetCharacterMovement()->RotationRate = FRotator(0.f, 850.f, 0.f); TurningInPlace = ETurningInPlace::ETIP_NotTurning; NetUpdateFrequency = 66.f; MinNetUpdateFrequency = 33.f; DissolveTimeline = CreateDefaultSubobject(TEXT("DissolveTimelineComponent")); } void ABlasterCharacter::GetLifetimeReplicatedProps(TArray& OutLifetimeProps) const { Super::GetLifetimeReplicatedProps(OutLifetimeProps); DOREPLIFETIME_CONDITION(ABlasterCharacter, OverlappingWeapon, COND_OwnerOnly); DOREPLIFETIME(ABlasterCharacter, Health); DOREPLIFETIME(ABlasterCharacter, bDisableGameplay); } void ABlasterCharacter::OnRep_ReplicatedMovement() { Super::OnRep_ReplicatedMovement(); SimProxiesTurn(); TimeSinceLastMovementReplication = 0.f; } void ABlasterCharacter::Eliminated() { if (Combat && Combat->EquippedWeapon) { Combat->EquippedWeapon->Dropped(); } MulticastEliminated(); GetWorldTimerManager().SetTimer( EliminationTimer, this, &ABlasterCharacter::EliminationTimerFinished, EliminationDelay ); } void ABlasterCharacter::MulticastEliminated_Implementation() { if (BlasterPlayerController) { BlasterPlayerController->SetHUDWeaponAmmo(0); } bEliminated = true; PlayEliminatedMontage(); // Start dissolve effect if (DissolveMaterialInstance) { DynamicDissolveMaterialInstance = UMaterialInstanceDynamic::Create(DissolveMaterialInstance, this); GetMesh()->SetMaterial(0, DynamicDissolveMaterialInstance); DynamicDissolveMaterialInstance->SetScalarParameterValue(TEXT("Dissolve"), 0.55f); DynamicDissolveMaterialInstance->SetScalarParameterValue(TEXT("Glow"), 200.f); } StartDissolve(); // Disable character movement GetCharacterMovement()->DisableMovement(); GetCharacterMovement()->StopMovementImmediately(); bDisableGameplay = true; if (Combat) { Combat->FireButtonPressed(false); } // Disable collision GetCapsuleComponent()->SetCollisionEnabled(ECollisionEnabled::NoCollision); GetMesh()->SetCollisionEnabled(ECollisionEnabled::NoCollision); // Elimination bot if (EliminationBotEffect) { const FVector EliminationBotSpawnPoint(GetActorLocation().X, GetActorLocation().Y, GetActorLocation().Z + 200.f); EliminationBotComponent = UGameplayStatics::SpawnEmitterAtLocation( GetWorld(), EliminationBotEffect, EliminationBotSpawnPoint, GetActorRotation() ); } if (EliminationBotSound) { UGameplayStatics::SpawnSoundAtLocation( this, EliminationBotSound, GetActorLocation() ); } if (IsLocallyControlled() && GetEquippedWeapon()->GetWeaponType() == EWeaponType::EWT_SniperRifle && IsAiming()) { ShowSniperScopeWidget(false); } } void ABlasterCharacter::EliminationTimerFinished() { ABlasterGameMode* BlasterGameMode = GetWorld()->GetAuthGameMode(); if (BlasterGameMode) { BlasterGameMode->RequestRespawn(this, Controller); } } void ABlasterCharacter::Destroyed() { Super::Destroyed(); if (EliminationBotComponent) { EliminationBotComponent->DestroyComponent(); } ABlasterGameMode* BlasterGameMode = Cast(UGameplayStatics::GetGameMode(this)); bool bMatchNotInProgress = BlasterGameMode && BlasterGameMode->GetMatchState() != MatchState::InProgress; if (Combat && Combat->EquippedWeapon && bMatchNotInProgress) { Combat->EquippedWeapon->Destroy(); } } void ABlasterCharacter::BeginPlay() { Super::BeginPlay(); UpdateHUDHealth(); if (HasAuthority()) { OnTakeAnyDamage.AddDynamic(this, &ABlasterCharacter::ReceiveDamage); } } void ABlasterCharacter::Tick(float DeltaTime) { Super::Tick(DeltaTime); RotateInPlace(DeltaTime); HideCameraIfCharacterClose(); PollInit(); } void ABlasterCharacter::RotateInPlace(float DeltaTime) { if (bDisableGameplay) { bUseControllerRotationYaw = false; TurningInPlace = ETurningInPlace::ETIP_NotTurning; return; } if (GetLocalRole() > ENetRole::ROLE_SimulatedProxy && IsLocallyControlled()) { AimOffset(DeltaTime); } else { TimeSinceLastMovementReplication += DeltaTime; if (TimeSinceLastMovementReplication > 0.25f) { OnRep_ReplicatedMovement(); } CalculateAO_Pitch(); } } void ABlasterCharacter::SetupPlayerInputComponent(UInputComponent* PlayerInputComponent) { Super::SetupPlayerInputComponent(PlayerInputComponent); PlayerInputComponent->BindAction("Jump", IE_Pressed, this, &ABlasterCharacter::Jump); PlayerInputComponent->BindAxis("MoveForward", this, &ABlasterCharacter::MoveForward); PlayerInputComponent->BindAxis("MoveRight", this, &ABlasterCharacter::MoveRight); PlayerInputComponent->BindAxis("Turn", this, &ABlasterCharacter::Turn); PlayerInputComponent->BindAxis("LookUp", this, &ABlasterCharacter::LookUp); PlayerInputComponent->BindAction("Equip", IE_Pressed, this, &ABlasterCharacter::EquipButtonPressed); PlayerInputComponent->BindAction("Crouch", IE_Pressed, this, &ABlasterCharacter::CrouchButtonPressed); PlayerInputComponent->BindAction("Aim", IE_Pressed, this, &ABlasterCharacter::AimButtonPressed); PlayerInputComponent->BindAction("Aim", IE_Released, this, &ABlasterCharacter::AimButtonReleased); PlayerInputComponent->BindAction("Fire", IE_Pressed, this, &ABlasterCharacter::FireButtonPressed); PlayerInputComponent->BindAction("Fire", IE_Released, this, &ABlasterCharacter::FireButtonReleased); PlayerInputComponent->BindAction("Reload", IE_Pressed, this, &ABlasterCharacter::ReloadButtonPressed); } void ABlasterCharacter::PostInitializeComponents() { Super::PostInitializeComponents(); if (Combat) { Combat->Character = this; } } void ABlasterCharacter::PlayFireMontage(bool bAiming) { if (Combat == nullptr || Combat->EquippedWeapon == nullptr) return; UAnimInstance* AnimInstance = GetMesh()->GetAnimInstance(); if (AnimInstance && FireWeaponMontage) { AnimInstance->Montage_Play(FireWeaponMontage); FName SectionName; SectionName = bAiming ? FName("RifleADS") : FName("RifleHip"); AnimInstance->Montage_JumpToSection(SectionName); } } void ABlasterCharacter::PlayReloadMontage() { if (Combat == nullptr || Combat->EquippedWeapon == nullptr) return; UAnimInstance* AnimInstance = GetMesh()->GetAnimInstance(); if (AnimInstance && ReloadMontage) { AnimInstance->Montage_Play(ReloadMontage); FName SectionName; switch (Combat->EquippedWeapon->GetWeaponType()) { case EWeaponType::EWT_AssaultRifle: SectionName = FName("Rifle"); break; case EWeaponType::EWT_RocketLauncher: SectionName = FName("Rifle"); // Todo: create rocket reload montage section break; case EWeaponType::EWT_Pistol: SectionName = FName("Rifle"); // Todo: create pistol reload montage section break; case EWeaponType::EWT_SubmachineGun: SectionName = FName("Rifle"); // Todo: create SMG reload montage section break; case EWeaponType::EWT_Shotgun: SectionName = FName("Rifle"); // Todo: create Shotgun reload montage section break; case EWeaponType::EWT_SniperRifle: SectionName = FName("Rifle"); // Todo: create Sniper reload montage section break; case EWeaponType::EWT_GrenadeLauncher: SectionName = FName("Rifle"); // Todo: create Grenade Launcher reload montage section break; default: SectionName = FName("Rifle"); break; } AnimInstance->Montage_JumpToSection(SectionName); } } void ABlasterCharacter::PlayEliminatedMontage() { UAnimInstance* AnimInstance = GetMesh()->GetAnimInstance(); if (AnimInstance && EliminatedMontage) { AnimInstance->Montage_Play(EliminatedMontage); } } void ABlasterCharacter::PlayHitReactMontage() { if (Combat == nullptr || Combat->EquippedWeapon == nullptr) return; UAnimInstance* AnimInstance = GetMesh()->GetAnimInstance(); if (AnimInstance && HitReactMontage) { AnimInstance->Montage_Play(HitReactMontage); FName SectionName("FromFront"); AnimInstance->Montage_JumpToSection(SectionName); } } void ABlasterCharacter::ReceiveDamage(AActor* DamagedActor, float Damage, const UDamageType* DamageType, AController* InstigatorController, AActor* DamageCauser) { Health = FMath::Clamp(Health - Damage, 0.f, MaxHealth); UpdateHUDHealth(); if (Health > 0.f) { PlayHitReactMontage(); } if (Health == 0.f) { ABlasterGameMode* BlasterGameMode = GetWorld()->GetAuthGameMode(); if (BlasterGameMode) { BlasterPlayerController = BlasterPlayerController == nullptr ? Cast(Controller) : BlasterPlayerController; ABlasterPlayerController* AttackerController = Cast(InstigatorController); BlasterGameMode->PlayerEliminated(this, BlasterPlayerController, AttackerController); } } } void ABlasterCharacter::MoveForward(float Value) { if (bDisableGameplay) return; if (Controller != nullptr && Value != 0.f) { const FRotator YawRotation(0.f, Controller->GetControlRotation().Yaw, 0.f); const FVector Direction(FRotationMatrix(YawRotation).GetUnitAxis(EAxis::X)); AddMovementInput(Direction, Value); } } void ABlasterCharacter::MoveRight(float Value) { if (bDisableGameplay) return; if (Controller != nullptr && Value != 0.f) { const FRotator YawRotation(0.f, Controller->GetControlRotation().Yaw, 0.f); const FVector Direction(FRotationMatrix(YawRotation).GetUnitAxis(EAxis::Y)); AddMovementInput(Direction, Value); } } void ABlasterCharacter::Turn(float Value) { AddControllerYawInput(Value); } void ABlasterCharacter::LookUp(float Value) { AddControllerPitchInput(Value); } void ABlasterCharacter::EquipButtonPressed() { if (bDisableGameplay) return; if (Combat) { if (HasAuthority()) { Combat->EquipWeapon(OverlappingWeapon); } else { ServerEquipButtonPressed(); } } } void ABlasterCharacter::ServerEquipButtonPressed_Implementation() { EquipButtonPressed(); } void ABlasterCharacter::CrouchButtonPressed() { if (bDisableGameplay) return; if (bIsCrouched) { UnCrouch(); } else { Crouch(); } } void ABlasterCharacter::ReloadButtonPressed() { if (bDisableGameplay) return; if (Combat) { Combat->Reload(); } } void ABlasterCharacter::AimButtonPressed() { if (bDisableGameplay) return; if (Combat) { Combat->SetAiming(true); } } void ABlasterCharacter::AimButtonReleased() { if (bDisableGameplay) return; if (Combat) { Combat->SetAiming(false); } } float ABlasterCharacter::CalculateSpeed() { FVector Velocity = GetVelocity(); Velocity.Z = 0.f; return Velocity.Size(); } void ABlasterCharacter::AimOffset(float DeltaTime) { if (Combat && Combat->EquippedWeapon == nullptr) return; float Speed = CalculateSpeed(); bool bIsInAir = GetCharacterMovement()->IsFalling(); if (Speed == 0.f && !bIsInAir) // Standing still, not jumping { bRotateRootBone = true; FRotator CurrentAimRotation = FRotator(0.f, GetBaseAimRotation().Yaw, 0.f); FRotator DeltaAimRotation = UKismetMathLibrary::NormalizedDeltaRotator(CurrentAimRotation, StartingAimRotation); AO_Yaw = DeltaAimRotation.Yaw; if (TurningInPlace == ETurningInPlace::ETIP_NotTurning) { InterpAO_Yaw = AO_Yaw; } bUseControllerRotationYaw = true; TurnInPlace(DeltaTime); } if (Speed > 0.f || bIsInAir) // Running or jumping { bRotateRootBone = false; StartingAimRotation = FRotator(0.f, GetBaseAimRotation().Yaw, 0.f); AO_Yaw = 0.f; bUseControllerRotationYaw = true; TurningInPlace = ETurningInPlace::ETIP_NotTurning; } CalculateAO_Pitch(); } void ABlasterCharacter::CalculateAO_Pitch() { AO_Pitch = GetBaseAimRotation().Pitch; if (AO_Pitch > 90.f && !IsLocallyControlled()) { // map pitch from [270, 360) to [-90, 0) FVector2D InRange(270.f, 360.f); FVector2D OutRange(-90.f, 0.f); AO_Pitch = FMath::GetMappedRangeValueClamped(InRange, OutRange, AO_Pitch); } } void ABlasterCharacter::SimProxiesTurn() { if (Combat == nullptr || Combat->EquippedWeapon == nullptr) return; bRotateRootBone = false; float Speed = CalculateSpeed(); if (Speed > 0.f) { TurningInPlace = ETurningInPlace::ETIP_NotTurning; return; } ProxyRotationLastFrame = ProxyRotation; ProxyRotation = GetActorRotation(); ProxyYaw = UKismetMathLibrary::NormalizedDeltaRotator(ProxyRotation, ProxyRotationLastFrame).Yaw; if (FMath::Abs(ProxyYaw) > TurnThreshold) { if (ProxyYaw > TurnThreshold) { TurningInPlace = ETurningInPlace::ETIP_Right; } else if (ProxyYaw < -TurnThreshold) { TurningInPlace = ETurningInPlace::ETIP_Left; } else { TurningInPlace = ETurningInPlace::ETIP_NotTurning; } return; } TurningInPlace = ETurningInPlace::ETIP_NotTurning; } void ABlasterCharacter::Jump() { if (bDisableGameplay) return; if (bIsCrouched) { UnCrouch(); } else { Super::Jump(); } } void ABlasterCharacter::FireButtonPressed() { if (bDisableGameplay) return; if (Combat) { Combat->FireButtonPressed(true); } } void ABlasterCharacter::FireButtonReleased() { if (bDisableGameplay) return; if (Combat) { Combat->FireButtonPressed(false); } } void ABlasterCharacter::TurnInPlace(float DeltaTime) { if (AO_Yaw > 90.f) { TurningInPlace = ETurningInPlace::ETIP_Right; } else if (AO_Yaw < -90.f) { TurningInPlace = ETurningInPlace::ETIP_Left; } if (TurningInPlace != ETurningInPlace::ETIP_NotTurning) { InterpAO_Yaw = FMath::FInterpTo(InterpAO_Yaw, 0.f, DeltaTime, 4.f); AO_Yaw = InterpAO_Yaw; if (FMath::Abs(AO_Yaw) < 15.f) { TurningInPlace = ETurningInPlace::ETIP_NotTurning; StartingAimRotation = FRotator(0.f, GetBaseAimRotation().Yaw, 0.f); } } } void ABlasterCharacter::HideCameraIfCharacterClose() { if (!IsLocallyControlled()) return; if ((FollowCamera->GetComponentLocation() - GetActorLocation()).Size() < CameraThreshold) { GetMesh()->SetVisibility(false); if (Combat && Combat->EquippedWeapon && Combat->EquippedWeapon->GetWeaponMesh()) { Combat->EquippedWeapon->GetWeaponMesh()->bOwnerNoSee = true; } } else { GetMesh()->SetVisibility(true); if (Combat && Combat->EquippedWeapon && Combat->EquippedWeapon->GetWeaponMesh()) { Combat->EquippedWeapon->GetWeaponMesh()->bOwnerNoSee = false; } } } void ABlasterCharacter::OnRep_Health() { UpdateHUDHealth(); if (Health > 0.f) { PlayHitReactMontage(); } } void ABlasterCharacter::UpdateHUDHealth() { BlasterPlayerController = BlasterPlayerController == nullptr ? Cast(Controller) : BlasterPlayerController; if (BlasterPlayerController) { BlasterPlayerController->SetHUDHealth(Health, MaxHealth); } } void ABlasterCharacter::PollInit() { if (BlasterPlayerState == nullptr) { BlasterPlayerState = GetPlayerState(); if (BlasterPlayerState) { // Initialize Score now we have the PlayerState BlasterPlayerState->IncreaseScore(0.f); BlasterPlayerState->IncreaseDefeats(0); } } } void ABlasterCharacter::UpdateDissolveMaterial(float DissolveValue) { if (DynamicDissolveMaterialInstance) { DynamicDissolveMaterialInstance->SetScalarParameterValue(TEXT("Dissolve"), DissolveValue); } } void ABlasterCharacter::StartDissolve() { DissolveTrack.BindDynamic(this, &ABlasterCharacter::UpdateDissolveMaterial); if (DissolveCurve && DissolveTimeline) { DissolveTimeline->AddInterpFloat(DissolveCurve, DissolveTrack); DissolveTimeline->Play(); } } void ABlasterCharacter::SetOverlappingWeapon(AWeapon* Weapon) { if (OverlappingWeapon) { OverlappingWeapon->ShowPickupWidget(false); } OverlappingWeapon = Weapon; if (IsLocallyControlled()) { if (OverlappingWeapon) { OverlappingWeapon->ShowPickupWidget(true); } } } void ABlasterCharacter::OnRep_OverlappingWeapon(AWeapon* LastWeapon) { if (OverlappingWeapon) { OverlappingWeapon->ShowPickupWidget(true); } if (LastWeapon) { LastWeapon->ShowPickupWidget(false); } } bool ABlasterCharacter::IsWeaponEquipped() { return Combat && Combat->EquippedWeapon; } bool ABlasterCharacter::IsAiming() { return Combat && Combat->bAiming; } AWeapon* ABlasterCharacter::GetEquippedWeapon() { if (Combat == nullptr) return nullptr; return Combat->EquippedWeapon; } FVector ABlasterCharacter::GetHitTarget() const { if (Combat == nullptr) return FVector(); return Combat->HitTarget; } ECombatState ABlasterCharacter::GetCombatState() const { if (Combat == nullptr) return ECombatState::ECS_MAX; return Combat->CombatState; }