【unity实战】使用unity的新输入系统InputSystem+有限状态机设计一个玩家状态机控制——实现玩家的待机 移动 闪避 连击 受击 死亡状态切换

最终效果

在这里插入图片描述

文章目录

  • 最终效果
  • 前言
  • 人物素材
  • 新输入系统InputSystem的配置
  • 动画配置
  • 代码文件路径
  • 状态机脚本
  • 创建玩家不同的状态脚本
  • 玩家控制
  • 源码
  • 完结

前言

前面我们已经写过了使用有限状态机制作一个敌人AI:【unity实战】在Unity中使用有限状态机制作一个敌人AI

那么玩家的状态机要怎么做呢?目前网上这一块内容也很少,当我们对人物的操作越来越多时,有限状态机技术可以很好的帮我们将各部分功能拆开,单独配置逻辑,代码更加优雅,接下来我们就用新输入系统InputSystem设计一个玩家状态机控制系统,其实跟之前的敌人有限状态机类似。

人物素材

https://bdragon1727.itch.io/16x16-pixel-adventures-character
在这里插入图片描述

新输入系统InputSystem的配置

新输入系统还不会使用的可以参考这篇文章:【推荐100个unity插件之18】Unity 新版输入系统InputSystem的基础使用

其实就是默认的配置加了攻击和闪避的操作
在这里插入图片描述

动画配置

动画基础知识:【Unity游戏开发教程】零基础带你从小白到超神27——混合状态,混合动画,动画分类

除了攻击动画,其他的都放在第一层
在这里插入图片描述
闪避动画我是通过不断修改玩家图片的FlipY值实现的
在这里插入图片描述

重点讲讲攻击动画连击
两个参数控制进入,isMeleeAttack主要是有效防止播放最后一段连击后,再播放一次第一段攻击
在这里插入图片描述
如果动画播放90%再次按下就会进入下一段攻击
在这里插入图片描述
所有动画播放为1,即播放完时退出
在这里插入图片描述

代码文件路径

在这里插入图片描述

状态机脚本

定义状态类型枚举

// 定义状态类型枚举
public enum StateType
{
    Idle, //待机
    Move, //移动
    Dodge, //闪避
    MeleeAttack, //近战攻击
    Hit, //受击
    Death //死亡
}

抽象基类,定义了所有状态类的基本结构

//抽象基类,定义了所有状态类的基本结构

public abstract class IState
{
    protected FSM manager;// 当前状态机
    protected Parameter parameter;// 参数
    public abstract void OnEnter();// 进入状态时的方法
    public abstract void OnUpdate();// 更新方法
    public abstract void OnFixedUpdate();// 固定更新方法
    public abstract void OnExit();// 退出状态时的方法
}

可序列化的参数类,存储了角色的各种状态参数和配置

// 可序列化的参数类,存储了角色的各种状态参数和配置
using System;
using UnityEngine;

[Serializable]
public class Parameter
{
    
    [Header("属性")]
    public float health; // TODO:生命值 仅仅用于测试,实际生命值可能并不放在这里
    [HideInInspector] public Animator animator;       // 角色动画控制器
    [HideInInspector] public AnimatorStateInfo animatorStateInfo;    // 动画状态信息
    [HideInInspector] public SpriteRenderer sr; // 精灵渲染器
    [HideInInspector] public Rigidbody2D rb; // 刚体
    [HideInInspector] public PlayerSystem inputSystem;//新的输入系统

    [Header("移动")]
    public float normalSpeed = 3f; // 默认移动速度
    public float attackSpeed = 1f; // 攻击时的移动速度
    [HideInInspector] public Vector2 inputDirection; // 输入的移动方向
    [HideInInspector] public float currentSpeed; // 当前移动速度

    [Header("攻击")]
    public float meleeAttackDamage; // 近战攻击造成的伤害
    [HideInInspector] public bool isMeleeAttack; // 是否进行近战攻击

    [Header("闪避")]
    public float dodgeForce; // 闪避的力量
    public float dodgeCooldown = 2f; // 闪避的冷却时间
    [HideInInspector] public bool isDodging = false; // 是否在闪避中
    [HideInInspector] public bool isDodgeOnCooldown = false; // 是否在闪避冷却中

    [Header("受伤与死亡")]
    [HideInInspector] public bool isHurt; // 是否受伤
    [HideInInspector] public bool isDead; // 是否死亡
    [HideInInspector] public bool getHit;             // 是否被击中
}

新增玩家状态机

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.InputSystem;

// 玩家有限状态机类
public class FSM : MonoBehaviour
{
    private IState currentState;        // 当前状态接口
    protected Dictionary<StateType, IState> states = new Dictionary<StateType, IState>();  // 状态字典,存储各种状态

    public Parameter parameter;  // 状态机参数

    public virtual void Awake()
    {
        parameter.rb = GetComponent<Rigidbody2D>();
        parameter.animator = GetComponent<Animator>();
        parameter.sr = GetComponent<SpriteRenderer>();

        // 初始化各个状态,并添加到状态字典中
        states.Add(StateType.Idle, new IdleState(this));
        states.Add(StateType.Move, new MoveState(this));
        states.Add(StateType.Dodge, new DodgeState(this));
        states.Add(StateType.MeleeAttack, new MeleeAttackState(this));
        states.Add(StateType.Hit, new HitState(this));
        states.Add(StateType.Death, new DeathState(this));

        TransitionState(StateType.Idle);    // 初始状态为Idle
    }

    public virtual void OnEnable()
    {
        currentState.OnEnter();
    }

    public virtual void Update()
    {
        //有效防止播放最后一段连击后,再播放一次第一段攻击
        parameter.animator.SetBool("isMeleeAttack", parameter.isMeleeAttack);

        parameter.animatorStateInfo = parameter.animator.GetCurrentAnimatorStateInfo(0);// 获取当前动画状态信息

        currentState.OnUpdate();
    }

    public virtual void FixedUpdate()
    {
        currentState.OnFixedUpdate();
    }

    // 状态转换方法
    public void TransitionState(StateType type)
    {
        if (currentState != null)
            currentState.OnExit();// 先调用退出方法

        currentState = states[type];    // 更新当前状态为指定类型的状态
        currentState.OnEnter();         // 调用新状态的进入方法
    }

    // 切换操作映射
    public void SwitchActionMap(InputActionMap actionMap)
    {
        parameter.inputSystem.Disable(); // 禁用当前的输入映射
        actionMap.Enable(); // 启用新的输入映射
    }

    public void Move()
    {
        // 根据当前状态设置角色速度
        parameter.currentSpeed = parameter.isMeleeAttack ? parameter.attackSpeed : parameter.normalSpeed;
        // 设置角色刚体的速度为移动方向乘以当前速度
        parameter.rb.velocity = parameter.inputDirection * parameter.currentSpeed;

        FlipTo();
    }
    
    // 翻转角色
    public void FlipTo()
    {
        if (parameter.inputDirection.x < 0)
        {
            transform.localScale = new Vector3(-1, 1, 1);
        }
        if (parameter.inputDirection.x > 0)
        {
            transform.localScale = new Vector3(1, 1, 1);
        }
    }

    // 开始闪避技能冷却的协程
    public void DodgeOnCooldown()
    {
        StartCoroutine(nameof(DodgeOnCooldownCoroutine));
    }

    public IEnumerator DodgeOnCooldownCoroutine()
    {
        yield return new WaitForSeconds(parameter.dodgeCooldown);// 等待闪避冷却时间
        parameter.isDodgeOnCooldown = false;// 闪避技能冷却结束,设置为不在冷却状态
    }
}

创建玩家不同的状态脚本

在这里插入图片描述
待机状态

using UnityEngine;
/// <summary>
/// 待机状态
/// </summary>

public class IdleState : IState
{
    public IdleState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }

    public override void OnEnter()
    {
        parameter.animator.Play("Idle");
    }
    public override void OnUpdate()
    {
        // parameter.anim.SetFloat("speed", parameter.rb.velocity.magnitude);
        // 如果受伤
        if (parameter.isHurt)
        {
            manager.TransitionState(StateType.Hit);
        }

        //如果输入的移动方向不为0
        if (parameter.inputDirection != Vector2.zero)
        {
            manager.TransitionState(StateType.Move);
        }
        //闪避
        if (parameter.isDodging)
        {
            manager.TransitionState(StateType.Dodge);
        }
        //真正近战攻击
        if (parameter.isMeleeAttack)
        {
            manager.TransitionState(StateType.MeleeAttack);
        }
    }

    public override void OnFixedUpdate() { }

    public override void OnExit() { }
}

移动状态


/// <summary>
/// 移动状态
/// </summary>
public class MoveState : IState
{
    public MoveState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }

    public override  void OnEnter()
    {
        parameter.animator.Play("Run");
    }

    public override  void OnUpdate()
    {
        // 如果想按速度切换移动或奔跑动画
        // parameter.animator.SetFloat("speed", player.rb.velocity.magnitude);
        //受伤
        if (parameter.isHurt)
        {
            manager.TransitionState(StateType.Hit);
        }
        //速度为0
        if (parameter.rb.velocity.magnitude < 0.01f)
        {
            manager.TransitionState(StateType.Idle);
        }
        //闪避
        if (parameter.isDodging)
        {
            manager.TransitionState(StateType.Dodge);
        }
        //近战攻击
        if (parameter.isMeleeAttack)
        {
            manager.TransitionState(StateType.MeleeAttack);
        }

    }
    public override  void OnFixedUpdate()
    {
        manager.Move();
    }
    public override  void OnExit() { }
}

近战攻击状态

/// <summary>
/// 近战攻击状态
/// </summary>
public class MeleeAttackState : IState
{
    public MeleeAttackState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }

    public override  void OnEnter()
    {
        parameter.animator.SetTrigger("MeleeAttack");
    }
    public override  void OnUpdate()
    {
        //受伤
        if (parameter.isHurt)
        {
            manager.TransitionState(StateType.Hit);
        }

        // 动画播放95%切换到待机状态
        if (parameter.animatorStateInfo.normalizedTime >= .95f)
        {
            manager.TransitionState(StateType.Idle);
        }
    }
    public override  void OnFixedUpdate()
    {
        manager.Move();
    }
    public override  void OnExit()
    {
        parameter.isMeleeAttack = false;
    }
}

闪避状态

using System.Collections;
using UnityEngine;
/// <summary>
/// 闪避状态
/// </summary>
public class DodgeState : IState
{
    public DodgeState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }

    public override void OnEnter()
    {
        parameter.animator.Play("Dodge");
    }
    public override void OnUpdate()
    {
        //受击
        if (parameter.isHurt)
        {
            manager.TransitionState(StateType.Hit);
        }
        //动画播放95%切换到待机状态
        if (parameter.animatorStateInfo.normalizedTime >= .95f)
        {
            manager.TransitionState(StateType.Idle);
        }
    }

    public override void OnFixedUpdate()
    {
        manager.Move();
        if(parameter.isDodging) Dodge();
    }

    public override void OnExit() {
        parameter.isDodging = false;
    }

    // 进行闪避操作的方法
    public void Dodge()
    {
        // 施加闪避力量,根据输入方向和设定的闪避力量
        parameter.rb.AddForce(parameter.inputDirection * parameter.dodgeForce, ForceMode2D.Impulse);
        parameter.isDodgeOnCooldown = true;
        manager.DodgeOnCooldown();// 开始闪避冷却
    }
}

受击状态

/// <summary>
/// 受击状态
/// </summary>
public class HitState : IState
{
    public HitState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }

    public override void OnEnter()
    {
        parameter.animator.Play("Hit");
        //TODO:仅用于测试
        parameter.health --;    // 减少角色生命值
    }
    public override  void OnUpdate()
    {
        //TODO:仅用于测试,如果角色生命值小于等于0,转换到死亡状态
        if (parameter.health <= 0)
        {
            manager.TransitionState(StateType.Death);
        }

        //动画播放95%切换到待机状态
        if (parameter.animatorStateInfo.normalizedTime >= .95f)
        {
            manager.TransitionState(StateType.Idle);
        }
    }
    public override  void OnFixedUpdate()
    {
        manager.Move();
    }

    public override  void OnExit()
    {
        parameter.isHurt = false;
    }
}

死亡状态

/// <summary>
/// 死亡状态
/// </summary>
public class DeathState : IState
{
    public DeathState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }

    public override void OnEnter()
    {
        parameter.isDead = true;
        manager.SwitchActionMap(parameter.inputSystem.UI);//切换为UI输入
        parameter.animator.Play("Dead");
    }

    public override void OnUpdate() { }

    public override void OnFixedUpdate() { }

    public override void OnExit() { }
}

玩家控制

新增PlayerController,获取玩家的输入

using UnityEngine;
using UnityEngine.InputSystem;

public class PlayerController : FSM {
    public override void Awake() {
        base.Awake();

        // 初始化输入控制
        parameter.inputSystem = new PlayerSystem();
        parameter.inputSystem.Player.Move.performed += Move;
        parameter.inputSystem.Player.Move.canceled += StopMove;
        parameter.inputSystem.Player.Dodge.started += Dodge;
        parameter.inputSystem.Player.MeleeAttack.started += MeleeAttack;
        
        SwitchActionMap(parameter.inputSystem.Player); // 切换到游戏操作的输入映射
    }

    void OnDisable()
    {
        // 禁用所有输入
        parameter.inputSystem.Disable();
    }

    public override void Update()
    {
        //TODO:用于测试 如果按下回车键,设置被击中状态为true
        if (Input.GetKeyDown(KeyCode.Return))
        {
            // PlayerHurt();
            parameter.isHurt = true;
        }

        base.Update();
    }

    public void Move(InputAction.CallbackContext context)
    {
        parameter.inputDirection = parameter.inputSystem.Player.Move.ReadValue<Vector2>();   
    }

    // 停止移动,将输入方向设为零向量
    public void StopMove(InputAction.CallbackContext context)
    {
        parameter.inputDirection = Vector2.zero;
    }

    // 触发闪避的方法
    public void Dodge(InputAction.CallbackContext context)
    {
        //如果当前不在冷却中,则开始闪避
       if(!parameter.isDodgeOnCooldown) parameter.isDodging = true;
    }

    // 近战攻击
    public void MeleeAttack(InputAction.CallbackContext context)
    {
        parameter.isMeleeAttack = true;
    }
}

效果
在这里插入图片描述

源码

整理好了我会放上来

完结

赠人玫瑰,手有余香!如果文章内容对你有所帮助,请不要吝啬你的点赞评论和关注,以便我第一时间收到反馈,你的每一次支持都是我不断创作的最大动力。当然如果你发现了文章中存在错误或者有更好的解决方法,也欢迎评论私信告诉我哦!

好了,我是向宇,https://xiangyu.blog.csdn.net

一位在小公司默默奋斗的开发者,出于兴趣爱好,最近开始自学unity,闲暇之余,边学习边记录分享,站在巨人的肩膀上,通过学习前辈们的经验总是会给我很多帮助和启发!php是工作,unity是生活!如果你遇到任何问题,也欢迎你评论私信找我, 虽然有些问题我也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

收银系统源码分享-PHP可二开

千呼新零售2.0系统是零售行业连锁店一体化收银系统&#xff0c;包括线下收银线上商城连锁店管理ERP管理商品管理供应商管理会员营销等功能为一体&#xff0c;线上线下数据全部打通。 适用于商超、便利店、水果、生鲜、母婴、服装、零食、百货、宠物等连锁店使用。 私有化独立…

面向对象-封装

一.包 1.简介 当我们把所有的java类都写src下的第一层级&#xff0c;如果是项目中&#xff0c;也许会有几百个java文件。 src下的文件会很多&#xff0c;开发的时候不方便查找&#xff0c;也不方便维护如果较多的文件中有同名的&#xff0c;十分麻烦 模块1中有一个叫test.ja…

Nuxtjs3教程

起步 官方文档 官方目录结构 安装 npx nuxi@latest init <project-name>后面跟着提示走就行 最后yarn run dev 启动项目访问localhost:3000即可 路由组件 app.vue为项目根组件 <nuxt-page />为路由显示入口 将app.vue更改内容如下 <template><d…

WPS中制作甘特图的详细教程

网上没几个详细说怎么在WPS中制作甘特图的&#xff0c;我自己整理了一下详细教程&#xff0c;最终效果如下图所示&#xff1a; 1.写好需要展示的项目相关信息&#xff0c;如下图所示&#xff1a; #####这个进度的百分比渐变效果这样设置就行了 2.现在我们需要计算已用时间和剩…

lodash中flush的使用(debounce、throttle)

在项目的配置中&#xff0c;看到了一个请求&#xff0c;类似是这样的 import { throttle } from lodash-es// 请求函数 async function someFetch(){const {data} await xxx.post()return data }// 节流函数 async function throttleFn(someFetch,1000)// 执行拿到数据函数 a…

Zabbix 配置MySQL数据库监控

Zabbix MySQL数据库监控简介 通过 Zabbix 监控 MySQL 数据库&#xff0c;可以获取有关数据库性能、运行状况和资源使用情况的详细信息&#xff0c;帮助及时发现和解决问题。 Zabbix官方提供了一个名为MySQL by Zabbix agent的监控模板&#xff0c;该模板专为 Zabbix 通过 Zabb…

Java中的文件IO

文件,我们之前在C语言中接触过,是在硬盘上存储数据的方式,操作系统帮我们把硬盘的一些细节都封装起来了,因此在这里我们只需要了解文件的相关接口即可. 首先硬盘是用来存储数据的,和内存相比,硬盘的存储空间更大,访问速度更慢,成本更低,可以实现持久化存储,而操作系统通过&quo…

Polkadot 安全机制揭秘:保障多链生态的互操作性与安全性

作者&#xff1a;Filippo Franchini&#xff0c;Web3 Foundation 原文&#xff1a;https://x.com/filippoweb3/status/1806318265536242146 编译&#xff1a;OneBlock Polkadot 是一个创新的多链区块链平台&#xff0c;旨在实现不同区块链之间的互操作性和共享安全性。本文将详…

c++习题02-浮点数求余

目录 一&#xff0c;问题 二&#xff0c;思路 三&#xff0c;代码 一&#xff0c;问题 二&#xff0c;思路 虽然在浮点类型中没有取余的运算&#xff08;无法直接使用%符号取余&#xff09;&#xff0c;但是我们都知道在数学中&#xff0c;除法是减法的连续运算&#xff…

软件测试最全面试题及答案整理(2024最新版)

1、你的测试职业发展是什么? 测试经验越多&#xff0c;测试能力越高。所以我的职业发展是需要时间积累的&#xff0c;一步步向着高级测试工程师奔去。而且我也有初步的职业规划&#xff0c;前3年积累测试经验&#xff0c;按如何做好测试工程师的要点去要求自己&#xff0c;不断…

师从IEEE fellow|博士后加拿大阿尔伯塔大学成行

V老师指定申请加拿大&#xff0c;优先对方出资的博士后&#xff0c;如果外方无资助&#xff0c;也可以自筹经费&#xff0c;但要求必须是博士后头衔。最终我们为其落实了加拿大阿尔伯塔大学的postdoctoral fellow&#xff08;博士后研究员&#xff09;&#xff0c;尽管是无薪职…

经典链表算法题:找到环的入口。清晰图示推导出来

Leetcode题目链接 原理 重画链表如下所示&#xff0c;线上有若干个节点。记蓝色慢指针为 slow&#xff0c;红色快指针为 fast。初始时 slow 和 fast 均在头节点处。 使 slow 和 fast 同时前进&#xff0c;fast 的速度是 slow 的两倍。当 slow 抵达环的入口处时&#xff0c;如…

前端播放RTSP视频流,使用FLV请求RTSP视频流播放(Vue项目,在Vue中使用插件flv.js请求RTSP视频流播放)

简述&#xff1a;在浏览器中请求 RTSP 视频流并进行播放时&#xff0c;直接使用原生的浏览器 API 是行不通的&#xff0c;因为它们不支持 RTSP 协议。为了解决这个问题&#xff0c;开发者通常会选择使用像 flv.js 这样的库&#xff0c;它专为在浏览器中播放 FLV 和其他流媒体格…

4款引以为豪的办公软件,使用起来,舒适度满满

Everything 是Windows神级搜索软件&#xff0c;能做到秒级响应。 Everything 之前小编在文章里提过好几次&#xff0c;但还有很多小伙伴不知道&#xff0c;那就再给大家种草一下哈。 只需要打开一次&#xff0c;Everything就会自动为你的文件建立索引&#xff0c;之后&#…

Spring MVC 中使用 RESTFul 编程风格

1. Spring MVC 中使用 RESTFul 编程风格 文章目录 1. Spring MVC 中使用 RESTFul 编程风格2. RESTFul 编程风格2.1 RESTFul 是什么2.2 RESTFul风格与传统方式对比 3. Spring MVC 中使用 RESTFul 编程风格(增删改查)的使用3.1 准备工作3.2 RESTFul 风格的 “查询” 所有&#xf…

概率论与数理统计_下_科学出版社

contents 前言第5章 大数定律与中心极限定理独立同分布中心极限定理 第6章 数理统计的基本概念6.1 总体与样本6.2 经验分布与频率直方图6.3 统计量6.4 正态总体抽样分布定理6.4.1 卡方分布、t 分布、F 分布6.4.2 正态总体抽样分布基本定理 第7章 参数估计7.1 点估计7.1.1 矩估计…

视频网关的作用

在数字化时代&#xff0c;视频通信已经成为了人们日常生活和工作中的重要部分。为了满足不同设备和平台之间的视频通信需求&#xff0c;各种视频协议应运而生。然而&#xff0c;这些协议之间的差异使得相互通信变得复杂。因此&#xff0c;视频网关作为一种重要的网络设备&#…

使用TensorRT进行加速推理(示例+代码)

目录 前言 一、TensorRT简介 1.1TensorRT 的主要特点 1.2TensorRT 的工作流程 二、具体示例 2.1代码 2.2代码结构 2.3打印结果 前言 TensorRT 是 NVIDIA 开发的一款高性能深度学习推理引擎&#xff0c;旨在优化神经网络模型并加速其在 NVIDIA GPU 上的推理性能。它支持…

告别写作难题,这些AI写作工具让你文思泉涌

在现实生活中&#xff0c;除了专业的文字工作者&#xff0c;各行各业都避免不了需要写一些东西&#xff0c;比如策划案、论文、公文、讲话稿、总结计划……等等。而随着科技的进步&#xff0c;数字化时代的深入发展&#xff0c;AI已经成为日常工作中必不可少的工具了&#xff0…

Django创建项目(1)

运行 注意 在本次创建Django项目时&#xff0c;出现了一点小问题&#xff0c;由于我之前pip换源过&#xff0c;换源用的是http&#xff0c;结果在创建时&#xff0c;pip只支持https&#xff0c;所以如果出现创建项目失败的问题&#xff0c;那么有可能是因为换源的问题&#xf…