在Unity中实现简单的动画转场

  • 1 思路
    • 1.1 原教程的思路
    • 1.2 我的思路
  • 2 制作转场动画
  • 3 代码
    • 3.1 Transition
    • 3.2 SceneLoader
    • 3.3 测试
  • 4 成品

看了这个视频(油管)之后的想法。思路就是按着教程的思路。但是原教程是直接加载场景的(让我感到很迷惑),我打算把这个稍微改改,弄成异步加载场景的。

1 思路

1.1 原教程的思路

做一个两个UI的过渡动画,分别是转场开始和转场结束。然后加载场景后播放转场结束动画,加载场景前先播放转场开始动画,然后加载场景。

1.2 我的思路

首先我觉得像原教程这种思路,在加载时间长的时候很难把控动画的结束时间。
然后我的思路是这样的。调用转场的方法后,开始异步加载下一个场景,同时播放开始转场的动画,然后一直检查动画是否播放完和场景是否加载完。只要有一个还没完成的话就还是停留在开始转场的结束位置。等确认场景也加载好了,转场开始的动画也加载好了之后再播放转场结束的动画。
此外我还打算把这个转场控制做成一个转场不会销毁的控件,弄成单例方便调用,并且在转场的时候可以动态的指定预备好的转场样式 (但是并没有做)。

2 制作转场动画

简单用UI做了两个动画,就不细说了。关键就是两个动画之间的那两帧起码是要遮住整个屏幕的,不过因为不同分辨率,其实还需要解决的一个就是适配的问题。这里主要就是调整Canvas上的canvas scaler组件,不过我也没有太深入了解。


除了这两个动画之外我还做了一个idle动画,就是什么都没有动的。还有比较重要的是 动画都要设置成不循环(× Loop time)。
然后就是弄一个Animator来简单的管理以下,大概就是这样。

注意在转换里面,Transition Duration(s) 要设置成0,不然会把你两个UI动画混合的非常奇怪。
我顺便把可转换到自身也给关了(之前好像开着会导致某些问题,但是现在再试下又找不到这个问题了,可能不是这个导致的。不过反正我们也不需要这个功能)。

3 代码

3.1 Transition

每个转场都要带上这个脚本,主要就是用来播放动画以及检查动画是否播放完成。其实也没有多少内容,可以不用特地像这样写一个类。我主要是为了在选择不同转场的时候不是指定GameObject而是指定一个Transition,看起来舒服些。也避免了拖错GameObject,或许以后还可以给Transition添加更多功能,留个后路。

using UnityEngine;[RequireComponent(typeof(Animator))]
public class Transition : MonoBehaviour
{[SerializeField]private Animator animator;private void Awake() {animator = GetComponent<Animator>();}/// <summary>/// 播放转场前的动画/// </summary>public void StartTrans(){animator.SetTrigger("Start");}/// <summary>/// 播放转场后的动画/// </summary>public void EndTrans(){animator.SetTrigger("End");}/// <summary>/// 当前动画是否播放完成/// </summary>/// <returns></returns>public bool IsAnimationDone(){if(animator.GetCurrentAnimatorStateInfo(0).normalizedTime >= 1)return true;else return false;}
}

3.2 SceneLoader

简单做了个单例,方便调用转场。

  • LoadScene,用于在外部调用转场函数。分别传入场景的名称和转场的类型。(也没有做校验场景是否存在,本来也想着是调用的时候还可以指定转场类型。但是比较麻烦的是这个转场必须实例化出来,而且还需要转换场景不销毁。后来懒得做就算了,只是留了个参数。)
  • LoadLevel,实际用来加载场景的协程。主要的逻辑在这里,注释也写了挺多了。首先因为我是播放动画的时候就开始加载场景,为了放置动画没播放完就转换了场景。我是把loading.allowSceneActivation设成了false。这个为false会让加载进度停止在0.9,为true之后继续加载完剩下的。本来是用while(loading.progress < 0.9)判断的。但是实际上会一直卡在这里,所以改成了0.99。然后检查动画播放完成之后让场景继续加载到1再播放下一个动画,因为测试的过程中发现如果不这样的话,可能因为0.9到1也需要一点点时间,有时候会有一帧左右还能看到上一个场景。然后动画播放完之后结束转场整个转场过程中不允许再次执行转场,以免画面错乱。
using System.Collections;
using UnityEngine;
using UnityEngine.SceneManagement;public class SceneLoader : MonoBehaviour
{public static SceneLoader _instance; // 实例public Transition currentTransition; // 当前的过场预设public bool LoadAble{ get; private set;} // 现在是否可以转场private void Awake() {if(_instance)Destroy(gameObject);else{_instance = this;DontDestroyOnLoad(gameObject);}LoadAble = true;}/// <summary>/// 使用转场加载场景/// </summary>/// <param name="SceneName">场景名称</param>/// <param name="transition">转场方式</param>public void LoadScene(string SceneName, Transition transition = null){// 如果现在不能转换场景直接返回if(!LoadAble) return;// 如果有设置过场就设置过场if (transition != null)currentTransition = transition;// 开始转换场景StartCoroutine(LoadLevel(SceneName));}/// <summary>/// 转换场景并且使用过场动画/// </summary>/// <param name="levelName">场景名称</param>/// <returns></returns>private IEnumerator LoadLevel(string levelName){// 异步加载场景AsyncOperation loading = SceneManager.LoadSceneAsync(levelName);// 不允许场景加载完后直接转换loading.allowSceneActivation = false;// 现在不再能转换场景LoadAble = false;// 开始过场currentTransition.StartTrans();// 等待一帧// 理由再下面有解释,但其实这里本来不需要,因为检查动画前还夹着一个检查加载的过程。基本不会在一帧内就加载完// 但是保险起见还是在播放动画后延迟一帧yield return null;// 等待场景加载几乎完成while(loading.progress < 0.899)yield return null;// 等待动画播放完成while(!currentTransition.IsAnimationDone())yield return null;// 允许场景加载完成loading.allowSceneActivation = true;// 等待场景加载彻底完成while(loading.progress != 1)yield return null;// 结束过场currentTransition.EndTrans();// 等待一帧// 因为我发现如果在开始动画后不等待一帧的话,第二个动画其实还没开始播放,// 后面检测动画完成检测的就是第一个动画,就起不到检测第二个动画的作用。yield return null;// 等待动画播放完成while(!currentTransition.IsAnimationDone())yield return null;// 可以继续转换场景LoadAble = true;}
}

3.3 测试

简单写个脚本测试一下。

using UnityEngine;public class TestScript : MonoBehaviour
{void Update(){if(Input.GetKeyDown(KeyCode.Alpha1)){SceneLoader._instance.LoadScene("Scene A");}if(Input.GetKeyDown(KeyCode.Alpha2)){SceneLoader._instance.LoadScene("Scene B");}if(Input.GetKeyDown(KeyCode.Alpha3)){SceneLoader._instance.LoadScene("Scene C");}if(Input.GetKeyDown(KeyCode.Alpha4)){SceneLoader._instance.LoadScene("Scene D");}}
}

4 成品

另外去商店下载了个场景加载测试,的确会在加载完场景前停在动画里面,完成后再播放剩下的动画。但是笔记本录屏之后特别卡,根本看不了。然后就改成用台式机录,是不卡了,但是场景也瞬间加载好了。就这样吧…
源码:github
视频:

视频

在Unity中实现简单的动画转场相关推荐

  1. 初探CardBoard:(1)在Unity中实现简单VR场景

    初探CardBoard:(1)在Unity中实现简单VR场景 为何使用CardBoard 价格低廉 兼容性 前期准备 开始工程 一.简单的环境布置 二.导入SDK 三.SDK基础组件说明 四.将之前的 ...

  2. Cinemachine教程 | Unity中如何制作路径动画?

    摘要:在Unity中制作路径动画是有挺多方式的,基于Cinemachine的Dolly Path是一个简单便捷的方法,咱们来快速的学习一下吧! 洪流学堂,让你快人几步.你好,我是跟着大智(VX: zh ...

  3. 如何在Unity中制作VR全景动画

    超简单的引言 本教程介绍的是如何一个人制作出像模像样的VR全景动画.笔者没有去详细的研究过动画的制作方式,或者说根本就不了解啊,所以本文章只是门外汉的一次自嗨,请勿认真,作为一种参考即可. 一 总览 ...

  4. Protobuf(Protocol Buffer)在Unity中的简单应用

    一.Protobuf的概念 高效轻便的数据存储格式(序列化和反序列化). 与平台和语言无关. 在网络通信和数据存储上应用广泛. 二.Protobuf的工作流 环境安装 使用Visual Studio2 ...

  5. 关于Unity中如何判断一个动画播放结束

    方法一(强力推荐): 在动画结束帧或其他帧处加个动画事件,在播放到这一帧的时候会自动调用这个动画函数 如图,找到对应动画的inspector面板,在里面有个Events下拉条,下拉后在想要的帧的位置添 ...

  6. 蒙皮骨骼动画 unity_在Unity中沿航路点设置蒙皮网格动画

    蒙皮骨骼动画 unity This post is a guide to importing a skinned mesh with a looping animation and then anim ...

  7. Spine变色动画在unity中失效的问题

    最近美术同事做spine变色动画的时候打开了Tint Black选项来做图片颜色过渡变化的效果.但是到了程序这边在unity中显示的是灰度,没有颜色变化. 在unit中有问题的spine动画材质图片如 ...

  8. Unity中的动画系统

    学习动画系统过程中的一些笔记 概述 Unity 的动画系统基于动画剪辑(Animation Clips)的概念,每个动画剪辑可以被认为是一个单一的线性记录,由动画状态机(Animator Contro ...

  9. Photon在unity中的使用

    ps:期末老师要求的一篇3000字的玩具级别论文,基本是官网和api手册原本就有的.基本我就是个搬运工,不知道有什么用,就先扔上来了.用的是writage,让word转为markdown,可能会出现一 ...

最新文章

  1. Xposed: 勾住(Hook) Android应用程序对象的方法,实现AOP
  2. MFC/Windows API 使用过的函数(持续更新)
  3. SCCM2012之客户端发现与安装
  4. 2016及以后的自动化测试趋势 -《测试技术六月刊》
  5. 通过制作数字桌面游戏和Web应用程序学习JavaScript
  6. ns2相关学习——TCL脚本编写(3)
  7. uniapp 开发基础环境搭建和配置
  8. centos java创建文件_CentOS java生成文件并赋予权限的问题
  9. 特征工程系列:特征预处理(下)
  10. .NET下一种简单的调试诊断方法
  11. CentOS6.5 webserver---网络配置
  12. c语言文件操作之图片+文件“合成器”(详解+源码+视频讲解)
  13. PSD格式截图软件 ScreenToLayers 1.2.3中文版
  14. 对ratings_data和trust_data进行分析
  15. LeGO-LOAM资料整理
  16. 电子计算机4个发展阶段的划分,计算机以什么划分发展阶段
  17. css3切角文本框_CSS3如何实现4个切角
  18. css:css样式背景图片设置透明度,css如何设置背景图片的透明度
  19. 【C++】C++静态库和动态库的区别
  20. parameter缩略语_parameter的缩写

热门文章

  1. 10个实用技巧让你的 Vue 代码更优雅
  2. 微软强调Win8.1 Update照顾惯用键鼠用户
  3. 2009年经典语录雷人总汇
  4. 英语中For和To的区别
  5. 微信开发获取昵称乱码 emoji表情
  6. [Eclips 安装] eclipse启动不了,出现“Java was...”如何解决
  7. eclips安装教程
  8. Python使用geany工具配置方法
  9. 星起航长期看好抖音小店的潜力
  10. 云边协同关3篇政策与技术跟踪专题文章摘录