unity中绑定动画的行为系统

主要代码逻辑是创建一个action队列,当动画播放结束时就移除队头,执行后面的事件


public class Enemy : MonoBehaviour
{public event Action E_AnimatorFin;//当动画播放完毕时public Action DefaultAction;//默认事件public Dictionary<Action, string> EventAnimator= new();//Event对应的动画public List<Action> Eventlist=new();//要做的event列表private Animator ani;public virtual void Start(){ani = GetComponent<Animator>();E_AnimatorFin += OnAnimatorFinish;StartCoroutine(SpawningAnimator());IEnumerator SpawningAnimator(){yield return new WaitForSeconds(0.02f);PlayAnimator(ani, EventAnimator[ Eventlist[0]],() =>{Debug.Log("动画播放前执行代码"+ EventAnimator[ Eventlist[0]]);},() =>{E_AnimatorFin?.Invoke();Debug.Log("动画播放完执行代码");});}}public virtual void OnAnimatorFinish(){StartCoroutine(spawnanimator());IEnumerator spawnanimator(){yield return new WaitForSeconds(0.02f);if (Eventlist.Count == 0){Eventlist.Add(DefaultAction);Debug.LogWarning(name+$"并没有新的事件,目前在执行默认的事件");}try{PlayAnimator(ani, EventAnimator[ Eventlist[0]],() =>{Debug.LogError($"执行前{EventAnimator[ Eventlist[0]]}");Eventlist[0]();},() =>{Debug.LogError("执行完");E_AnimatorFin?.Invoke();}); }catch (KeyNotFoundException e){Debug.LogError(Eventlist[0].Method.Name+"没有匹配动画!");}Eventlist.RemoveAt(0);}}#region Animatorpublic void PlayAnimator(Animator animator, string clipName, Action startAct = null, Action endAct = null){StartCoroutine(PlayAnimationItor(animator, clipName, startAct, endAct));}/// <summary>/// Animation动画播放迭代器/// </summary>/// <param name="animation">Animation组件</param>/// <param name="clipName">clip片段名</param>/// <param name="startAct">委托函数</param>/// <param name="endAct">委托函数</param>/// <returns></returns>private IEnumerator PlayAnimationItor(Animator animator, string clipName, Action startAct=null, Action endAct=null){startAct?.Invoke();animator.Play(clipName);// 获取目标动画的名称string targetClipName =clipName;yield return StartCoroutine(WaitForEndOfAnimr(targetClipName));IEnumerator WaitForEndOfAnimr(string targetName){yield return new WaitForSeconds(0.1f);//为过渡动画预留时间AnimatorClipInfo[] clipInfo = animator.GetCurrentAnimatorClipInfo(0);Debug.Log("Targetname>>"+targetName);while (animator.GetCurrentAnimatorStateInfo(0).IsName(targetName)){// 获取当前动画片段信息clipInfo = animator.GetCurrentAnimatorClipInfo(0);if (clipInfo.Length > 0){// 获取当前播放的动画片段的名称string currentClipName = clipInfo[0].clip.name;Debug.Log("当前播放的动画片段名称: " + currentClipName);}yield return new WaitForSeconds(0.01f);}}
//                Debug.LogError($"{animator.GetCurrentAnimatorClipInfo(0)[0].clip.name}+{targetClipName}");endAct?.Invoke();}#endregion#region  AI逻辑/// <summary>/// 设置没有事件列表时自动执行的事件/// </summary>/// <param name="t"></param>public void SetDefaultEvent(Action t){DefaultAction = t;}/// <summary>/// 为怪物添加一个事件到列表/// </summary>/// <param name="action"></param>public void AddEvent(Action action){Eventlist.Add(action);}/// <summary>/// 重置怪物事件列表为一个新的列表/// </summary>/// <param name="actions"></param>public void SetEvent(Action[] actions){foreach (var VARIABLE in Eventlist){Eventlist.Remove(VARIABLE);}foreach (var VARIABLE in actions){Eventlist.Add(VARIABLE);}}/// <summary>/// 强制加入一个事件到下一个行动/// </summary>/// <param name="add">增加的事件</param>public void NextEvent(Action addAction){Debug.Log("length="+Eventlist.Count);if (Eventlist.Count>0){Eventlist.Insert(1,addAction);}else{Eventlist.Add(addAction);}}/// <summary>/// 强制指定下一个事件且立刻执行/// </summary>/// <param name="addAction"></param>public void DoNextEvent(Action addAction){StopAllCoroutines();NextEvent(addAction);if (Eventlist.Count>1){Eventlist.RemoveAt(0);}PlayAnimator(ani, EventAnimator[ Eventlist[0]],() =>{Eventlist[0]();Debug.Log("动画播放前执行代码"+ EventAnimator[ Eventlist[0]]);},() =>{E_AnimatorFin?.Invoke();Debug.Log("动画播放完执行代码");});}#endregion

使用示例

public class Cow : Enemy
{private GameObject normalATK;public  void Awake(){normalATK = transform.Find("NormalATK").gameObject;normalATK.SetActive(false);EventAnimator.Add(AI_DoRandomRelex,"Start");//为事件指定对应动画EventAnimator.Add(Animator_Idle,"idle");EventAnimator.Add(Animator_Rest,"rest");EventAnimator.Add(AI_FindPlayer,"walk");EventAnimator.Add(AI_Attack,"attack");SetDefaultEvent(AI_DoRandomRelex);//设置默认动画AddEvent(DefaultAction);//添加默认动画E_HPChanged += () => {Debug.LogError("收到伤害"); };E_HPChanged += ToHostile;//收到伤害时敌对}/// <summary>/// 收到伤害的时候调用,改为攻击模式/// </summary>void ToHostile(){DoNextEvent(AI_FindPlayer);SetDefaultEvent(AI_FindPlayer);FriendlyTag = FriendlyLevel.Hostile;}public void AI_DoRandomRelex()//中立状态时随机做待机动画{int rd = Random.Range(0, 100);if (rd<51){Debug.Log("next>>idle");NextEvent(Animator_Idle);}else{Debug.Log("next>>rest");NextEvent(Animator_Rest);}}private void Animator_Idle()//待机动画只有动画效果,没有要执行的内容{Debug.Log("IDLE");}private void Animator_Rest(){Debug.Log("REST");}private void Debug3(){}bool finder;//只有在事件进行的时候才追踪.当事件结束后退出追踪循环public override void AI_FindPlayer()//在敌对状态默认追踪玩家{finder = true;StartCoroutine(basefind());E_AnimatorFin += setfinder;IEnumerator basefind(){while (finder){base.AI_FindPlayer();yield return new WaitForSeconds(0.1f);}E_AnimatorFin -= setfinder;}void setfinder(){finder = false;}StartCoroutine(atrange());E_AnimatorFin += () => { StopCoroutine(atrange()); };IEnumerator atrange(){do{if (IsPlayerInAttackRange)//如果玩家在攻击范围内就进行攻击事件{DoNextEvent(AI_Attack);}yield return new WaitForSeconds(0.02f);} while (true);}}public void AI_Attack()//攻击事件{base.AI_FindPlayer();Debug.LogError("AIATK");StartCoroutine(WaitATK());E_AnimatorFin += () => {des();  };IEnumerator WaitATK(){yield return new WaitForSeconds(0.2f * EventSpeed);normalATK.SetActive(true);} void des(){normalATK.SetActive(false);E_AnimatorFin -= des;}}public override void Update(){base.Update();//如果玩家在索敌范围外就变回中立if (!IsPlayerInFindingRange&& (FriendlyTag == FriendlyLevel.Hostile)){FriendlyTag = FriendlyLevel.Neutral;SetDefaultEvent(AI_DoRandomRelex);}}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/152102.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java版工程行业管理系统源码-专业的工程管理软件-提供一站式服务

项目背景 一、随着公司的快速发展&#xff0c;企业人员和经营规模不断壮大。为了提高工程管理效率、减轻劳动强度、提高信息处理速度和准确性&#xff0c;公司对内部工程管理的提升提出了更高的要求。 二、企业通过数字化转型&#xff0c;不仅有利于优化业务流程、提升经营管理…

2019年[海淀区赛 第2题] 阶乘

题目描述 n的阶乘定义为n!n*(n -1)* (n - 2)* ...* 1。n的双阶乘定义为n!!n*(n -2)* (n -4)* ...* 2或n!!n(n - 2)*(n - 4)* ...* 1取决于n的奇偶性&#xff0c;但是阶乘的增长速度太快了&#xff0c;所以我们现在只想知道n!和n!!末尾的的个数 输入格式 一个正整数n &#xff…

AI驱动的3D模型无缝纹理生成

创建无缝纹理一直是一个需要艺术技巧的劳动密集型过程。 然而&#xff0c;随着稳定扩散模型&#xff08;Stable Diffusion&#xff09;的出现&#xff0c;情况发生了变化。 通过将文本转换为逼真、无边界的图像&#xff0c;稳定扩散彻底改变了纹理创建&#xff0c;使其变得易于…

Axios、SASS学习笔记

目录 前言 一、Axios基础认识 1、简介 2、相关文档 3、基本配置 4、基础快捷使用 二、Axios封装 1、公共配置文件 2、细化每个接口的配置 3、使用并发送请求 三、SASS 1、简介 2、相关文档 3、使用前奏 4、使用变量 5、嵌套规则 6、父级选择器标识 & 前言…

Leetcode hot 100之二叉树

目录 (反)序列化二叉树&#xff08;str<->tree&#xff09;&#xff1a;前序 前序遍历&#xff08;迭代&#xff09;/路径 stack.length 入栈&#xff1a;中右左 出栈&#xff1a;中左右 中序遍历&#xff08;迭代&#xff09; cur||stack.length 后序遍历&#x…

计算机视觉处理的开源框架

计算机视觉是一门涉及图像和视频分析的领域&#xff0c;有许多开源的框架和库可用于构建计算机视觉应用程序。以下是一些常见的计算机视觉开源框架及其特点&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合…

一盏茶的功夫帮你彻底搞懂JavaScript异步编程从回调地狱到async/await

&#x1f3ac; 江城开朗的豌豆&#xff1a;个人主页 &#x1f525; 个人专栏 :《 VUE 》 《 javaScript 》 ⛺️ 生活的理想&#xff0c;就是为了理想的生活 ! 目录 &#x1f4d8; 1. 引言 &#x1f4d8; 2. 使用方法 &#x1f4d8; 3. 实现原理 &#x1f4d8; 4. 写到最后…

Android 自定义PopupWindow,实现下拉框

1、效果图 2、前言 1、页面由 MagicIndicator ViewPager2 Fragment 实现&#xff1b; 2、下拉框是基于WindowManager实现&#xff1b; 3、我使用PopupWindow实现下拉框时&#xff0c;发现一个问题&#xff0c;PopupWindow 在窗口显示的情况下&#xff0c;无法直接从外部修…

(c语言进阶)指针的进阶

一.字符指针 1.一般应用 &#xff08;1&#xff09;%c的应用 &#xff08;2&#xff09;%s的应用 字符指针没有权限通过解引用去改变指针指向的值 2.笔试题 题目&#xff1a;判断输出结果 int main() { const char* p1 "abcdef"; const char* p2 "…

无法向会话状态服务器发出会话状态请求。请确保 ASP.NET State Service (ASP.NET 状态服务)已启动,并且客户端端口与服务器端口相同

“/”应用程序中的服务器错误。 无法向会话状态服务器发出会话状态请求。请确保 ASP.NET State Service (ASP.NET 状态服务)已启动&#xff0c;并且客户端端口与服务器端口相同。如果服务器位于远程计算机上&#xff0c;请检查 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Se…

【云备份】

文章目录 [toc] 1 :peach:云备份的认识:peach:1.1 :apple:功能了解:apple:1.2 :apple:实现目标:apple:1.3 :apple:服务端程序负责功能:apple:1.4 :apple:服务端功能模块划分:apple:1.5 :apple:客户端程序负责功能:apple:1.6 :apple:客户端功能模块划分:apple: 2 :peach:环境搭建…

[VC++]圆形进度条

[VC]圆形进度条 源码开发环境&#xff1a;VC6.0 WIN10 64位下编译通过利用绘制饼图的原理&#xff0c;来制作的圆形进度条&#xff0c;可以显示百分比。软件运行截图如下&#xff1a; 附件源码下载(点击下载&#xff09;

基于ensp的园区网络搭建综合实验

目录 &#x1f552; 1. 技术介绍&#x1f552; 2. 需求分析&#x1f558; 2.1 项目背景和需求&#x1f558; 2.2 项目需求分析 &#x1f552; 3. 网络结构设计&#x1f552; 4. 网络拓扑规划&#x1f552; 5. 网络设备基本配置&#x1f558; 5.1 规划VLAN&#x1f558; 5.2 MST…

【RK3588】YOLO V5在瑞芯微板子上部署问题记录汇总

YOLO V5训练模型部署到瑞芯微的板子上面&#xff0c;官方是有给出案例和转过详情的。并且也提供了Python版本的推理代码&#xff0c;以及C语言的代码。 但是&#xff0c;对于转换过程中的细节&#xff0c;哪些需要改&#xff1f;怎么改&#xff1f;如何改&#xff0c;和为什么…

Java中树形菜单的实现方式(超全详解!)

前言 这篇文中&#xff0c;我一共会用两种方式来实现目录树的数据结构&#xff0c;两种写法逻辑是一样的&#xff0c;只是一种适合新手理解&#xff0c;一种看着简单明了但是对于小白不是很好理解。在这里我会很详细的讲解每一步代码&#xff0c;主要是方便新人看懂&#xff0…

typescript: Builder Pattern

/*** file: CarBuilderts.ts* TypeScript 实体类 Model* Builder Pattern* 生成器是一种创建型设计模式&#xff0c; 使你能够分步骤创建复杂对象。* https://stackoverflow.com/questions/12827266/get-and-set-in-typescript* https://github.com/Microsoft/TypeScript/wiki/…

4.Docker 搭建 redis6

1.下载redis docker pull redis:6.2.62.创建需要挂载的宿主机文件夹 mkdir -p /data/redis/conf mkdir -p /data/redis/data3.配置redis 切换到/data/redis/conf文件夹下&#xff0c;创建redis.conf,复制redis.conf配置文件内容到redis.conf文件中&#xff0c;然后按下键盘 …

黑豹程序员-架构师学习路线图-百科:AJAX

文章目录 1、什么是AJAX2、发展历史3、工作原理4、一句话概括 1、什么是AJAX Ajax即Asynchronous&#xff08;呃森可乐思&#xff09; Javascript And XML&#xff08;异步JavaScript和XML&#xff09; 在 2005年被Jesse James Garrett&#xff08;杰西詹姆斯加勒特&#xff09…

GD32F103x 定时器

1. 定时器的基本介绍 STM32的定时器主要分为三种&#xff1a;高级定时器、通用定时器、基本定时器。 即&#xff1a;高级定时器具有捕获/比较通道和互补输出&#xff0c;死区时间&#xff0c;通用定时器只有捕获/比较通道&#xff0c;基本定时器没有以上两者。 1. 基本定时…

网络安全:六种常见的网络攻击手段

1、什么是VPN服务&#xff1f; 虚拟专用网络&#xff08;或VPN&#xff09;是您的设备与另一台计算机之间通过互联网的安全连接。VPN服务可用于在离开办公室时安全地访问工作计算机系统。但它们也常用于规避政府审查制度&#xff0c;或者在电影流媒体网站上阻止位置封锁&#…