본문 바로가기

레퍼런스

데미지 프레임워크

www.unrealengine.com/ko/blog/damage-in-ue4

 

blog.naver.com/destiny9720/220914505079

 

데미지 프레임워크에 대해서

유니티에는 데미지 프레임워크가 없어서 내 맘대로 구조체를 만들어서 사용했었다. ㅜㅜ

언리얼은 데미지를 주고받는 함수가 미리 구현되어있다.

 

엄청나게 편하다고 느꼈다.

 

데미지 프레임워크의 종류

 

데미지를 주는 함수의 종류

ApplyDamage : 기본적인 데미지 함수.

ApplyPointDamage : 데미지를 주는 위치의 정보도 얻어올 수 있는 데미지 함수. (어디서 데미지를 주었는지.)

ApplyRadialDamage : 범위 데미지를 줄 때 사용. (Origin으로부터 Radius만큼 데미지를 뿌린다.라는 느낌.)

 

데미지를 받는 함수의 종류

블루프린트 : AnyDamage 노드, PointDamage 노드, RadialDamage 노드. (쓰기 쉬우라고 구분되어 있음.)

C++ : Actor 클래스의 TakeDamage 함수 오버라이딩. (함수 내에서 구분해서 처리함.)

float ACPP_Player::TakeDamage(float DamageAmount, FDamageEvent const& DamageEvent, AController* EventInstigator, AActor* DamageCauser)
{
	float Damage = Super::TakeDamage(DamageAmount, DamageEvent, EventInstigator, DamageCauser);

	// ㅡㅡㅡDamageEvent를 이용해 구분을 짓는다.ㅡㅡㅡ
    
	// PointDamage 받기.
	if (DamageEvent.IsOfType(FPointDamageEvent::ClassID))
	{
		const FPointDamageEvent* PointDamageEvent = static_cast<const FPointDamageEvent*>(&DamageEvent);
		if (0 == (PointDamageEvent->HitInfo.BoneName).Compare(FName(TEXT("Head"))))
		{
			Damage *= 5; // 맞은 부위가 Head면, 데미지 5배.
		}
	}
	// RadialDamage 받기.
	else if (DamageEvent.IsOfType(FRadialDamageEvent::ClassID)) 
	{
		const FRadialDamageEvent* RadialDamageEvent = static_cast<const FRadialDamageEvent*>(&DamageEvent);
	}
    
    
	CurrentHP -= Damage;
	
	return Damage;
}

 

※ 참고

BP와 C++에서 사용하는 방법이 조금 달라서,

블루프린트 노드를 보고 C++로 옮기는 작업을 할 때 당황할 수도 있다.

 

AnyDamage 노드는 BP에서만 사용 가능하다.

Actor 클래스에 가보면, AnyDamage 함수에 BlueprintImplementableEvent가 선언되어있다.

/** Event when this actor takes ANY damage */
UFUNCTION(BlueprintImplementableEvent, BlueprintAuthorityOnly, meta=(DisplayName = "AnyDamage"), Category="Game|Damage")
void ReceiveAnyDamage(float Damage, const class UDamageType* DamageType, class AController* InstigatedBy, AActor* DamageCauser);

선언의 의미는

C++ : 함수 헤더를 작성.

BP : 함수를 오버라이딩해서 구현 부분을 작성.

 

그래서 C++에서는 AnyDamage 함수를 구현하지 못하기 때문에, 사용할 수 없다.

 

BP로 구현

각각의 노드를 사용하면 된다.

 

C++로 구현

UGameplayStatics의 ApplyDamage 함수로 데미지를 주고,

Actor 클래스의 TakeDamage 함수를 오버라이드해서 데미지를 받는다.

// 오버랩된 오브젝트들에게 데미지 주기.

TArray<TEnumAsByte<EObjectTypeQuery>> ObjectTypes;
TArray<AActor*> IgnoreActors;
TArray<AActor*> OutActors;

ObjectTypes.Add(UEngineTypes::ConvertToObjectType(ECollisionChannel::ECC_Pawn));
IgnoreActors.Add(this);

bool isOverlapped = UKismetSystemLibrary::SphereOverlapActors(GetWorld(), Pos, Radius, ObjectTypes, nullptr, IgnoreActors, OutActors);
if (isOverlapped)
{
	for (int i = 0; i < OutActors.Num(); i++)
	{
		UE_LOG(LogClass, Warning, TEXT("ApplyDamage"));
		UGameplayStatics::ApplyDamage(OutActors[i], Damage, GetController(), nullptr, NULL);
	}
}
// 데미지 받기.

// 헤더파일
virtual float TakeDamage(float DamageAmount, struct FDamageEvent const& DamageEvent, class AController* EventInstigator, AActor* DamageCauser) override;

// CPP파일
float ACPP_Bear::TakeDamage(float DamageAmount, FDamageEvent const& DamageEvent, AController* EventInstigator, AActor* DamageCauser)
{
	float Damage = Super::TakeDamage(DamageAmount, DamageEvent, EventInstigator, DamageCauser);

	UE_LOG(LogClass, Warning, TEXT("Damage : %f"), Damage);
	
	return Damage;
}

'레퍼런스' 카테고리의 다른 글

designerd.tistory  (0) 2024.03.29
비헤이버트리 AI Perception  (0) 2024.02.27
SetViewPort 카메라 전환하기  (0) 2024.01.07
Montage Notify Interface  (0) 2023.12.31
믹사모 컨버터  (0) 2023.12.30