最简单的 UE 4 C++ 教程 —— 给 Actor 添加径向推力【十八】

2023-10-16 21:30

本文主要是介绍最简单的 UE 4 C++ 教程 —— 给 Actor 添加径向推力【十八】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原教程是基于 UE 4.18,我是基于 UE 4.25】

英文原地址

接上一节教程,本教程非常有趣,我们将通过在设定范围内的所有物体上添加径向推力来模拟爆炸。

创建一个名为 AddRadialForce 的新角色。我们不需要对头文件做任何操作。下面是由虚幻生成的默认头文件。

AddRadialForce.h

#pragma once#include "CoreMinimal.h"
#include "GameFramework/Actor.h"
#include "AddRadialForce.generated.h"UCLASS()
class UNREALCPP_API AAddRadialForce : public AActor
{GENERATED_BODY()public:	// Sets default values for this actor's propertiesAAddRadialForce();protected:// Called when the game starts or when spawnedvirtual void BeginPlay() override;public:	// Called every framevirtual void Tick(float DeltaTime) override;};

为了在调试过程中提供帮助,让我们将 DrawDebugHelpers 头文件添加到代码中。

#include "AddRadialForce.h"
// add debug helpfers
#include "DrawDebugHelpers.h"

在这个例子中,我们将在 BeginPlay() 函数中执行所有的逻辑。我们想要收集我们范围内的所有命中结果,并从范围的扫描中获得结果。为此,我们将使用 TArray 跟踪重叠的 actor。

void AAddRadialForce::BeginPlay()
{Super::BeginPlay();// create tarray for hit resultsTArray<FHitResult> OutHits;// crate tarray for sweep actorsTArray<AActor*> SweepActors;}

接下来我们要声明 TArray 数组 OutHits 。我们希望扫描范围从 actor 的位置开始同时也从该位置结束,并让 CollisionShape 是一个 500 单位的球体。您可以使用 GetActorLocation() 来获取 actor 的位置,它返回一个向量。我们使用 FCollisionShape:: makephere (500.0f) 来创建一个CollisionShape

// Called when the game starts or when spawned
void AAddRadialForce::BeginPlay()
{Super::BeginPlay();// create tarray for hit resultsTArray<FHitResult> OutHits;// get actor locationsFVector MyLocation = GetActorLocation();// start and end locations. The sphere will create the radial sweep.FVector Start = MyLocation;FVector End = MyLocation;// create a collision sphereFCollisionShape MyColSphere = FCollisionShape::MakeSphere(500.0f);}

为了可视化扫描的球体,我们将绘制一个调试球体。

// Called when the game starts or when spawned
void AAddRadialForce::BeginPlay()
{Super::BeginPlay();// create tarray for hit resultsTArray<FHitResult> OutHits;// get actor locationsFVector MyLocation = GetActorLocation();// start and end locations. The sphere will create the radial sweep.FVector Start = MyLocation;FVector End = MyLocation;// create a collision sphereFCollisionShape MyColSphere = FCollisionShape::MakeSphere(500.0f);// draw collision sphereDrawDebugSphere(GetWorld(), GetActorLocation(), MyColSphere.GetSphereRadius(), 50, FColor::Cyan, true);
}

接下来我们要检查我们的 actor 在我们 BeginPlay() 时是否击中了任何东西。每个 actor 都有GetWorld 功能。从 ,我们将使用从 GetWorld() 函数得到的 SweepMultiByChannel() 函数并对其设置上面刚创建的变量作为参数。这将返回一个 bool,指示是否有其他 actor 在该 actor 的范围内。

void AAddRadialForce::BeginPlay()
{Super::BeginPlay();// create tarray for hit resultsTArray<FHitResult> OutHits;// get actor locationsFVector MyLocation = GetActorLocation();// start and end locations. The sphere will create the radial sweep.FVector Start = MyLocation;FVector End = MyLocation;// create a collision sphereFCollisionShape MyColSphere = FCollisionShape::MakeSphere(500.0f);// draw collision sphereDrawDebugSphere(GetWorld(), GetActorLocation(), MyColSphere.GetSphereRadius(), 50, FColor::Cyan, true);// check if something got hit in the sweepbool isHit = GetWorld()->SweepMultiByChannel(OutHits, Start, End, FQuat::Identity, ECC_WorldStatic, MyColSphere);}

如果 isHit 为真,我们将通过遍历数组 OutHits ,并向每个成功 cast 的 actor 的根组件添加径向推力。

if (isHit){// loop through TArrayfor (auto& Hit : OutHits){UStaticMeshComponent* MeshComp = Cast<UStaticMeshComponent>((Hit.GetActor())->GetRootComponent());if (MeshComp){// alternivly you can use  ERadialImpulseFalloff::RIF_Linear for the impulse to get linearly weaker as it gets further from origin.// set the float radius to 500 and the float strength to 2000.MeshComp->AddRadialImpulse(GetActorLocation(), 500.f, 2000.f, ERadialImpulseFalloff::RIF_Constant, true);}}}

最终完整 cpp 代码如下

#include "AddRadialForce.h"
// add debug helpfers
#include "DrawDebugHelpers.h"// Sets default values
AAddRadialForce::AAddRadialForce()
{// Set this actor to call Tick() every frame.  You can turn this off to improve performance if you don't need it.PrimaryActorTick.bCanEverTick = true;}// Called when the game starts or when spawned
void AAddRadialForce::BeginPlay()
{Super::BeginPlay();// create tarray for hit resultsTArray<FHitResult> OutHits;// get actor locationsFVector MyLocation = GetActorLocation();// start and end locations. The sphere will create the radial sweep.FVector Start = MyLocation;FVector End = MyLocation;// create a collision sphereFCollisionShape MyColSphere = FCollisionShape::MakeSphere(500.0f);// draw collision sphereDrawDebugSphere(GetWorld(), GetActorLocation(), MyColSphere.GetSphereRadius(), 50, FColor::Cyan, true);// check if something got hit in the sweepbool isHit = GetWorld()->SweepMultiByChannel(OutHits, Start, End, FQuat::Identity, ECC_WorldStatic, MyColSphere);if (isHit){// loop through TArrayfor (auto& Hit : OutHits){UStaticMeshComponent* MeshComp = Cast<UStaticMeshComponent>((Hit.GetActor())->GetRootComponent());if (MeshComp){// alternivly you can use  ERadialImpulseFalloff::RIF_Linear for the impulse to get linearly weaker as it gets further from origin.// set the float radius to 500 and the float strength to 2000.MeshComp->AddRadialImpulse(GetActorLocation(), 500.f, 2000.f, ERadialImpulseFalloff::RIF_Constant, true);}}}}// Called every frame
void AAddRadialForce::Tick(float DeltaTime)
{Super::Tick(DeltaTime);}

实际运行起来的效果图如下

这篇关于最简单的 UE 4 C++ 教程 —— 给 Actor 添加径向推力【十八】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/220919

相关文章

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

使用Docker构建Python Flask程序的详细教程

《使用Docker构建PythonFlask程序的详细教程》在当今的软件开发领域,容器化技术正变得越来越流行,而Docker无疑是其中的佼佼者,本文我们就来聊聊如何使用Docker构建一个简单的Py... 目录引言一、准备工作二、创建 Flask 应用程序三、创建 dockerfile四、构建 Docker

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函