草庐IT

【Unity3D小技巧】Unity3D中判断Animation以及Animator动画播放结束,以及动画播放结束之后执行函数

恬静的小魔龙 2023-04-18 原文

推荐阅读

大家好,我是佛系工程师☆恬静的小魔龙☆,不定时更新Unity开发技巧,觉得有用记得一键三连哦。

一、前言

在日常开发中,可能会遇到要判断Animation或者Animator动画播放是否结束的情况。

判断Animation是否结束可以这么写:

using System;
using System.Collections;
using UnityEngine;

public class AnimationPlayControl : MonoBehaviour
{
    Animation ani;
    void Update()
    {
        AnimationState state = ani["Open"];
        if (state.normalizedTime >= 1) 
        {
            // 判断动画播放结束normalizedTime的值为0~1,0为开始,1为结束。
            Debug.Log("动画播放完毕");
        }
    }
}

或者简单一点:

using System;
using System.Collections;
using UnityEngine;

public class AnimationPlayControl : MonoBehaviour
{
    Animation ani;
	void Start()
	{
		if(!ani.isPlaying)
		{
			Debug.Log("没有播放动画,或播放结束动画");
		}
	}
}

判断Animator是否结束可以这么写:

using System;
using System.Collections;
using UnityEngine;

public class AnimationPlayControl : MonoBehaviour
{
    Animator anir;
    void Start()
    {
        //获取动画层 0 指Base Layer.
        AnimatorStateInfo stateinfo = anir.GetCurrentAnimatorStateInfo(0);
        //判断是否正在播放walk动画.
        if (!stateinfo.IsName("Base Layer.walk"))
        {
        	Debug.Log("没有播放walk动画,或播放结束动画");
        }
    }
}

但是,如果说要判断动画结束之后再执行某些事件就有些麻烦,首先需要在Update里面一直判断是否播放完,播放完再去执行事件。

消耗性能,也不好管理,所以就写了一个迭代器也就是协程来判断动画是否结束,结束后执行回调函数。

二、Animation动画播放结束判断

2-1、第一种协程写法

代码参考:

using System;
using System.Collections;
using UnityEngine;

public class AnimationPlayControl : MonoBehaviour
{
    Animation ani;
    void Start()
    {
        PlayAnimation(ani, "Open",
        () =>
        {
            Debug.Log("动画播放前执行代码");
        },
        () =>
        {
            Debug.Log("动画播放完执行代码");
        });
    }

    // 封装函数
    public void PlayAnimation(Animation animation, string clipName, Action startAct = null, Action endAct = null)
    {
        StartCoroutine(PlayAnimationItor(animation, clipName, startAct, endAct));
    }

    /// <summary>
    /// Animation动画播放迭代器
    /// </summary>
    /// <param name="animation">Animation组件</param>
    /// <param name="clipName">clip片段名</param>
    /// <param name="startAct">委托函数</param>
    /// <param name="endAct">委托函数</param>
    /// <returns></returns>
    IEnumerator PlayAnimationItor(Animation animation, string clipName, Action startAct, Action endAct)
    {
        startAct?.Invoke();

        animation.Play(clipName);
        yield return new WaitForSeconds(animation[clipName].length);

        endAct?.Invoke();
    }
}

PS:主要是使用了协程中的new WaitForSeconds也就是等待X秒,这里就是等待动画播放的X秒后执行委托函数,更加详细的协程可以参考我其他文章,这里就不详细说明了。

2-2、第二种协程写法

using System;
using System.Collections;
using UnityEngine;

public class AnimationPlayControl : MonoBehaviour
{
    Animation ani;
    void Start()
    {
        PlayAnimation(ani, "Open",
        () =>
        {
            Debug.Log("动画播放前执行代码");
        },
        () =>
        {
            Debug.Log("动画播放完执行代码");
        });
    }

    // 封装函数
    public void PlayAnimation(Animation animation, string clipName, Action startAct = null, Action endAct = null)
    {
        StartCoroutine(PlayAnimationItor(animation, clipName, startAct, endAct));
    }

    /// <summary>
    /// Animation动画播放迭代器
    /// </summary>
    /// <param name="animation">Animation组件</param>
    /// <param name="clipName">clip片段名</param>
    /// <param name="startAct">委托函数</param>
    /// <param name="endAct">委托函数</param>
    /// <returns></returns>
    private IEnumerator PlayAnimationItor(Animation animation, string clipName, Action startAct, Action endAct)
    {
        startAct?.Invoke();

        AnimationState animationState = animation[clipName];
        animation.Play(clipName);
        yield return StartCoroutine(new WaitForEndOfAnim(animationState));

        endAct?.Invoke();
    }
}

// 实现WaitForEndOfAnim迭代器
public class WaitForEndOfAnim : IEnumerator
{
    AnimationState m_animState;
    public WaitForEndOfAnim(AnimationState animState)
    {
        m_animState = animState;
    }
    public object Current
    {
        get
        {
            return null;
        }
    }
    public bool MoveNext()
    {
        return m_animState.enabled;
    }
    public void Reset()
    {
    }
}

这种方式,实现WaitForEndOfAnim迭代器,在动画播放完后执行后面的代码。

三、Animator动画播放结束判断

首先,要明白Animator是动画状态机,是用来控制动画片段,然后去切换动画片段的,如下图所示:


所以说,Animator默认就会开始播放一种动画,不论是Idle动画还是Walk动画,总会播放一种动画,所以就不用像Animation去Play某一个动画了,只需要判断当前动画是否播放完即可。

代码参考如下,两种方法写一起了:

using System;
using System.Collections;
using UnityEngine;

public class AnimationPlayControl : MonoBehaviour
{
    Animator ani;
    void Start()
    {
        PlayAnimation(ani, "Open",
        () =>
        {
            Debug.Log("动画播放前执行代码");
        },
        () =>
        {
            Debug.Log("动画播放完执行代码");
        });
    }

    // 封装函数
    public void PlayAnimator(Animator animator, string clipName, Action startAct = null, Action endAct = null)
    {
        StartCoroutine(PlayAnimationItor(animator, clipName, startAct, endAct));
    }

    /// <summary>
    /// Animation动画播放迭代器
    /// </summary>
    /// <param name="animation">Animation组件</param>
    /// <param name="clipName">clip片段名</param>
    /// <param name="startAct">委托函数</param>
    /// <param name="endAct">委托函数</param>
    /// <returns></returns>
    private IEnumerator PlayAnimationItor(Animator animator, string clipName, Action startAct, Action endAct)
    {
        startAct?.Invoke();

        AnimatorStateInfo animatorStateInfo = animator.GetCurrentAnimatorStateInfo(0);
        yield return StartCoroutine(new WaitForEndOfAnimr(animatorStateInfo,clipName));

        endAct?.Invoke();
    }

    /// <summary>
    /// Animation动画播放迭代器
    /// </summary>
    /// <param name="animation">Animation组件</param>
    /// <param name="clipName">clip片段名</param>
    /// <param name="startAct">委托函数</param>
    /// <param name="endAct">委托函数</param>
    /// <returns></returns>
    IEnumerator PlayAnimationItor2(Animator animator, string clipName, Action startAct, Action endAct)
    {
        startAct?.Invoke();
        yield return new WaitForSeconds(animator.GetCurrentAnimatorClipInfo(0)[0].clip.length);
        endAct?.Invoke();
    }
}

// 实现WaitForEndOfAnim迭代器
public class WaitForEndOfAnimr : IEnumerator
{
    AnimatorStateInfo m_animState;
    public WaitForEndOfAnimr(AnimatorStateInfo animState,string clipName)
    {
        m_animState = animState;
    }
    public object Current
    {
        get
        {
            return null;
        }
    }
    public bool MoveNext()
    {
        return m_animState.IsName(clipName);
    }
    public void Reset()
    {
    }
}

四、后记

结束,本篇文章讲解了Animation和Animator动画播放结束的判断代码。

以及如何实现在Animation和Animator动画播放结束判断,并且执行回调函数的实现。


你的点赞就是对博主的支持,有问题记得留言:

博主主页有联系方式。

博主还有跟多宝藏文章等待你的发掘哦:

专栏方向简介
Unity3D开发小游戏小游戏开发教程分享一些使用Unity3D引擎开发的小游戏,分享一些制作小游戏的教程。
Unity3D从入门到进阶入门从自学Unity中获取灵感,总结从零开始学习Unity的路线,有C#和Unity的知识。
Unity3D之UGUIUGUIUnity的UI系统UGUI全解析,从UGUI的基础控件开始讲起,然后将UGUI的原理,UGUI的使用全面教学。
Unity3D之读取数据文件读取使用Unity3D读取txt文档、json文档、xml文档、csv文档、Excel文档。
Unity3D之数据集合数据集合数组集合:数组、List、字典、堆栈、链表等数据集合知识分享。
Unity3D之VR/AR(虚拟仿真)开发虚拟仿真总结博主工作常见的虚拟仿真需求进行案例讲解。
Unity3D之插件插件主要分享在Unity开发中用到的一些插件使用方法,插件介绍等
Unity3D之日常开发日常记录主要是博主日常开发中用到的,用到的方法技巧,开发思路,代码分享等
Unity3D之日常BUG日常记录记录在使用Unity3D编辑器开发项目过程中,遇到的BUG和坑,让后来人可以有些参考。

有关【Unity3D小技巧】Unity3D中判断Animation以及Animator动画播放结束,以及动画播放结束之后执行函数的更多相关文章

  1. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  2. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

  3. 【鸿蒙应用开发系列】- 获取系统设备信息以及版本API兼容调用方式 - 2

    在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList​()Obt

  4. Unity 热更新技术 | (三) Lua语言基本介绍及下载安装 - 2

    ?博客主页:https://xiaoy.blog.csdn.net?本文由呆呆敲代码的小Y原创,首发于CSDN??学习专栏推荐:Unity系统学习专栏?游戏制作专栏推荐:游戏制作?Unity实战100例专栏推荐:Unity实战100例教程?欢迎点赞?收藏⭐留言?如有错误敬请指正!?未来很长,值得我们全力奔赴更美好的生活✨------------------❤️分割线❤️-------------------------

  5. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

  6. unity---接入Admob - 2

    目录1.AdmobSDK下载地址2.将下载好的unityPackagesdk导入到unity里​编辑 3.解析依赖到项目中

  7. Unity 3D 制作开关门动画,旋转门制作,推拉门制作,门把手动画制作 - 2

    Unity自动旋转动画1.开门需要门把手先动,门再动2.关门需要门先动,门把手再动3.中途播放过程中不可以再次进行操作觉得太复杂?查看我的文章开关门简易进阶版效果:如果这个门可以直接打开的话,就不需要放置"门把手"如果门把手还有钥匙需要旋转,那就可以把钥匙放在门把手的"门把手",理论上是可以无限套娃的可调整参数有:角度,反向,轴向,速度运行时点击Test进行测试自己写的代码比较垃圾,命名与结构比较拉,高手轻点喷,新手有类似的需求可以拿去做参考上代码usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;u

  8. [Vuforia]二.3D物体识别 - 2

    之前说过10之后的版本没有3dScan了,所以还是9.8的版本或者之前更早的版本。 3d物体扫描需要先下载扫描的APK进行扫面。首先要在手机上装一个扫描程序,扫描现实中的三维物体,然后上传高通官网,在下载成UnityPackage类型让Unity能够使用这个扫描程序可以从高通官网上进行下载,是一个安卓程序。点到Tools往下滑,找到VuforiaObjectScanner下载后解压数据线连接手机,将apk文件拷入手机安装然后刚才解压文件中的Media文件夹打开,两个PDF图打印第一张A4-ObjectScanningTarget.pdf,主要是用来辅助扫描的。好了,接下来就是扫描三维物体。将瓶

  9. 阿里云国际版免费试用:如何注册以及注意事项 - 2

    作为新的阿里云用户,您可以50免费试用多种优惠,价值高达1,700美元(或8,500美元)。这将让您了解和体验阿里云平台上提供的一系列产品和服务。如果您以个人身份注册免费试用,您将获得价值1,700美元的优惠。但是,如果您是注册公司,您可以选择企业免费试用,提交基本信息通过企业实名注册验证,即可开始价值$8,500的免费试用!本教程介绍了如何设置您的帐户并使用您的免费试用版。​关于免费试用在我们开始此试用之前,您还必须遵守以下条款和条件才能访问您的免费试用:只有在一年内创建的账户才有资格获得阿里云免费试用。通过此免费试用优惠,用户可以免费试用免费试用活动页面上列出的每种产品一次。如果您有多个帐

  10. ruby - 如果满足给定条件,则结束 ruby​​ 程序 - 2

    基本上,我只是试图在满足特定条件时停止程序运行其余行。unlessraw_information.firstputs"Noresultswerereturnedforthatquery"breakend然而,在程序运行之前我得到了这个错误:Invalidbreakcompileerror(SyntaxError)执行此操作的正确方法是什么? 最佳答案 abort("Noresultswerereturnedforthatquery")unlesscondition或unlessconditionabort("Noresultswer

随机推荐