【unity】制作一个角色的初始状态(左右跳二段跳)【2D横板动作游戏】

前言

        hi~ 大家好!欢迎大家来到我的全新unity学习记录系列。现在我想在2d横板游戏中,实现一个角色的初始状态-闲置状态、移动状态、空中状态。并且是利用状态机进行实现的。

        本系列是跟着视频教程走的,所写也是作者个人的学习记录笔记。如有错误请联系我指正!

观看教程链接:https://www.udemy.com/course/2d-rpg-alexdev/

教程游戏资源链接:https://pan.baidu.com/s/1IlUbYlUB0LP0dQfQPkvjZA 
提取码:0721

目录

一、Unity和资源准备

二、状态机创建和Debug测试

1.有限状态机描述

2.有限状态机编码基础

三、动画控制器和动画组件

三、玩家初始状态制作

1.玩家闲置和移动状态的切换

2.玩家整体的翻转

3.玩家在地面状态和跳跃状态的切换

碰撞检测

4.玩家实现二段跳


一、Unity和资源准备

        下载Unity,制定好程序编辑器,创建2D项目,进入unity编辑器界面。(我使用的版本:2022.3.2f1c1)

        如上图所示,如果没有Console组件(程序控制台)、Animation(动画控制、动画)组件,如下图展示打开路径。

        然后将我们的游戏资源导入(游戏的美术资源),如下图:

        导入成功后,我们可以开始准备制作游戏了。

二、状态机创建和Debug测试

1.有限状态机描述

        我一开始就提到了状态机。这里的状态机指的是unity中的有限状态机,我们将使用它来控制接下来角色状态的转换。

        我们从角色状态来具体到有限状态机有什么作用。

        闲置状态、移动状态、跳跃状态。角色初始为闲置状态,我们通过键盘上的a和d键切换角色的状态,变为移动状态,在闲置和移动状态(地面状态)中可以通过space切换角色为跳跃状态,并且在第一段跳跃状态中可以切换为二段跳。

        可以看到,角色的状态有一个初始的状态,并且我们可以随时的进行切换状态,切换状态是存在条件的。并且一个状态实际上存在开始、执行、结束(起始,中间、结束)的过程,那么切换状态时均需要实现这些过程。

        切换状态过程我可以以跳跃距离,比如我们从闲置状态切换到跳跃状态,跳跃状态的初始过程就需要我们给予一个向上的速度,持续过程中不需要。

        上述描述的其实就是有限状态机。做的就是将有限的一些状态通过条件决定切换。可以发现,通过这个,我们实际上就可以完成每个状态的独立。可以想象,如果不存在状态机,那么我们在编写角色状态切换时,就需要同时考虑到其他状态的情况,比如攻击时不可移动等,那么随着新的状态加入,我们需要自己写的限制条件就会越来越多,导致编码困难。

2.有限状态机编码基础

        回到我们的当前游戏上。有限状态机要完成的是状态的切换,我们需要两个组件:状态机(StateMachine)和对应的状态类型(PlayerState)。另外,我们需要unity中的组件,对他们进行一个调用,此组件就是Player玩家组件(Player)。

        括号中就是我们需要进行的C#编程文件。C#是一种面向对象语言,接下来我们实现状态机的过程很多就是用的面向对象的思想(类、继承、多态)。

[Assets->Script]

        首先说明一下编码的目的和基本过程:我们需要通过Player来获取游戏对象的组件,并且能够在游戏开始时通过其Unity脚本达成每帧调用(MonoBehaviour)。而我们的状态并不需要继承MonoBehaviour类,也就不会参与到游戏的调用中(顺便也节约了资源)。另外,状态机设置初始阶段和切换状态(注意其中的三个过程:开始、中间、结束)。而Player状态则表示我们操作的角色所有状态的父类,它不继承任何类(状态机也是),可以通过多态操作,让角色状态做一些共同的事情(比如随时检测玩家的Xspeed水平方向的输入),这也是多态的目的。最后在Player中完成对这些状态对象的创建,通过状态机在update中完成对某一状态的随帧调用,然后在具体的状态中检测条件完成切换。

        实际上,有时候状态的切换,此状态可能时一些状态的集合,比如闲置和移动状态,它们都是在地面上,地面上随时可以进行跳跃。通过继承和多态我们可以随便完成这些集合切换的要求(多态->子类对象调用重写的方法时会执行父类的被重写函数方法)

        利用下面一张图解释上面的说法:

        我们首先创建角色的闲置、移动状态(PlayerState的子类),在Player中完成基础的调用。代码如下:

PlayerState

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class PlayerSate
{protected Player player;  // 方便调用游戏中获取的资源protected StateMachine stateMachine;  // 游戏状态机,实现状态的切换protected string stateBoolName;  // 状态名字 - 和后面游戏对象的动画机组件相关public PlayerSate(Player _player, StateMachine _stateMachine, string _stateBoolName){player = _player;stateMachine = _stateMachine;stateBoolName = _stateBoolName;}// 开始状态public virtual void Enter(){}// 中间状态public virtual void Update(){}// 退出状态public virtual void Exit(){}
}

StateMachine

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class StateMachine
{public PlayerSate currentState { get; private set; }  // 想让外界访问,但是不允许修改// 初始状态public void Initialize(PlayerSate _startState){currentState = _startState;currentState.Enter();  // 启动开始阶段}// 转换状态public void ChangeState(PlayerSate _newState){currentState.Exit();  // 前一个状态先退出currentState = _newState;currentState.Enter();  // 执行开始阶段}}

Player 

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class Player : MonoBehaviour
{#region Componentprivate StateMachine stateMachine;#endregion#region State  // C#语法,可以整理段落public PlayerIdleState idleState { get; private set; }  // 后续在状态中可能会调用到,但是不希望被修改public PlayerMoveState moveState { get; private set; }#endregion// 初始化变量private void Awake(){stateMachine = new StateMachine();idleState = new PlayerIdleState(this, stateMachine, "Idle");moveState = new PlayerMoveState(this, stateMachine, "Move");}void Start(){stateMachine.Initialize(idleState);  // 一开始为闲置状态}void Update(){stateMachine.currentState.Update();  // 多态调用 父类对象调用重写方法,实现同种类型展示出不同的效果}}

        然后我们在Unity层次界面内创建空对象,将Player脚本挂接上去。这样游戏加载时就能加载Player脚本代码。点击游戏窗口,敲f键时观察代码控制台,查看状态的切换。

        注意红色标注,按此时控制台显示的重复信息会显示成数字置于消息框右下角,避免刷屏。

        另外,在脚本编辑过程中,子类生成虚函数和构造函数时,点击选中当前类名,alt + enter键可以快速构造:

        可以看到,此时我们的脚本已经可以完成基础的转换的演示,那么我们让它实际运行起来,展示对玩家基础状态的演示。

三、动画控制器和动画组件

        在正式制作之前,先介绍一下Unity内的动画组件。窗口的创建先前布局时已经说过,其中Animator就是动画控制器,在这里可以制作动画切换条件,动画顺序等。那么在动画控制之前,我们还需要一段一段的动画,通过Animation进行控制(将美术资源中的动画一张一张连续播放出来)。

        我们在美术资源中找到角色精灵图:

        在此精灵图中,实际上已经都切分好了,这里简单介绍一下如何切分精灵图的(本质是由一张一张的图组合而成,到Unity中进行切分而已)

        模式中我们选中Multiple多个模式,选择图片编辑器(Sprite Editor)进行编辑

        在如何切分中我们可以选择三种进行切分,看好如何切分,这也和美术资源整理时相关,尽量每张子图一致,Point表示这张图的支点(有时图片不一致时利用此存在对齐的效果)。切分好后按apply进行应用。

        应用完后,Pixels Per Unit表示单位像素数,在此可以设计图片大小......

        我们将其中一张闲置图片拖到场景界面上,此时Unity会为我们默认创建一个对象(命名为Animator),将其挂载在Player对象下成为子对象,调整子对象的距离,将我们父对象的中心点对号角色对象的中心点。

        为Animator组件添加 Animator组件,其就是动画控制的组件,另外此对象为空,我们需要在文件中进行创建不同的动画控制对象,以便不同的对象控制不同的状态动画。

上图为创建玩家控制器的过程(可以利用文件夹进行分类)

        将对象托拽到玩家子对象Animator组件Animator上去,在选择此对象的状态的条件下查看动画控制窗口(Animator)就可以看到如下图:

        这个就很容易看出和我们的状态机对应的关系,但是存在一个任意状态。也就是说,在没有状态机的条件下我们也可以进行创作角色状态转换,只不过互相之间的制约需要玩家自己控制,非常麻烦。

        最后在看一下某一段动画编辑。选中玩家动画控制器的情况下,点击下图中的Create就可以创建一段动画序列。

        将角色一段闲置动画序列拖进此窗口,在此窗口的右边三小点上选择Show Sample Rate来控制动画的播放速度。

 

可以像上图那样观察角色动画状态。

        移动类似创建。这样在动画控制窗口我们可以创建条件来决定动画状态的切换了。其中右键角色状态,选择make transition创建动画过渡连线,在右边窗口的Animation的+创建条件,因为状态机的切换与Player组件挂钩使用通用的Bool,所以创建两个条件Idle和Move作为切换条件。点击连线选择条件切换,控制前后两端动画的退出状态和过度状态,这里两个切换我们均不设置退出时间和转换持续时间。

         就这样,对动画序列和动画控制的基本了解到此结束,需要更详细的了解请前去学习。

三、玩家初始状态制作

        首先,我们需要在player玩家脚本上获取角色的动画控制组件(以便设置条件为true控制动画的播放),每次在切换或者初始状态的时对条件进行控制(公共操作,在父类上进行)。另外,我们需要我们的角色拥有重力,所以需要Unity中的组件Rigidbody 2D对玩家进行基本的物理控制。所以对Player(注意获取的组件还存在子类的组件)脚本和PlayerState脚本的修改如下:

player:

PlayerState:

        设置玩家物理控制组件的基本状态:

        禁止Z轴旋转(这和我们重力下降相关),同时需要调整参数插值和重力检测为连续。(因为重力元素,作用到碰撞器上后,角色由于形状因素,由于是2D平面,导致z轴旋转让其倒下)

        然后创建一个简单平台,增加角色和平台的碰撞器(存在碰撞器才能发生碰撞以及检测)。需要注意碰撞器的类型和2D状态。效果如下:

         初始状态研究完毕,开始游戏我们就会看到玩家角色掉落,并且不会发生图片z轴翻转。

1.玩家闲置和移动状态的切换

        动画序列和动画控制我们在之前的步骤已经完成,现在我们只需要在脚本里进行控制即可实现此状态的切换。

        由于状态机初始状态就是闲置状态,一开始我们角色就应该播出闲置动画。在闲置的动画状态类里,由于此时状态是此,在游戏对象的随帧Update调用中,当我们检测到玩家按下了a和d键(Unity中存在Horizontal对AD的检测,使用方法GetAxis进行检测即可)时,应该从闲置状态切换到移动状态。但是即然存在移动,我们就需要方向上的确定,获取的ad状态存在方向,由于多种状态需要此值,可以设计xSpeed检测方向于PlayerState上。

        移动状态需要进行移动,我们通过Player脚本获取其脚本进行速度位移。但是速度位移的很多状态下都需要进行控制,我们将其方法设计到Player下,形参传递x和y方向上的速度即可。初始和中间过程均需要维持速度。当检测到xSpeed为0时,移动状态应该转换为闲置状态,并且闲置状态的速度应该为0,所以在闲置状态的开始状态速度设置为0。

        由于我们需要能够在Unity中自由控制玩家的速度,在Player声明空开变量MoveSpeed来控制角色位移速度。 

        脚本编写:

Player:

 

PlayerState:

PlayerIdleState: 

PlayerMoveState:

        我们试着运行一下:

        可以看到我们实现了角色闲置状态和移动状态的切换。但是此时发现一个bug,角色图片只有向右的动画,没有向左的动画,需要重写制作动画设置条件吗?并不需要,我们利用脚本就可以控制。 

2.玩家整体的翻转

        我们是不是只需要角色在往左移动的时候将角色整体向左翻转即可?

        那么我们只需要控制角色方向切换移动时能够进行控制翻转,而我们输入的xSpeed正好可以管控角色输入的方向。但是需要注意,翻转时比如x<0向左翻转,但是必须控制当前情况需要为向右的状态,如果本来是向左的状态那么翻转就失败了。

        控制是否左右的变量我们设置在Player(isRight bool),并且为了之后的碰撞检测线的翻转问题,我们留下positionDir指定玩家-1为左移动,1为右移动。初始均为右移动。因为翻转是随时的,并且在改变速度时才可翻转,那么我们设置在Player上的意义又多了一个。将Dir设置为公开,isRight设置为私有并且为true。start时检测外界是否修改,修改需要设置对应false。这样翻转才正确。

Player:

修改上述文件即可,我们查看效果即可。 

3.玩家在地面状态和跳跃状态的切换

        在之前的状态机中我们已经谈过,我们处于地面状态时(移动和闲置),均可以进行跳跃。利用类与对象基础的关系,我们床在移动和闲置类的父类,玩家状态类的子类,由此地面类进行控制玩家从地面状态到跳跃状态的切换。

        跳跃状态在一开始需要一个向上的速度,我们需要空开jump跳跃数据进行控制。但是跳跃状态存在两种动画:向上跳动画+向下跳动画。那么我们如何控制这两种动画的切换呢?

        首先通过动画序列创建向上跳和向下跳的动画序列(之前的资源进行寻找)。然后在动画控制界面我们做一点不一样的事情,此时条件变成了一个float数,从1到-1.我们通过创建混合树的方式,控制角色在空中时的状态:

        然后玩家跳跃后在PlayerState可为其添加角色下落速度方向为其修改(整个空中状态)。

        因为此时为状态树模式,我们控制的状态在同一时刻存在一种状态,所以常见的无限跳bug也解决了,但是玩家落地的时候如何判定为落地状态呢?能只是简单的判断y方向上为0吗?比如玩家滑行模式中(fly?)y方向速度为0不应该为闲置状态,我们需要的是碰撞检测。

碰撞检测

        我们需要实现一个地面碰撞检测,为了方便后续,我们同步实现墙体检测(滑墙和登墙跳操作)。

        地面检测我们通过Unity的Physics2D中的Raycast向量检测进行。此函数能够在空间中确定一个从起点到上、下、左、右方向的距离的一个向量,并且可指定layerMask(层级蒙版)来检测特定的layer,当此向量检测到存在对应的碰撞体时,就会返回相应个数(实际我们只需要知道碰撞到即可)

        墙面检测类似。那么我们想要将其形象化的表示出来才能进行更好的调节参数。表示出来我们可以使用Gizmos中的DrawLine功能,就能进行绘画向量,由起始和终点位置决定。并且为了更好的进行调节实现,起点位置的GameObject可以公开出去,这样我们可以自由决定检测的起始位置。

        另外,你是否存在这种想法:玩家跳跃时表明一个状态即可,墙体检测IsGroundCheck随时在update。是的,如果只是维持一个文件表明此跳跃空中状态的化,由于初始给予向上速度和IsGroundCheck可能重叠,导致跳跃失败的情况出现。(可自行测试)

        综上,我们的设计思路如下:(分成两个文件表示跳跃的上跳和下跳也和之后的实现相关)

 代码整理:

        首先文件确定如下:

        PlayerAirState文件表示空中状态,为PlayerJumpState和PlayerFallState的父类,能执行一些共同的事情,比如update检测YDir,空中能随时的确定x速度和方向,后续的多段跳功能。JumpState表示上跳过程,Enter给予向上的速度,Player脚本设计JumpSpeed确定数值。update只需检测如果玩家刚体速度小于0进行切换到下落状态。FallState表示下落过程,注意Update检测是否地面碰撞。(分开的主要原因错开初始向上速度和地面检测重叠一起,导致跳跃失败)

        Player文件new上述状态,并且提供Draw方法和向量检测,暴露一些属性决定位置和数值大小。

Player:

 

PlayerAirState:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class PlayerAirState : PlayerState
{public PlayerAirState(Player _player, StateMachine _stateMachine, string _stateBoolName) : base(_player, _stateMachine, _stateBoolName){}public override void Enter(){base.Enter();}public override void Exit(){base.Exit();}public override void Update(){base.Update();player.SetVelocity(xDir * player.moveSpeed, player.rb.velocity.y);// 随时控制跳跃的上跳和下跃状态player.animator.SetFloat("YDir", player.rb.velocity.y);}
}

 PlayerJumpState:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class PlayerJumpState : PlayerAirState
{public PlayerJumpState(Player _player, StateMachine _stateMachine, string _stateBoolName) : base(_player, _stateMachine, _stateBoolName){}public override void Enter(){base.Enter();// 初始给玩家一个向上跳的动作player.SetVelocity(player.rb.velocity.x, player.jumpSeed);}public override void Exit(){base.Exit();}public override void Update(){base.Update();if (player.rb.velocity.y < 0) stateMachine.ChangeState(player.fallState);}
}

PlayerFallState:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class PlayerFallState : PlayerAirState
{public PlayerFallState(Player _player, StateMachine _stateMachine, string _stateBoolName) : base(_player, _stateMachine, _stateBoolName){}public override void Enter(){base.Enter();}public override void Exit(){base.Exit();}public override void Update(){base.Update();// 地面检测,如果碰撞到了地面,转换为闲置状态if (player.IsGroundCheck()) stateMachine.ChangeState(player.idleState);}
}

 Player中地面和墙体检测表现效果:

4.玩家实现二段跳

        那么我们要实现二段跳如何实现?设想一下,是不是我们只要能在跳跃状态中在检测一下空格键,转换为跳跃状态是否就能实现多段跳功能?

        但是只是上述那个条件,就可以造成无限跳了。为了限制为2段跳,我们不妨设计一个计数器,记录为2个,每往上跳一次--,落地时恢复为2,在空中状态中按下空格键时检测计数器是否>0即可。那么此计数器必须在状态切换时始终唯一,所以该变量就设在Player中去,Jump文件控制--,Fall文件确定恢复。双段跳就可以简单的实现出来了:

Player:

PlayerAirState:

 

 PlayerJumpState:

PlayerFallState:

 

实际效果:

 

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

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

相关文章

C#对字典容器Dictionary<TKey, TValue>内容进行XML序列化或反序列化报错解决方法

一、问题描述 在使用C#对字典容器Dictionary<TKey, TValue>内容进行XML序列化报错【System.Exception:“不支持类型 System.Collections.Generic.Dictionary2[[System.String, mscorlib, Version2.0.0.0, Cultureneutral, PublicKeyTokenb77a5c561934e089],[System.Strin…

【广州华锐互动】车辆零部件检修AR远程指导系统有效提高维修效率和准确性

在快速发展的科技时代&#xff0c;我们的生活和工作方式正在被重新定义。这种变化在许多领域都有所体现&#xff0c;尤其是在汽车维修行业。近年来&#xff0c;AR&#xff08;增强现实&#xff09;技术的进步为这个行业带来了前所未有的可能性。通过将AR技术与远程协助系统相结…

尚品甄选2023全新SpringBoot+SpringCloud企业级微服务项目

最适合新手入门的SpringBootSpringCloud企业级微服务项目来啦&#xff01;如果你已经学习了Java基础、SSM框架、SpringBoot、SpringCloud&#xff0c;想找一个项目来实战练习&#xff1b;或者你刚刚入行&#xff0c;需要可以写到简历中的微服务架构项目&#xff01; 项目采用前…

基于springboot实现在线动漫信息交流分享平台项目【项目源码+论文说明】计算机毕业设计

基于springboot实现在线动漫信息交流分享平台演示 摘要 随着社会互联网技术的快速发展&#xff0c;每个行业都在努力与现代先进技术接轨&#xff0c;通过科技手段提高自身的优势&#xff1b;对于在线动漫信息平台当然也不能排除在外&#xff0c;随着网络技术的不断成熟&#x…

【LeetCode 算法专题突破】二分查找(⭐)

文章目录 前言1. 二分经典模板题目题目描述代码&#xff1a; 2. 在排序数组中查找元素的第一个和最后一个位置题目描述代码 3. 有效的完全平方数题目描述代码 4. 寻找峰值题目描述代码 5. 寻找旋转排序数组中的最小值题目描述代码 6. 点名题目描述代码 总结 前言 我刷过不少算…

Visual Studio Code配置C/C++开发环境

C/C开发中的IDE非常多&#xff0c;网上有推荐安装Visual Studio 2019/2020/2022。但是登录官方网址下载&#xff0c;此软件体积非常大(8G以上)&#xff0c;且企业版、专业版会收费。 因此&#xff0c;我们推荐大家可以尝试通过Visual Studio Code来配置C/C开发环境 环境准备 Mi…

ThreeJS-3D教学六-物体位移旋转

之前文章其实也有涉及到这方面的内容&#xff0c;比如在ThreeJS-3D教学三&#xff1a;平移缩放物体沿轨迹运动这篇中&#xff0c;通过获取轨迹点物体动起来&#xff0c;其它几篇文章也有旋转的效果&#xff0c;本篇我们来详细看下&#xff0c;另外加了tween.js知识点&#xff0…

【置顶】关于博客的一些公告

所谓 万事开头难&#xff0c;最开始的两个专栏 《微机》 和 《骨骼动作识别》 定价 29.9 &#xff0c;因为&#xff1a; 刚开始确实比较困难&#xff0c;要把自己学的知识彻底搞懂讲给别人&#xff0c;还要 码字排版&#xff0c;从 Markdown 语法开始学起&#xff08;这都是 花…

【C++】Stack Queue -- 详解

一、stack的介绍和使用 1、stack的介绍 https://cplusplus.com/reference/stack/stack/?kwstack 1. stack 是一种容器适配器&#xff0c;专门用在具有后进先出操作的上下文环境中&#xff0c;其删除只能从容器的一端进行元素的插入与提取操作。 2. stack 是作为容器适配器被…

bin-editor-next实现josn序列化

线上链接 BIN-EDITOR-NEXThttps://wangbin3162.gitee.io/bin-editor-next/#/editor gitee地址bin-editor-next: ace-editor 的vue3升级版本https://gitee.com/wangbin3162/bin-editor-next#https://gitee.com/link?targethttps%3A%2F%2Funpkg.com%2Fbin-editor-next%2F 实现…

如何在Apache和Resin环境中实现HTTP到HTTPS的自动跳转:一次全面的探讨与实践

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

【SpringCloud】Ribbon负载均衡原理、负载均衡策略、饥饿加载

&#x1f40c;个人主页&#xff1a; &#x1f40c; 叶落闲庭 &#x1f4a8;我的专栏&#xff1a;&#x1f4a8; c语言 数据结构 javaEE 操作系统 Redis 石可破也&#xff0c;而不可夺坚&#xff1b;丹可磨也&#xff0c;而不可夺赤。 Ribbon 一、 Ribbon负载均衡原理1.1 负载均…