UnityVR--组件7--动画事件BlendTree

news2025/1/16 20:08:48

目录

应用1:使用BlendTree实现站立和移动

应用2:人物跳跃事件&播放跳跃动画

应用3:开火动画事件&动画片段中建立事件监听


  上一篇(组件5--Animation动画)已经做了2个动画片段,HeroIdle和HeroJump,另外实现了简单的动画转场控制。本篇使用BlendTree来控制动画转场,并使用事件发送的方式来控制动画事件。

应用1:使用BlendTree实现站立和移动

   BlendTree用于对于两个或以上相似的动画混合(而不是过渡),比如人物在走路到达一定速度时与跑步的动画混合,或者在走动时同时向某个角度倾斜。

  下面使用BlendTree实现一下组件5--Animation动画中的站立和移动:

  1. 建立BlendTree:在状态机中点右键->CreateState->FromNewBlendTree

  

   2. 添加动画:在BlendTree上点右键->AddMotion,就会在右边的Inspector中添加一个Motion,添加两个Motion后,就会出现两个动画重叠转换的权重。分别将两个动画片段拖入下面的Motion名字框。下面这个Automate不要勾选,就可以自己配置权重。同时在Animator中也会看到这些动画片段:

  

  3. 脚本中添加动画事件:对于事件中心的修改,需要增加动画的事件类型、事件传输数据等。(关于事件中心,详见事件中心1等3篇)

  (1)添加事件类型——在EventType.c中,在枚举数据中添加下面这个播放动画的代码:

    OnPlayAnime=1005

  (2)添加事件传输数据——在EventDataBase.cs中,添加发送动画事件时同时需要发送的参数:

public class EventDataAnime:EventDataBase
{
    public EAnimeType animeType; 
    public float speed; //速度判断:站、走、跑
}

  (3)添加播放动画类型——建立一个枚举类型(也可以直接写在EventDataBase.cs里面),添加要播放的几种动画:

public enum EAnimeType
{
    Walk,Run,Jump,Idle,Fire
}

  4. 添加动画播放事件。在挂载Animator组件的节点上(我这里直接挂在主角Hero身上),新建一个AnimePlay.cs脚本:

public class AnimePlay : MonoBehaviour
{//使用事件控制动画
    private Animator animator;
    void Start()
    {
        animator = GetComponent<Animator>(); //获取同一节点上的Animator组件
        EventManager.Instance.AddEvent(EventType.OnPlayAnime, this, PlayAnime);
        //注册一个动画播放的事件,监听是此脚本
    }
    void PlayAnime(EventDataBase data)
    {
        var eventData = data as EventDataAnime;
        //根据传输的动画类型判断
        switch (eventData.animeType)
        {
            case EAnimeType.Idle:
                animator.SetFloat("changeFloat", 0);break;  
            //如果播放类型设置为站立Idle时,把转场参数设置为0,就播放站立动画
            case EAnimeType.Walk:
                animator.SetFloat("changeFloat", eventData.speed);break;
            //在移动时达到一定的速度就播放走路动画(其实是跳,因为没有做走路的动画效果)
        }
    }
}

  5. 修改主角行走脚本(移动事件的建立,详见新版InputSystem和EventManager),主角行走的脚本HeroMoveEvent.cs中的FixedUpdate()修改如下:

    void FixedUpdate()
    {
        movement.Set(moveInput.x, 0, moveInput.y);//moveInput的数据通过事件数据发送
        movement.Normalize();
        //检测是否有输入
        bool hInput = !Mathf.Approximately(moveInput.x, 0);
        bool vInput = !Mathf.Approximately(moveInput.y, 0);
        
        //增加:对于行走速度的判断,以便播放相应动画
        //定义一个isRunning的变量,用于判断上一帧是否有水平、垂直方向的输入

        if (isRunning&&!(hInput||vInput))
        {//判断如果上一帧是正在移动,而这一帧没有收到移动输入
            EventManager.Instance.SendEvent(EventType.OnPlayAnime, new EventDataAnime
            {//发送速度为0的事件,并播放站立动画
                animeType = EAnimeType.Idle,
                speed = 0
            });
        }

        isRunning = hInput || vInput;  //这一帧的isRunning赋值
        if (isRunning)
        {
            movement = Quaternion.Euler(0, Camera.main.transform.eulerAngles.y, 0) * movement;
            //增加发送走路动画事件
            EventManager.Instance.SendEvent(EventType.OnPlayAnime, new EventDataAnime
            {
                animeType = EAnimeType.Walk,
                speed = movement.sqrMagnitude  //Vector3的API,把数值变为Float值
            }) ;
        }
        Vector3 lookForward = Vector3.RotateTowards(transform.forward, movement, rotateSpeed * Time.fixedDeltaTime, 360);
        targetRotation = Quaternion.LookRotation(lookForward);
        rb.MovePosition(rb.position + movement * moveSpeed * Time.fixedDeltaTime);
        rb.MoveRotation(targetRotation);
    }

  6. 场景中的设置与上一篇 (组件5--Animation动画)一样,需要设置一个float参数作为转场的条件,也就是行走速度,初始值为0,当它达到0以上,也就是有速度时,就播放行走动画:

 最终效果,不动的时候播放站立动画,一呼一吸地,一旦发生移动就跳起来了:

应用2:人物跳跃事件&播放跳跃动画

  1. 按照上面的例子,先修改事件中心:

  (1)添加事件类型——在枚举EventType中加入:

OnPlayerJump=1006

  (2)在上面例子的(2)、(3)中已经添加了枚举的动画类型EAnimeType(里面已经写了Jump)和动画数据EventDataAnime,这里就不需要再修改了;

  2. 定义跳跃使用的按键:在Hero身上任何一个脚本的Update()中,直接接收一个控制跳跃的按键(当然也可以在新版InputSystem中定义,并设置监听跳跃事件,详见新版InputSystem),我这里统一放到InputManager.cs中了

        if (Input.GetKeyDown(KeyCode.Space))
        {//发送跳跃事件,在主角移动HeroMoveEvent中监听
            EventManager.Instance.SendEvent(EventType.OnPlayerJump, null);
        }

 3. 实现跳跃:挂在Hero身上

public class HeroMoveEvent : MonoBehaviour
{
    //定义一个跳跃力
    public float jumpGravity = -19.8f;  
    public float addJumpForce = 8;

    void Start()
    {
        rb = GetComponent<Rigidbody>();
        //注册一个主角跳跃事件,回调JumpAnime函数
        EventManager.Instance.AddEvent(EventType.OnPlayerJump, this, JumpAnime);
    }

    void JumpAnime(EventDataBase data)   
    {
        Physics.gravity = new Vector3(Physics.gravity.x, jumpGravity, Physics.gravity.z);
        rb.velocity = new Vector3(rb.velocity.x, addJumpForce / rb.mass, rb.velocity.y);
        //向上的力除以质量

        //跳的时候同时发送一个播放跳跃动画的事件
        //统一在hero身上的AnimePlayer接收
        EventManager.Instance.SendEvent(EventType.OnPlayAnime, new EventDataAnime { animeType = EAnimeType.Jump });
    }

  4. 上面这个例子中写的AnimePlayer.cs中已经注册了动画播放事件,并且定义了回调函数PlayAnime(),我们只要在回调函数的Switch中再加一个跳跃动画的判断就行

        switch (eventData.animeType)
        {
            …………省略…………
            case EAnimeType.Jump:
                animator.SetTrigger("JumpTrigger");break;
            //跳跃只需要Trigger控制,按一次跳一次
        }

  5. 对于场景中的Animator的设置,这里就不使用BlendTree了,简单的使用转场箭头连接一下:

   6. 这样就实现成功了,在运行时可以对于重力和跳跃力的参数进行调整,调到项目需要的状态

应用3:开火动画事件&动画片段中建立事件监听

  1. 建立一个动画片段Animation(HeroFire),作为开火时播放的动画。制作粗陋,就不展示了

  2. 在Animator中如下设置:将HeroFire动画片段放入,并建立与默认动画“BlendTree”的转场箭头,最后在左边建立一个Trigger类型的参数,取名为FireTrigger:

   3. 开火使用的按键在InputSystem中定义过,并且在InputManager.cs中有开火事件(OnFire())的定义,详见新版InputSystem,不再赘述。这里需要修改一下注册开火事件的脚本:

public class BulletFireEvent : MonoBehaviour
{
    public GameObject bullet;
    public bool canFire=true; //判断是否可以进行本次攻击
    void Start()
    {
        EventManager.Instance.AddEvent(EventType.OnPlayAnime, this, data =>
        {
            if ((data as EventDataAnime).animeType == EAnimeType.FireEnd) canFire = true;
        });
        EventManager.Instance.AddEvent(EventType.OnPlayerFire, this, callback =>
        {
            if (canFire == false) return;
            //判断上一次动画播放是否完成,接到canFire==true时才能往下操作
            canFire = false;
            EventManager.Instance.SendEvent(EventType.OnPlayAnime, new EventDataAnime
            {
                animeType = EAnimeType.Fire
            });
            //实例化子弹
            bullet = Resload.Instance.LoadPrefab("Bullet");
            bullet.SetActive(true);//激活
            bullet.transform.position = transform.position;
            bullet.transform.rotation = transform.rotation;
            Destroy(bullet, 2);//子弹2秒后自毁
        });
    }
    void OnFireEnd()
    {
            EventManager.Instance.SendEvent(EventType.OnPlayAnime, new EventDataAnime
            {
                animeType = EAnimeType.FireEnd
            });
    }
}

  这里需要说明的是:

  (1)增加了一个布尔变量-canFire,判断是否能够播放开火。这是因为必须等上一次的开火动画播放完成再播放;

  (2)注册了2个事件:OnPlayAnime和OnPlayerFire,一个用于开火,一个用于播放动画;

  (3)在上面应用1中建立的动画类型的枚举数据(EAnimeType)中再增加一个FireEnd的数据,用于传输动画是否已经播完的信号

public enum EAnimeType
{
    Walk, Run, Jump, Idle, Fire, FireEnd
}

  (4)最后的这个OnFireEnd()函数,是需要到开火动画去监听的,设置方法看步骤4.

  4. 打开HeroFire的动画片段,在结束处加入一个动画事件:

   5. 点击这个动画事件,在Inspector面板中将回调函数设置为OnFireEnd

    6.完整的运行结果:开火时发一个子弹,并且小鬼的身子转一下。可以看到在开火动画播放完成后,布尔参数canFire就变成true允许下一次动画的执行。

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

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

相关文章

【C语言】qsort详细将解

系列文章目录 qsort目录 系列文章目录一、前言二、qosort是什么&#xff1f;二、qsort的使用1、原型2、参数3、头文件&#xff08;1&#xff09;qsort参数中的函数指针讲解 三、使用示例和运行截图1、整形例子&#xff08;升序&#xff09;3、字符例子&#xff08;降序&#xf…

Android——使用Service服务实现通信

实验目的&#xff1a; &#xff08;1&#xff09;能创建、启动和关闭服务 &#xff08;2&#xff09;能实现服务的通信 实验内容及原理&#xff1a; 设计一个服务的具体应用&#xff0c;实现服务的通信 实验设备及实验步骤&#xff1a; 实验设备&#xff1a;WindowsAndro…

VPS 和GPS 、SLAM 之间的爱恨情仇

注&#xff1a;该文章首发3D视觉工坊&#xff0c;链接如下3D视觉工坊 VPS 、GPS 、SLAM 的区别与联系 首先简单的阐述一下三者的定义&#xff1a; VPS全称为Visual Positioning System&#xff0c;即视觉定位系统。手机端(移动时代&#xff09;的VPS首次出现时间节点为2019年&…

Linux 负载均衡集群 LVS_NAT模式 LVS_DR模式

集群 由多台主机组成&#xff0c;只做一件事&#xff0c;对外表现为一个整体。 只干一件事 &#xff1a;集群 干不同的事&#xff1a;分布式 企业集群分类 负载均衡群集&#xff08;load balance cluster&#xff09; 提高系统响应效率&#xff0c;处理更多的访问请…

Qt6 C++基础入门3 对话框与MainWindow

目录 对话框MainWindow菜单工具栏 对话框 目前的对话框主要有以下几大类 文件对话框( QFile Dialog)消息对话框( QMessageBox)输入对话框( QInputDialog)颜色对话框( QColorDialog)字体对话框( QFontDialog) 这是七大对话框及其基本用法的实例参考&#xff0c;所有代码都写在…

《星岛日报》专访:欧科云链AML,助力数字资产合规及风险防控

6月1日&#xff0c;香港《适用于虚拟资产交易平台营运者的指引》及《打击洗钱指引》正式施行&#xff0c;香港虚拟资产发牌制度正式生效。作为深耕香港市场多年的Web3科技企业&#xff0c;欧科云链OKLink也正式推出的Onchain AML反洗钱合规解决方案&#xff0c;利用多年积累的海…

Windows下安装python和pip

Windows下安装python和pip 1、安装python 注意&#xff1a;windows10 安装时强烈建议不用使用 Windows Store 安装。避免后期python运行时牵扯权限相关问题。 具体步骤&#xff1a; 1、前往python官网下载windows python 安装包 下载文件 2、双击运行安装&#xff08;强力…

实时日志滚动显示 springboot+vue3

-:后端使用ssemiter保持客户端链接:http 这里不用websocket的原因是,sse很轻,整合方便,可发送日志,消息,群发等都可以。 -:前端使用vue3+ansi_up做页面展示 第一: 刷新页面导致session问题 可以在java的session中记录,如果是同一个客户重新链接的话,直接返回java…

【轴承故障诊断】用于轴承故障诊断的集中时频分析研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

SQL SERVER case when的使用方法

一、case when的使用方法 Case具有两种格式。简单Case函数和Case搜索函数。 第一种 格式 : 简单Case函数 : 格式说明     case 列名     when 条件值1 then 选项1     when 条件值2 then 选项2…     else 默认值 end eg:     select     case   job…

2021年国赛高教杯数学建模B题乙醇偶合制备C4烯烃解题全过程文档及程序

2021年国赛高教杯数学建模 B题 乙醇偶合制备C4烯烃 原题再现 C4 烯烃广泛应用于化工产品及医药的生产&#xff0c;乙醇是生产制备 C4 烯烃的原料。在制备过程中&#xff0c;催化剂组合&#xff08;即&#xff1a;Co 负载量、Co/SiO2 和 HAP 装料比、乙醇浓度的组合&#xff0…

JUC源码分析:通过ReentrantLock阅读AbstractQueuedSynchronizer源码

一、概述 ReentrantLock进行上锁的流程如下图所示&#xff0c;我们将按照下面的流程分析ReentrantLock上锁的流程&#xff0c;在此过程中阅读AbstractQueuedSynchronizer源码。 AQS 的数据结构如下图所示。 AQS大家还记得吗&#xff1f;最核心的是它的一个共享的int类型值叫做…

电脑自动关机是什么原因?如何解决?

案例&#xff1a;有时候我的电脑用着就突然关机&#xff0c;会导致一些没有保存的文件丢失。有没有小伙伴知道电脑为什么会自动关机&#xff1f;怎样做才能避免这个问题&#xff1f; 在使用电脑过程中&#xff0c;遇到电脑自动关机的问题是很常见的。当我们在进行重要任务时&a…

Netty核心源码剖析(四)

1.Netty心跳(heartbeat)服务源码剖析 1>.Netty作为一个网络框架,提供了诸多功能,比如编码解码等,Netty还提供了非常重要的一个服务–心跳机制heartbeat.通过心跳检查对方是否有效,这是RPC框架中是必不可少的功能.下面我们分析一下Netty内部心跳服务源码实现; 2>.Netty提…

电磁仿真需要牢记的内功心法

在射频、微波设计中&#xff0c;各种“强大”的商用电磁仿真软件的功能包罗万象&#xff0c;这篇“内功心法”从算法角度出发&#xff0c;提示大家如何谨慎选择仿真软件。 心法一&#xff1a;场”与“路”的区分 世上本无“路”&#xff0c;“场”近似得多了就变成了“路”&a…

千人规模亚马逊云科技出海日将于6月9日开启,助推企业出海出圈

向全球价值链上游奋进 中国企业增强国际竞争力的关键&#xff0c;是努力朝全球价值链上游奋进&#xff0c;发力技术出海。中国的出海新机遇&#xff0c;背后曾是疫情在全球按下数字互联和数字化升级的快进键&#xff0c;跨境电商、在线社交、移动支付、数字服务等数字经济迎来…

什么是 Vue.js 中的 keep-alive 组件?如何使用 keep-alive 组件?

Vue.js 中的 Keep-alive 组件 Vue.js 是一款流行的前端框架&#xff0c;它提供了许多实用的组件和工具&#xff0c;其中之一就是 Keep-alive 组件。Keep-alive 组件是 Vue.js 的一个高阶组件&#xff0c;它可以帮助我们缓存组件实例&#xff0c;提高应用程序的性能和响应速度。…

python3写一个http接口服务(get, post),给别人调用6

python3写一个http接口服务(get, post)&#xff0c;给别人调用6 一、python3写一个http接口服务(get, post)&#xff0c;给别人调用6 近年来异步web服务器比较火热&#xff0c;例如falcon/bottle/sanic/aiohttp&#xff0c;今天也来玩玩sanic。 Sanic是一个支持Python 3.7的w…

Vue.js 中的 v-for 中的 key 属性

Vue.js 中的 v-for 中的 key 属性 Vue.js 是一个流行的 JavaScript 前端框架&#xff0c;它提供了一种简单的方式来构建可复用的组件和应用程序。在 Vue.js 中&#xff0c;v-for 指令用于循环渲染一个数组或对象&#xff0c;并将每个元素渲染为一个 DOM 元素。在使用 v-for 指…

数据安全架构设计

在提到安全架构之前&#xff0c;我们先看看安全的定义&#xff1a;安全是产品的质量属性&#xff0c;安全的目标是保障产品里信息资产的保密性&#xff08;Confidentiality&#xff09;、完整性&#xff08;Integrity&#xff09;和可用性&#xff08;Availability&#xff09;…