데미지를 주는 함수의 종류
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로 구현
※ 참고
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++로 구현
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;
}
[Unreal BP & C++] 엔진에서 제공하는 데미지 프레임워크 — 민규야 개발하자 (tistory.com)
'게임엔진 > Unreal' 카테고리의 다른 글
[Unreal] Reflection (리플렉션) (0) | 2023.12.04 |
---|---|
[Unreal] 비동기 에셋 로딩 (0) | 2023.12.04 |
[Unreal] 미니맵 및 캐릭터 위치 표시 (0) | 2023.11.15 |
[Unreal] Programming Subsystem (프로그래밍 서브시스템) (0) | 2023.11.14 |
[Unreal] Level Blueprint 를 C++로 접근하기 (0) | 2023.11.06 |