71. UE5 RPG 实现敌人召唤技能

2024-06-20 00:20

本文主要是介绍71. UE5 RPG 实现敌人召唤技能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这一篇文章中,我们要实现敌人的召唤师能够召唤自己的仆从进行作战。
要实现这个技能,我们首先创建新的敌人蓝图,用于召唤。接着,我们将实现一个召唤技能基类,在基类中实现在召唤师的周围获取到可以生成的位置点,然后在技能蓝图中在对应的位置生成敌人的Actor,并在后续对其进行优化。接下来,我们将一步步实现此功能。

创建新的敌人

我们首先创建两个新的敌人,这两个恶魔类型的敌人,用于召唤使用。
这里也不仔细讲解如何创建,我们在64. UE5 RPG 创建新的双手攻击怪物中也专门讲解了如何创建一个新的敌人,这里只说一下项目源码的改动。
在这里插入图片描述
由于黑色的恶魔是远程释放火球的,但是它没有武器,我们之前释放火球的都是带有武器的角色(萨满,玩家英雄)为了兼容没有武器的角色,我们又增加了一个可以通过骨骼标签和骨骼名称去获取位置的。
在这里插入图片描述

在角色基类里面实现它

FVector ARPGCharacter::GetCombatSocketLocationByTag_Implementation(const FGameplayTag SocketTag, const FName SocketName) const
{if(SocketTag.MatchesTagExact(FRPGGameplayTags::Get().CombatSocket_Weapon)){return Weapon->GetSocketLocation(SocketName);}return GetMesh()->GetSocketLocation(SocketName);
}

然后将生成投掷物的技能接口修改掉
在这里插入图片描述
在敌人的技能里,蓝图修改
在这里插入图片描述
在玩家英雄技能里,修改蓝图
在这里插入图片描述

生成位置点

我们要实现召唤技能,首先创建一个新的召唤技能基类,由于它不是一个攻击性的技能,也不会造成伤害,所以,我们基于技能基类创建即可
在这里插入图片描述
创建一个名为RPGSummonAbility的召唤技能
在这里插入图片描述
在.h文件中,我们需要添加一个获取随机位置的函数,并添加一些可以自定义的配置项

UCLASS()
class AURA_API URPGSummonAbility : public URPGGameplayAbility
{GENERATED_BODY()public:UFUNCTION(BlueprintCallable)TArray<FVector> GetSpawnLocations();UPROPERTY(EditDefaultsOnly, Category="Summoning")int32 NumMinions = 5; // 召唤的数量UPROPERTY(EditDefaultsOnly, Category="Summoning")TArray<TSubclassOf<APawn>> MinionClasses; //召唤生成的角色类UPROPERTY(EditDefaultsOnly, Category="Summoning")float MinSpawnDistance = 50.f; //召唤物距离召唤师最近的距离UPROPERTY(EditDefaultsOnly, Category="Summoning")float MaxSpawnDistance = 250.f; //召唤物距离召唤师最远的距离UPROPERTY(EditDefaultsOnly, Category="Summoning")float SpawnSpread = 90.f; //召唤物在召唤师前面的角度范围
};

在cpp中,我们实现此函数,我们想让召唤物生成在召唤师的前面角度范围内,需要获取到召唤师的位置和朝向。然后,将角度范围的值进行分段

	const FVector Forward = GetAvatarActorFromActorInfo()->GetActorForwardVector(); //获取召唤师的朝向const FVector Location = GetAvatarActorFromActorInfo()->GetActorLocation(); //获取召唤师的位置const float DeltaSpread = SpawnSpread / NumMinions; //将召唤的角度范围进行分段,在每段里面生成一个召唤物

然后,我们再获取到角色的最左和最右的生成角度,通过debug调试一下,查看获取到的角度是否正确

	const FVector LeftOfSpread = Forward.RotateAngleAxis(-SpawnSpread / 2.f, FVector::UpVector); //获取到最左侧的角度UKismetSystemLibrary::DrawDebugArrow(GetAvatarActorFromActorInfo(), Location, Location + LeftOfSpread * MaxSpawnDistance, 4.f, FLinearColor::Red, 3.f);const FVector RightOfSpread = Forward.RotateAngleAxis(SpawnSpread / 2.f, FVector::UpVector); //获取到最右侧的角度UKismetSystemLibrary::DrawDebugArrow(GetAvatarActorFromActorInfo(), Location, Location + RightOfSpread * MaxSpawnDistance, 4.f, FLinearColor::Red, 3.f);

我们创建了一个技能类,写了一些代码,接下来,去UE里面测试一下,功能是否有效。
打开UE创建一个蓝图,基于我们创建的召唤技能类
在这里插入图片描述
右侧设置一下技能标签,用于技能激活
在这里插入图片描述
蓝图这里直接连一下,调用函数
在这里插入图片描述
在数据配置里面将技能给对应的角色设置
在这里插入图片描述
将敌人放到场景测试,查看是否能够成功绘制出debug线
在这里插入图片描述
接下来,查看我们是否能够获取到准确的位置,我们绘制调试球体

	DrawDebugSphere(GetWorld(), Location + LeftOfSpread * MinSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);DrawDebugSphere(GetWorld(), Location + LeftOfSpread * MaxSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);DrawDebugSphere(GetWorld(), Location + RightOfSpread * MinSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);DrawDebugSphere(GetWorld(), Location + RightOfSpread * MaxSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);

然后打印查看
在这里插入图片描述
接下来,我们想根据生成的个数,平均角度,然后在每个角度上面,从最远和最近的距离随机一个位置来生成角色。这里如果需要5个位置,我们其实分为四段就可以,大家可以画图测试一下。

	const FVector Forward = GetAvatarActorFromActorInfo()->GetActorForwardVector(); //获取召唤师的朝向const FVector Location = GetAvatarActorFromActorInfo()->GetActorLocation(); //获取召唤师的位置const float DeltaSpread = SpawnSpread / (NumMinions - 1.f); //将召唤的角度范围进行分段,在每段里面生成一个召唤物const FVector LeftOfSpread = Forward.RotateAngleAxis(-SpawnSpread / 2.f, FVector::UpVector); //获取到最左侧的角度TArray<FVector> SpawnLocations;for(int32 i = 0; i < NumMinions; i++) //遍历,在每个分段上面获取位置{const FVector Direction = LeftOfSpread.RotateAngleAxis(DeltaSpread * i, FVector::UpVector); //获取当前分段的角度FVector ChosenSpawnLocation = Location + Direction * FMath::FRandRange(MinSpawnDistance, MaxSpawnDistance); //随机位置,加上原始位置就是偏移的位置DrawDebugSphere(GetWorld(), ChosenSpawnLocation, 15.f,12,FColor::Green,false, 3.f);SpawnLocations.Add(ChosenSpawnLocation);}

接着编译去测试代码,查看效果
在这里插入图片描述
接下来,我们添加上射线拾取,防止生成的角色悬空出现。

TArray<FVector> URPGSummonAbility::GetSpawnLocations()
{const FVector Forward = GetAvatarActorFromActorInfo()->GetActorForwardVector(); //获取召唤师的朝向const FVector Location = GetAvatarActorFromActorInfo()->GetActorLocation(); //获取召唤师的位置const float DeltaSpread = SpawnSpread / (NumMinions - 1.f); //将召唤的角度范围进行分段,在每段里面生成一个召唤物const FVector LeftOfSpread = Forward.RotateAngleAxis(-SpawnSpread / 2.f, FVector::UpVector); //获取到最左侧的角度TArray<FVector> SpawnLocations;for(int32 i = 0; i < NumMinions; i++) //遍历,在每个分段上面获取位置{const FVector Direction = LeftOfSpread.RotateAngleAxis(DeltaSpread * i, FVector::UpVector); //获取当前分段的角度FVector ChosenSpawnLocation = Location + Direction * FMath::FRandRange(MinSpawnDistance, MaxSpawnDistance); //随机位置,加上原始位置就是偏移的位置FHitResult Hit;GetWorld()->LineTraceSingleByChannel(Hit, ChosenSpawnLocation + FVector(0.f, 0.f, 400.f), ChosenSpawnLocation - FVector(0.f, 0.f, 400.f), ECC_Visibility);if(Hit.bBlockingHit){ChosenSpawnLocation = Hit.ImpactPoint;}DrawDebugSphere(GetWorld(), ChosenSpawnLocation, 15.f,12,FColor::Green,false, 3.f);SpawnLocations.Add(ChosenSpawnLocation);}const FVector RightOfSpread = Forward.RotateAngleAxis(SpawnSpread / 2.f, FVector::UpVector); //获取到最右侧的角度UKismetSystemLibrary::DrawDebugArrow(GetAvatarActorFromActorInfo(), Location, Location + LeftOfSpread * MaxSpawnDistance, 4.f, FLinearColor::Red, 3.f);UKismetSystemLibrary::DrawDebugArrow(GetAvatarActorFromActorInfo(), Location, Location + RightOfSpread * MaxSpawnDistance, 4.f, FLinearColor::Black, 3.f);DrawDebugSphere(GetWorld(), Location + LeftOfSpread * MinSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);DrawDebugSphere(GetWorld(), Location + LeftOfSpread * MaxSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);DrawDebugSphere(GetWorld(), Location + RightOfSpread * MinSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);DrawDebugSphere(GetWorld(), Location + RightOfSpread * MaxSpawnDistance, 15.f,12,FColor::Blue,false, 3.f);return SpawnLocations;
}

在这里插入图片描述

实现技能蓝图

在技能蓝图里面,我们获取到了召唤生成的位置,后续我们要根据位置生成Actor,所以,我们将生成的位置保存下来,后续使用,并且不想一次性全部生成,一次性生成最大的问题是在一帧内创建多个资源会造成卡顿。
我们将生成的点保存为变量,并生成一个下标索引
在这里插入图片描述
接下来就是实现在位置生成测试盒子,查看在蓝图实现生成,这里在每个生成以后,延迟一秒继续生成。
在这里插入图片描述
查看效果
在这里插入图片描述
当然,我们还可以将数组内的位置点排序打乱,这样,生成时,不会只按一个方向生成
在这里插入图片描述
接下来,我们继续优化,在生成Actor之前,先在生成位置创建一个粒子特效,表现出后续会出现生成的物体。我们这里使用一个序列,遍历位置数组,在地面生成粒子特效
在这里插入图片描述
效果如下
在这里插入图片描述

生成召唤物

接下来,我们修改debug内容,将生成的调试盒子修改成真正需要生成的内容。
首先,将需要生成的内容,添加到配置项中
在这里插入图片描述
由于可以添加多个类,我们需要从其中随机一个类使用,在c++中添加一个新的函数,这个可以不需要引脚

	UFUNCTION(BlueprintPure, Category="Summoning")TSubclassOf<APawn> GetRandomMinionClass(); //获取随机的召唤物类

在实现这里,随机一个下标即可

TSubclassOf<APawn> URPGSummonAbility::GetRandomMinionClass()
{const int32 Selection = FMath::RandRange(0, MinionClasses.Num() - 1);return MinionClasses[Selection];
}

编译,我们打开去修改蓝图,通过节点spawn Actor from Class来生成实例,由于位置处于地面,我们生成的角色都会半身陷入到地面里,这里对z轴进行一个偏移,
在这里插入图片描述
发现生成是成功了,但是都不会动
在这里插入图片描述
我们再给它增加一个节点,生成一个默认的控制器,并控制目标
在这里插入图片描述
这样,我们就实现了召唤物的生成。

添加蒙太奇动画

实现召唤技能最后一步工作还需要制作,就是召唤师还没有设置动画。
创建一个蒙太奇
在这里插入图片描述
我们在蓝图中,设置了技能的相关设置
在这里插入图片描述
但是,召唤技能不需要设置这些内容,这里设置的是角色攻击的,我们将召唤的写死在技能里,在蒙太奇里面添加一个动画通知
在这里插入图片描述
在执行召唤之前,添加蒙太奇播放
在这里插入图片描述
为了防止出现滑步的现象,我们需要对召唤动画开启根运动,这样,蒙太奇播放时,会根据动画内的位移修改,召唤的动画没有位移,所以会固定在当前位置
在这里插入图片描述
接着我们在播放蒙太奇后,监听游戏事件,在监听到游戏事件后,会触发后续生成事件
在这里插入图片描述
这里需要注意的是,如果游戏结束,但是后面生成没有完成,也会被终止掉,所以,我们触发事件到蒙太奇动画播放完成,需要留有足够生成召唤物的时间。
接下来,就是查看是否能够成功播放蒙太奇和生成召唤物
在这里插入图片描述
没有问题本章内容完成,后面一章,我们将实现如何限制召唤物的召唤数量,并优化一些内容。
最后附上技能的蓝图节点全身照
在这里插入图片描述

这篇关于71. UE5 RPG 实现敌人召唤技能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

基于UE5和ROS2的激光雷达+深度RGBD相机小车的仿真指南(五):Blender锥桶建模

前言 本系列教程旨在使用UE5配置一个具备激光雷达+深度摄像机的仿真小车,并使用通过跨平台的方式进行ROS2和UE5仿真的通讯,达到小车自主导航的目的。本教程默认有ROS2导航及其gazebo仿真相关方面基础,Nav2相关的学习教程可以参考本人的其他博客Nav2代价地图实现和原理–Nav2源码解读之CostMap2D(上)-CSDN博客往期教程: 第一期:基于UE5和ROS2的激光雷达+深度RG

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现