在Unity中实现简单的动画转场
在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中实现简单的动画转场相关推荐
- 初探CardBoard:(1)在Unity中实现简单VR场景
初探CardBoard:(1)在Unity中实现简单VR场景 为何使用CardBoard 价格低廉 兼容性 前期准备 开始工程 一.简单的环境布置 二.导入SDK 三.SDK基础组件说明 四.将之前的 ...
- Cinemachine教程 | Unity中如何制作路径动画?
摘要:在Unity中制作路径动画是有挺多方式的,基于Cinemachine的Dolly Path是一个简单便捷的方法,咱们来快速的学习一下吧! 洪流学堂,让你快人几步.你好,我是跟着大智(VX: zh ...
- 如何在Unity中制作VR全景动画
超简单的引言 本教程介绍的是如何一个人制作出像模像样的VR全景动画.笔者没有去详细的研究过动画的制作方式,或者说根本就不了解啊,所以本文章只是门外汉的一次自嗨,请勿认真,作为一种参考即可. 一 总览 ...
- Protobuf(Protocol Buffer)在Unity中的简单应用
一.Protobuf的概念 高效轻便的数据存储格式(序列化和反序列化). 与平台和语言无关. 在网络通信和数据存储上应用广泛. 二.Protobuf的工作流 环境安装 使用Visual Studio2 ...
- 关于Unity中如何判断一个动画播放结束
方法一(强力推荐): 在动画结束帧或其他帧处加个动画事件,在播放到这一帧的时候会自动调用这个动画函数 如图,找到对应动画的inspector面板,在里面有个Events下拉条,下拉后在想要的帧的位置添 ...
- 蒙皮骨骼动画 unity_在Unity中沿航路点设置蒙皮网格动画
蒙皮骨骼动画 unity This post is a guide to importing a skinned mesh with a looping animation and then anim ...
- Spine变色动画在unity中失效的问题
最近美术同事做spine变色动画的时候打开了Tint Black选项来做图片颜色过渡变化的效果.但是到了程序这边在unity中显示的是灰度,没有颜色变化. 在unit中有问题的spine动画材质图片如 ...
- Unity中的动画系统
学习动画系统过程中的一些笔记 概述 Unity 的动画系统基于动画剪辑(Animation Clips)的概念,每个动画剪辑可以被认为是一个单一的线性记录,由动画状态机(Animator Contro ...
- Photon在unity中的使用
ps:期末老师要求的一篇3000字的玩具级别论文,基本是官网和api手册原本就有的.基本我就是个搬运工,不知道有什么用,就先扔上来了.用的是writage,让word转为markdown,可能会出现一 ...
最新文章
- Xposed: 勾住(Hook) Android应用程序对象的方法,实现AOP
- MFC/Windows API 使用过的函数(持续更新)
- SCCM2012之客户端发现与安装
- 2016及以后的自动化测试趋势 -《测试技术六月刊》
- 通过制作数字桌面游戏和Web应用程序学习JavaScript
- ns2相关学习——TCL脚本编写(3)
- uniapp 开发基础环境搭建和配置
- centos java创建文件_CentOS java生成文件并赋予权限的问题
- 特征工程系列:特征预处理(下)
- .NET下一种简单的调试诊断方法
- CentOS6.5 webserver---网络配置
- c语言文件操作之图片+文件“合成器”(详解+源码+视频讲解)
- PSD格式截图软件 ScreenToLayers 1.2.3中文版
- 对ratings_data和trust_data进行分析
- LeGO-LOAM资料整理
- 电子计算机4个发展阶段的划分,计算机以什么划分发展阶段
- css3切角文本框_CSS3如何实现4个切角
- css:css样式背景图片设置透明度,css如何设置背景图片的透明度
- 【C++】C++静态库和动态库的区别
- parameter缩略语_parameter的缩写