Unity动画系统学习笔记(二)根运动、动画事件与状态机行为

news2024/9/30 15:23:10

一、根运动

在学习根运动前需要了解两个名词:

  • 身体变换:身体变换是角色的质心。它用于 Mecanim 的重定向引擎,并提供最稳定的移位模型。身体方向是相对于 Avatar T 形姿势的下身和上身方向的平均值。身体变换和方向存储在动画剪辑中(使用 Avatar 中设置的肌肉定义)。它们是动画剪辑中存储的唯一世界空间曲线。所有其他:肌肉曲线和 IK(反向动力学)目标(手和脚)都是相对于身体变换进行存储的。
  • 根变换:根变换是身体变换在 Y 平面上的投影,并在运行时计算。在每一帧都会计算根变换的变化。变换的此变化随后应用于游戏对象以使其移动。

以上出自Unity官方文档,看完还是一脸懵逼。。。举个简单的例子来说:假如现在角色有向前走动的动画,如果是身体变换,就是角色的模型在走动,但角色在世界中的位置并没有变化;如果是根变换,那么角色在模型上的移动就会反映到根节点上,也就是说角色不仅模型在走动,在世界上的位置也在移动。

可以看到,上方的角色因为没有采用根变换,其位置始终没有改变,只是在重复播放模型的动画。

1.1 开启或关闭根运动

那么如何开启或关闭根运动呢?这里涉及到两个选项。
首先在动画剪辑面板上,如果动画能够影响角色的位置或旋转,一般会有如下选项

这里面有个属性叫做「Bake Into Pose」,意思是将方向保持在身体变换上。也就是说,如果勾选这个属性,就是将根变换存放在动画中,即使用身体变换。

除此之外,还有在角色身上挂载的「Animator」组件中,有个「Apply Root Motion」选项。只有在开启这个选项时,根变换才会应用到模型身上。

这两个选项不同的排列组合也会对动画产生不同的影响。

  • 「Bake Into Pose」开启,「Apply Root Motion」关闭或开启
    只要「Bake Into Pose」选项开启,动画就使用了身体变换。就会出现如下效果

  • 「Bake Into Pose」关闭,「Apply Root Motion」开启
    此时角色会正常移动

  • 「Bake Into Pose」关闭,「Apply Root Motion」关闭
    此时因为使用了根变换,但不允许应用,所以角色会原地踏步

1.2 通过脚本控制根运动的发生

在某些情况下,我们希望一部分状态启用根运动,另一部分关闭根运动。此时就可以使用脚本控制根运动的发生。

在脚本中添加OnAnimatorMove()生命周期函数,就会发现「Animator」组件的「Apply Root Motion」变为了「Handled By Script」。意味着根运动已被脚本接管

接下来我们只需要在OnAnimatorMove()中实现我们的控制逻辑即可。

比如我们希望由根运动控制角色的位置。但在跳跃时,由于角色的根节点位置只存在Y轴方向的变换,就会造成只能原地起跳的问题。这种情况下就可以在这个函数中对当前状态进行判断。如果当前是跳跃状态,就直接通过代码控制角色的位置。

private void OnAnimatorMove()
{
	// 如果当前状态的标签不是"NoRootMotion",则由根运动控制角色位置
	if (!_animator.GetCurrentAnimatorStateInfo(0).IsTag("NoRootMotion"))
	{
		_noRootMotion = false;
		_parent.position += _animator.deltaPosition;
		_parent.rotation *= _animator.deltaRotation;
	}
	// 否则由其他代码控制
	else
	{
		_noRootMotion = true;
	}
}

看下效果

1.3 目标匹配

当我们的角色需要与其他角色或物体互动时,由于位置的原因,可能会出现严重的穿模现象。比如下方的踢腿动作

我们更希望在踢腿时,能够恰好踢中对方的某个位置,且不应该直接穿过对方的身体。这时就可以用到目标匹配。

简单来说,目标匹配实际上就是Animator类中的MatchTarget()方法。它需要传入如下几个参数:

  • Vector3 matchPosition:目标位置
  • Quaternion matchRotation:目标旋转
  • AvatarTarget targetBodyPart:自身需要匹配的部位
  • MatchTargetWeightMask weightMask:位置和旋转的权重
  • float startNormalizedTime:动画开始百分比(0~1)
  • float targetNormalizedTime:动画结束百分比(0~1)
  • bool completeMatch:函数中断时是否强制移动到匹配位置

我们可以将目标匹配的代码放在Update()中,当动画状态机进入到踢腿的状态时执行

if (_animator.GetCurrentAnimatorStateInfo(0).IsName("Kick"))
{
	_animator.MatchTarget(machTarget.position,transform.rotation, 
		AvatarTarget.LeftFoot,new MatchTargetWeightMask(Vector3.one,1 ),
		0f,0.64f);
}

效果如下。可以看到这个方法会强制将角色的左脚匹配到目标点上,即便两者距离很远,也会直接位移到目标点前。

二、动画事件

动画事件可以让我们在动画执行的过程中触发指定的脚本方法。可以在制作技能等场景时派上用场。
它使用起来也非常简单,打开角色的「Animation」面板。选择要添加事件的动画剪辑,然后点击右侧的「Add Event」按钮,就可以在时间轴上添加一个事件

点击时间轴上添加的事件,可以在面板中指定要触发的方法。

我们让它在触发时在控制台输出“Shoot”,看下效果

如果直接选中动画剪辑文件,再打开「Animation」面板,选中之前添加的动画事件。就会发现检视面板多出来几个属性

也就是说我们可以为触发的方法添加参数,并在这里指定。

private void Shoot(int param)
{
	Debug.Log("Shoot:"+param);
}

我们把int类型的参数指定为10,看下效果

不过这种方式只能传递一个参数,我们并不能添加多个参数来接收面板上所有指定好的参数。不过Unity为我们提供了AnimationEvent类来封装这些传入的参数。通过它就可以接收到所有传入的参数

private void Shoot(AnimationEvent param)
{
	Debug.Log("Shoot:"+param.intParameter);
	Debug.Log("Shoot:"+param.floatParameter);
	Debug.Log("Shoot:"+param.stringParameter);
	Debug.Log("Shoot:"+param.objectReferenceParameter);
	Debug.Log("Shoot:"+param.functionName);
}

效果如下

有了动画事件,我们就可以在动画播放过程中的适当的时机,触发一些指定的效果,比如在拉完弓箭时射出一枚箭矢、抬手后释放技能等。

三、状态机行为

Unity允许我们给动画状态机中的单个状态挂载独立的脚本,以在动画播放时处理额外的逻辑。具体的方法是:

首先选中动画状态机中的状态或子状态机,然后在检视面板中会出现「Add Behaviour」按钮。然后就可以手动创建脚本并进行挂载。

打开脚本可以发现,该类自动继承了StateMachineBehaviour类,并用注释的形式给出了一系列生命周期函数。

接下来我们通过这种方式,实现「在角色攻击时不允许移动」的效果
首先在角色控制器中添加是否允许移动的判断条件,并在移动方法中进行判断

public bool CanMove = true;

public void Move()
{
	if(!CanMove)  
	    return;
	// ...
}

然后在状态机行为脚本中,对条件进行控制

public class CharacterBehaviourController : StateMachineBehaviour
{
	private SaCharacterController _controller;

	public override void OnStateEnter(Animator animator, AnimatorStateInfo stateInfo, int layerIndex)
	{
		_controller = animator.gameObject.GetComponent<SaCharacterController>();
		_controller.CanMove = false;
	}
	public override void OnStateMachineExit(Animator animator, int stateMachinePathHash)
	{
		_controller.CanMove = true;
	}
}

看下使用状态机行为前的效果

再看下使用之后的效果

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用VisualStutio2022开发第一个C++程序

使用VisualStudio2022创建C项目 第一步&#xff1a;新建C的控制台应用 第二步&#xff1a;填写项目名称和代码存放位置&#xff0c;代码的存放目录不要有中文名 第三步:点击创建&#xff0c;VisualStudio会自动开始帮我们创建项目 第四步&#xff1a;项目创建好以后&…

【PostGIS】PostgreSQL15+对应PostGIS安装教程及空间数据可视化

一、PostgreSQL15与对应PostGIS安装 PostgreSQL15安装&#xff1a;下载地址PostGIS安装&#xff1a;下载地址&#xff08;选择倒数第二个&#xff09; 1、PostgreSQL安装 下载安装包&#xff1b;开始安装&#xff0c;这里使用默认安装&#xff0c;一直next直到安装完成&…

接口测试的持续集成的工具(git代码管理工具,jenkins持续集成)

持续集成的概念&#xff1a;大白话就是持续的做一件事情&#xff0c;使其使用起来更加流畅&#xff1b;结合测试来讲就是说用工具管理好代码的同时&#xff0c;使代码运行的更加自动以及智能&#xff1b;提升测试效率。 ⽹址&#xff1a;https://git-scm.com/downloads 长这个…

SpringSecurity6 | 失败后的跳转

✅作者简介:大家好,我是Leo,热爱Java后端开发者,一个想要与大家共同进步的男人😉😉 🍎个人主页:Leo的博客 💞当前专栏: Java从入门到精通 ✨特色专栏: MySQL学习 🥭本文内容: SpringSecurity6 | 失败后的跳转 📚个人知识库: Leo知识库,欢迎大家访问 学习…

【网络安全/CTF】unseping 江苏工匠杯

该题考察序列化反序列化及Linux命令执行相关知识。 题目 <?php highlight_file(__FILE__);class ease{private $method;private $args;function __construct($method, $args) {$this->method $method;$this->args $args;}function __destruct(){if (in_array($thi…

TypeScript学习笔记归纳(持续更新ing)

文章目录 前言 二、TypeScript的优势体现在哪里&#xff1f; 1、执行时间上的区别 2、基础数据类型区别 3、TS优势 三、TypeScript的关键特性 四、TypeScript的类型系统 1、什么是类型注释&#xff1f; 2、类型系统核心 - 常用类型 1&#xff09; 基本类型&#xff0…

vue 自定义的通用的表格组件(使用div)

vue 自定义的通用的表格组件(使用div) 做项目的时候由于传统的table及elementUI的el-table的tr和td没办法设置间距,满足不了UI提供的设计稿,为了还原,自己封装了该组件 该表格组件的特点 表头排序功能:支持点击表头进行升序、降序排序,并显示相应的排序图标。复选框功能…

【unity】关于unity3D摄像机视角移动的几种方式详解

目录 一、前言 二、Transform基础 1、几种坐标系 2、position和localPosition属性 3、rotation属性 三、摄像机的平移 1、键盘控制平移 2、鼠标控制平移 3、整合 四、摄像机的旋转 1、绕自身旋转 2、绕目标物体旋转 3、整合 五、优化功能 1、调整速率 2、切换…

【数据结构】布隆过滤器原理详解及其代码实现

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能AI、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推荐--…

差生文具多之(二): perf

栈回溯和符号解析是使用 perf 的两大阻力&#xff0c;本文以应用程序 fio 的观测为例子&#xff0c;提供一些处理它们的经验法则&#xff0c;希望帮助大家无痛使用 perf。 前言 系统级性能优化通常包括两个阶段&#xff1a;性能剖析和代码优化&#xff1a; 性能剖析的目标是寻…

【期末考试】计算机网络、网络及其计算 考试重点

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 计算机网络及其计算 期末考点 &#x1f680;数…

Netty Review - 深入理解Netty: ChannelHandler的生命周期与事件处理机制

文章目录 概述CodeLifeCycleInBoundHandler 验证 概述 Netty的ChannelHandler是处理网络事件&#xff08;如数据读取、数据写入、连接建立、连接关闭等&#xff09;的核心组件。 在Netty中&#xff0c;ChannelHandler的生命周期与Channel的状态紧密相关&#xff0c;主要涉及到…

从mice到missForest:常用数据插值方法优缺点

一、引言 数据插值方法在数据处理和分析中扮演着至关重要的角色。它们可以帮助我们处理缺失数据&#xff0c;使得数据分析更加准确和可靠。数据插值方法被广泛应用于金融、医疗、社会科学等领域&#xff0c;以及工程和环境监测等实际应用中。 在本文中&#xff0c;我们将探讨三…

制作小米导航实验

一、实验题目 制作一个小米"手机"的标签&#xff0c;当鼠标移动到"手机"上后&#xff0c;显示右边的白色菜单&#xff0c;当鼠标离开"手机"时&#xff0c;菜单消失。 二、实验代码 <!DOCTYPE html> <html lang"en"> <…

高级人工智能之群体智能:蚁群算法

群体智能 鸟群&#xff1a; 鱼群&#xff1a; 1.基本介绍 蚁群算法&#xff08;Ant Colony Optimization, ACO&#xff09;是一种模拟自然界蚂蚁觅食行为的优化算法。它通常用于解决路径优化问题&#xff0c;如旅行商问题&#xff08;TSP&#xff09;。 蚁群算法的基本步骤…

python分割字符串 split函数

split函数用于字符串的分割&#xff0c;可以完成基于特定字符将字符串分割成若干部分并存于列表中 url "http://localhost:5000/static/images/DICOMDIR 5.26-6.1/10283674_GOUT_5_0_2.png" # 获取最后一个_的前一个数字 parts url.split(/) print(parts)在这里讲…

Kali Linux—借助 SET+MSF 进行网络钓鱼、生成木马、获主机shell、权限提升、远程监控、钓鱼邮件等完整渗透测试(二)

远控木马 SET 同时集成了木马生成工具&#xff0c;可以生成木马并调用MSF框架对远程主机进行控制。直接使用MSF生成木马并控制主机的可参考之前另一篇博文&#xff1a;渗透测试-Kali入侵Win7主机。 控制主机 1、运行 SET&#xff0c;选择创建攻击载荷和监听器&#xff1a; 2…

Tomcat日志乱码了怎么处理?

【前言】 tomacat日志有三个地方&#xff0c;分别是Output(控制台)、Tomcat Localhost Log(tomcat本地日志)、Tomcat Catalina Log。 启动日志和大部分报错日志、普通日志都在output打印;有些错误日志&#xff0c;在Tomcat Localhost Log。 三个日志显示区&#xff0c;都可能…

ETN21与CJ2M-CPU33通讯

实验设备:CJ2M-CPU33,CJ1W-EIP21,交叉网线 实验目的:手动建立数据链接表,建立TAG通讯。 实验步骤: IP地址设置:①usb线连上电脑,打开I/O表,将ETN21模块的ip地址与CJ2M-CPU33设置为同一个网段不同节点,节点号跟硬件上的node number一样,下载重启模块,如下: 配置n…

百分百能遇到的接口自动化测试面试题,看完的现在已经在办理入职了...

1. 什么是接口自动化测试&#xff1f; 答&#xff1a;接口自动化测试是指使用自动化工具对接口进行测试&#xff0c;验证接口的正确性、稳定性和性能等方面的指标。 2. 为什么要进行接口自动化测试&#xff1f; 答&#xff1a;接口自动化测试可以提高测试效率&#xff0c;减…