【Unity/Animator动画系统】多层动画状态机实现角色的基本移动
文章目录
- 前言
- 实现
- 顶层
- 地面状态
- 四方向混合树
- 计算动画所需参数
- 空中状态
- 分层动画
前言
最近打算做个Rougelike + RPG + 塔科夫 混搭风格的冒险游戏。暂且就当是一个有随机元素,有基地,死亡会掉落物品的近战塔科夫罢。
花了三天时间,整合了Mixamo的动画和模型资源,通过改动一点Unity自带的第三人称模板的控制器实现移动和动画控制,然后自己做了个动画状态机。写这篇主要是记录制作动画状态机时出现的问题状况和解决方案。
当下动画状态机实现了以下特性
- 基于人物速度的站立到奔跑的混合动画
- 基于人物朝向和速度的四方向混合动画(行走和蹲伏)
- 基于人物速度的跳跃和坠落的混合动画(站立跳跃和奔跑跳跃)
- 通过分层动画实现不同的武器装备和攻击动画
我自我认为这个动画状态机写的还算是比较简洁的,一开始接触Unity的动画系统,第一感觉就是这也不好复制,那也不好复制,一直在重复赋值,缺乏批量操作。现在呢,现在也这么觉得。
Unity状态机如果不维护好切换关系和划分层级,很快就会连的跟个蜘蛛网一样。我基于之前写过的角色状态机的经验,决定将动画状态机和角色状态机分一样的层,维护基本一致的转换关系,并且条件尽量只和角色自身属性有关,尽可能少的通过操作来判断转换条件。
在具体说明这个动画状态机前,先看一下Unity官方给的这个ThirdPersonController,这东西把所有东西都塞进了一个类中,东西全是挺全,就是写和扩展肯定是地狱级别的,我在添加控制的Animator的参数时,需要修改多处代码,十分甚至九分的麻烦,现在只是原型设计,后续肯定会把它重构成状态机结构的。
实现
顶层
先看一下状态机的拓扑图

顶层拓扑图维护了最高层的关系,即玩家从地面到空中的转换,引起这种变化的原因只有两个
- 玩家输入Jump操作
- 玩家悬空
而从空中状态返回到地面状态则只有一种原因,即玩家落地
地面状态
接下来再看看子状态机GroundState:

这一层主要维护所有的地面操作,包括正常的四方向混合和蹲下的四方向混合,二者之间通过玩家是否按下下蹲操作进行转换。
因为外层的状态机不会在条件满足时自动终止内部的操作,因此仍需要重复对二者进行判断是否离开地面或者玩家是否跳跃的条件进行判断,进而确定何时离开状态机。
四方向混合树
Idle/Walk/Run看名字就知道是个混合树。其内部长这样:

先通过方向混合得到四方向动画,再通过速度应用动画。这里只做了向前的奔跑操作,是我个人需求。在我的操作方式中,玩家默认是会自动朝向移动方向的(也是官方示例的默认功能),因此不需要其他的奔跑方向。但是我额外添加了一个瞄准操作,该操作会使玩家始终面向摄像机方向,此时玩家向前方以外的方向移动时,速度会被限制,无法奔跑,因此只需要其他方向的行走动画,也不需要其他方向的奔跑动画。
当然,四方向的移动效果并不好,如果有条件肯定是八个方向会更好,因为只有四方向对于斜向移动会导致插值出滑步。
下蹲和这个大同小异,由于下蹲时不能奔跑,速度恒定,因此也无需混合速度这一维数据。
这里再说一下如何获得图上的DirectionX和DirectionY参数。
计算动画所需参数
玩家速度和朝向一致时得到向量{0, 1},相反得到{0, -1},垂直且速度在朝向的右边得到{1, 0},左边则是{-1, 0}。这个是最终得到的输入要求,通过这个要求我们就可以得到这样的函数:
private Vector2 GetXYDirectionRelativeControllerRotation()
{// 获取玩家控制器的前向方向(面向方向)Vector2 facingDirection = new Vector2(_controller.transform.forward.x, _controller.transform.forward.z);// 获取玩家的速度方向,并且进行归一化Vector2 velocityDirection = new Vector2(_controller.velocity.normalized.x, _controller.velocity.normalized.z);// 如果速度为零,返回 0 或者其他默认值,表示没有移动if (velocityDirection.sqrMagnitude == 0f){return Vector2.zero; // 玩家没有移动时,角度为 0(可以自定义)}// 计算面向方向和速度方向之间的夹角float angle = Vector2.SignedAngle(facingDirection, velocityDirection);Quaternion quaternion = Quaternion.Euler(0, 0, angle);return (quaternion * Vector2.up).normalized;
}
计算两个向量的夹角,再按照两个夹角的角度去旋转一个(0, 1)向量。十分简单(也是踩了不少的坑)
这里是有问题的,通过这个获取的向量变化非常快,对于玩家角色而言就是很容易出现角色动画的跳变,因此需要在修改参数时额外进行一次插值,如下
Vector2 direction = GetXYDirectionRelativeControllerRotation();
Vector2 curDirection = new Vector2(_animator.GetFloat(_animIDDirectionX), _animator.GetFloat(_animIDDirectionY));
Vector2 smoothDirection = Vector2.Lerp(curDirection, direction, 10f * Time.deltaTime);
_animator.SetFloat(_animIDDirectionX, smoothDirection.x);
_animator.SetFloat(_animIDDirectionY, smoothDirection.y);
这样就可以获得比较好的方向混合效果了。
空中状态
下面是空中状态的拓扑图

空中拓扑实际上和官方的跳跃流程是差不多的,只是有两个比较明显的区别
- 区别一:使用Switch判断是玩家跳跃输入还是玩家悬空导致的进入该状态机
- 区别二:除Switch是空节点外,其余三个都是基于速度的混合树,用于混合从静止跳跃到奔跑跳跃的动画。
这个就比较常规了,但是因为涉及到不同动画片段的混合,尽量不要使用固定持续时间,尽量使用百分比持续时间,因为动画长短不一,容易导致在一个状态中绕不出去。退出时间也是同理。
分层动画
以上动画都工作在BaseLayer层,我的预期是用作下半身移动和空手时的移动操作。
接下来是WeaponLayer层,即玩家装备物品时会需要的层
这一层使用了AvatarMask进行蒙版,但在使用这玩意儿的时候遇到了个史诗级大坑。Mask的详细面板有两个条目,Humanoid和Transform,按照官方文档的说法,第一个可以大致选定蒙版的区域,手啊,躯干啊,头啊。第二个可以更加细致的调整哪些骨骼会收到影响。
我兴致勃勃的先用Humanoid进行蒙版,嗯,工作的很好。然后调整Transform让WeaponLayer只能影响到躯干的上半身。结果呢,是的,完全没有反应。
上网扒拉半天,得到的结果是:导入的骨骼网格体,如果使用Generic则可以使用Transform,使用Humanoid则可以使用Humanoid。搞了半天这个Mask的两个选项甚至不能同时用,而且和导入的骨骼设置有关。多有意思,我只好把骨骼改成Generic才能使用Transform。并且编辑这个Transform时,对号还得一个一个自己取消,也没个取消父级自动取消子级的。
是的,这就是Unity,随手一做就能发现一个6年前就有人在问的问题。
总之分层是可以工作了,这一层目前还没做多少东西,只是导了个武器和待机动作,更多东西等做出来过几天再发。

相关文章:
【Unity/Animator动画系统】多层动画状态机实现角色的基本移动
文章目录 前言实现顶层地面状态四方向混合树计算动画所需参数 空中状态分层动画 前言 最近打算做个Rougelike RPG 塔科夫 混搭风格的冒险游戏。暂且就当是一个有随机元素,有基地,死亡会掉落物品的近战塔科夫罢。 花了三天时间,整合了Mixa…...
每日算法一练:剑指offer——栈与队列篇(1)
1.图书整理II 读者来到图书馆排队借还书,图书管理员使用两个书车来完成整理借还书的任务。书车中的书从下往上叠加存放,图书管理员每次只能拿取书车顶部的书。排队的读者会有两种操作: push(bookID):把借阅的书籍还到图书馆。pop…...
【Java】ArrayList与LinkedList详解!!!
目录 一🌞、List 1🍅.什么是List? 2🍅.List中的常用方法 二🌞、ArrayList 1🍍.什么是ArrayList? 2🍍.ArrayList的实例化 3🍍.ArrayList的使用 4🍍.ArrayList的遍…...
怎么用VIM查看UVM源码
利用ctags工具可以建立源码的索引表,在使用VIM或其他文本编辑器时,就可以跳转查看所调用的UVM或VIP的funtcion/task/class等源码了。 首先需要确认ctags安装,一般安装VIM后都有,如果没有可以手动安装。在VIM中可以输入:help ctag…...
数据结构C语言描述3(图文结合)--双链表、循环链表、约瑟夫环问题
前言 这个专栏将会用纯C实现常用的数据结构和简单的算法;有C基础即可跟着学习,代码均可运行;准备考研的也可跟着写,个人感觉,如果时间充裕,手写一遍比看书、刷题管用很多,这也是本人采用纯C语言…...
第二十五章 TCP 客户端 服务器通信 - TCP 设备的 READ 命令
文章目录 第二十五章 TCP 客户端 服务器通信 - TCP 设备的 READ 命令TCP 设备的 READ 命令READ 修改 $ZA 和 $ZB$ZA 和 READ 命令 第二十五章 TCP 客户端 服务器通信 - TCP 设备的 READ 命令 TCP 设备的 READ 命令 从服务器或客户端发出 READ 命令以读取客户端或服务器设置的…...
【C++】哈希表的实现详解
哈希表的实现详解 一、哈希常识1.1、哈希概念1.2、哈希冲突1.3、哈希函数(直接定执 除留余数)1.4、哈希冲突解决闭散列(线性探测 二次探测)开散列 二、闭散列哈希表的模拟实现2.1、框架2.2、哈希节点状态的类2.3、哈希表的扩容2…...
高阶C语言之五:(数据)文件
目录 文件名 文件类型 文件指针 文件的打开和关闭 文件打开模式 文件操作函数(顺序) 0、“流” 1、字符输出函数fputc 2、字符输入函数fgetc 3、字符串输出函数fputs 4、 字符串输入函数fgets 5、格式化输入函数fscanf 6、格式化输出函数fpr…...
服务器上部署并启动 Go 语言框架 **GoZero** 的项目
要在服务器上部署并启动 Go 语言框架 **GoZero** 的项目,下面是一步步的操作指南: ### 1. 安装 Go 语言环境 首先,确保你的服务器上已安装 Go 语言。如果还没有安装,可以通过以下步骤进行安装: #### 1.1 安装 Go 语…...
【Java SE 】继承 与 多态 详解
🔥博客主页🔥:【 坊钰_CSDN博客 】 欢迎各位点赞👍评论✍收藏⭐ 目录 1. 继承 1.1 继承的原因 1.2 继承的概念 1.3 继承的语法 2. 子类访问父类 2.1 子类访问父类成员变量 2.1.1 子类与父类不存在同名成员变量 2.1.2 子类…...
【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法
【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法 目录 文章目录 【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法目录摘要:研究背景:问题与挑战:如何解…...
秋招大概到此结束了
1、背景 学院本,软工,秋招只有同程,快手和网易面试,后两家kpi(因为面试就很水),秋招情况:哈啰(实习转正ing),同程测开offer。 2、走测开的原因 很…...
华为OD机试真题---字符串化繁为简
华为OD机试真题中的“字符串化繁为简”题目是一个涉及字符串处理和等效关系传递的问题。以下是对该题目的详细解析: 一、题目描述 给定一个输入字符串,字符串只可能由英文字母(a~z、A~Z)和左右小括号((、)࿰…...
概念解读|K8s/容器云/裸金属/云原生...这些都有什么区别?
随着容器技术的日渐成熟,不少企业用户都对应用系统开展了容器化改造。而在容器基础架构层面,很多运维人员都更熟悉虚拟化环境,对“容器圈”的各种概念容易混淆:容器就是 Kubernetes 吗?容器云又是什么?容器…...
初识Arkts
创建对象: 类: 类声明引入一个新类型,并定义其字段、方法和构造函数。 定义类后,可以使用关键字new创建实例 可以使用对象字面量创建实例 在以下示例中,定义了Person类,该类具有字段name和surname、构造函…...
基本的SELECT语句
1.SQL概述 SQL(Structured Query Language)是一种用于管理和操作关系数据库的编程语言。它是一种标准化的语言,用于执行各种数据库操作,包括创建、查询、插入、更新和删除数据等。 SQL语言具有简单、易学、高效的特点,…...
51c自动驾驶~合集30
我自己的原文哦~ https://blog.51cto.com/whaosoft/12086789 #跨越微小陷阱,行动更加稳健 目前四足机器人的全球市场上,市场份额最大的是哪个国家的企业?A.美国 B.中国 C.其他 波士顿动力四足机器人 云深处 绝影X30 四足机器人 …...
Python Tutor网站调试利器
概述 本文主要是推荐一个网站:Python Tutor. 网站首页写道: Online Compiler, Visual Debugger, and AI Tutor for Python, Java, C, C++, and JavaScript Python Tutor helps you do programming homework assignments in Python, Java, C, C++, and JavaScript. It contai…...
h5小游戏实现获取本机图片
h5小游戏实现获取本机图片 本文使用cocos引擎 1.1 需求 用户通过文件选择框选择图片。将图片内容转换为Cocos Creator的纹理 (cc.Texture2D),将纹理设置到 cc.SpriteFrame 并显示到节点中。 1.2 实现步骤 创建文件输入框用于获取文件 let input document.createElement(&quo…...
前端 javascript a++和++a的区别
前端 javascript a和a的区别 a 是先执行表达式后再自增,执行表达式时使用的是a的原值。a是先自增再执行表达示,执行表达式时使用的是自增后的a。 var a0 console.log(a); // 输出0 console.log(a); // 输出1var a0 console.log(a); // 输出1 console.l…...
开局掌控者:EdB Prepare Carefully - RimWorld自定义体验革命
开局掌控者:EdB Prepare Carefully - RimWorld自定义体验革命 【免费下载链接】EdBPrepareCarefully EdB Prepare Carefully, a RimWorld mod 项目地址: https://gitcode.com/gh_mirrors/ed/EdBPrepareCarefully 副标题:如何告别随机开局…...
C#实战:基于WebAPI与Modbus构建EMS核心采集服务
1. 为什么需要EMS核心采集服务? 在工业现场,我们经常会遇到几十台甚至上百台智能电表、传感器等设备需要监控。这些设备可能来自不同厂家,使用不同的通信协议,数据格式也各不相同。想象一下,如果每个设备都需要单独开发…...
NTP配置避坑指南:华三/华为/思科设备时间同步差异对比
NTP配置避坑指南:华三/华为/思科设备时间同步差异对比 在网络运维中,时间同步是确保日志分析、安全审计和故障排查准确性的基础。不同厂商的设备在NTP配置上存在细微但关键的差异,这些差异往往成为混合环境部署中的"暗坑"。本文将深…...
Halcon一维码识别避坑指南:从模糊图像到精准解码
Halcon一维码识别实战:攻克模糊图像与复杂场景的五大策略 在物流分拣线上,传送带以每秒2米的速度运行,扫码枪却频繁报错——这不是设备故障,而是Halcon参数配置与图像预处理策略的缺失。当条形码出现在褶皱包装、反光表面或运动模…...
1949–2024年中国县级行政区划(逐年)|全国范围、75年连续、SHP格式
🔍 数据简介 本数据集完整覆盖 1949年至2024年 共 76个年份 的中国县级行政区划边界,是目前公开可获取的时间跨度最长、更新粒度最细的全国县级历史区划产品。 每一年份均提供独立、闭合、无重叠的面状矢量边界,属性表包含标准名称、行政区划…...
机器视觉C# 调用相机:从 USB 摄像头到海康工业相机(WinForms WPF)
🎥 机器视觉C# 调用相机:从 USB 摄像头到海康工业相机(WinForms & WPF) 📝 前言 在工业自动化、医疗影像或简单软件开发中,调用摄像头是一个绕不开的话题。在项目中同时遇到了两种需求: …...
【超全】基于Springboot多维分类的知识管理系统【包括源码+文档+调试】
💕💕发布人: 码上青云 💕💕各类成品Java毕设 。javaweb,ssm,springboot等项目,欢迎咨询。 💕💕程序开发、技术解答、代码讲解、文档, ἱ…...
手把手教你用4G Cat.1 bis开发智能硬件:从电路设计到低功耗优化的完整实战
4G Cat.1 bis智能硬件开发实战:从电路设计到低功耗优化的全流程指南 在共享充电宝扫码即用的便利背后,隐藏着一场关于低功耗通信的技术革命。当传统4G模块因高功耗让硬件开发者束手无策时,4G Cat.1 bis以单天线设计、10Mbps传输速率和μA级待…...
OpenClaw任务编排技巧:Qwen3.5-4B-Claude-4.6-Opus-Reasoning-Distilled-GGUF复杂流程分解策略
OpenClaw任务编排技巧:Qwen3.5-4B-Claude-4.6-Opus-Reasoning-Distilled-GGUF复杂流程分解策略 1. 为什么需要任务编排 上周我尝试用OpenClaw自动完成一篇技术博客的写作和发布,结果遭遇了连环翻车:模型先花20分钟生成了偏离主题的初稿&…...
手把手教你解决Unity视频播放问题:H264编码设置与RawImage的正确用法
Unity视频播放全攻略:H264编码优化与RawImage实战解析 在Unity项目开发中,视频播放功能看似简单,却暗藏诸多技术细节。许多开发者都曾遇到过视频不同步、颜色失真或性能低下的困扰。本文将深入剖析视频播放的核心技术要点,从编码格…...
