学习Unity的摄像机功能,可以帮助我们实现摄像机对人物的跟随移动,还可以使用这个工具自带的插件,摄像机震动,颤动,增强打击感;
首先来安装一下这个插件,window菜单--packageManage--左上角Unity Registry--右上角搜索插件Cinemachine,install安装
在Hierarchy窗口直接来添加摄像机
左上角“ + ”--Cinemachine--2D Cinema,保留原来的命名
然后会发现当前Game窗口,人物和场景都不见了
因为Virtual Camera并没有选择跟随谁,或者正在观看谁
我们把Player拖拽进来,Look At,Follow
现在就可以成功看到Player,把摄像机放在列表上方;
有了这个摄像机,默认情况下就可以实现人物跟随,运行游戏查看效果
我们还可以调整一下详细的设置,在Body的下拉菜单当中
可以调整跟随的中心点位置,人物是脚底为中心点,可以调整一下y轴,保证跟随人物中心点位置
Dead Zone Width,Dead Zone Height,通过调整这个高度和宽度可以给摄像机有一定的缓冲值;所以如果人物没有移动超过这个范围以上,他是不会进行摄像机的跟随的
点击空白位置,然后运行游戏
稍微调整位移,小范围的位移,摄像机不会进行跟随,一旦超出范围,摄像机才做一个跟随(里面有非常多的选项,可以进行自主调整)
接下来解决摄像机跟随我们到什么程度
切换到Scene的场景当中,我们先把这个场景画的大一些
我们希望限制一下我的摄像机跟随的范围,不希望摄像机穿帮透露出我们的画面下边和上边没有绘制的区域,所以我们要给他添加一个边界的限制
我们留意在virtal Camera选择的情况下,inspector窗口下方,有Extensions可以给他添加扩展内容,这个扩展内容当中有非常多的东西,点击下拉菜单,有很多的选项,
首先我们要添加一个非常重要的东西Cinemachine Pixel Perfect(帮助我们在我们的像素旋转或产生畸变的时候,不会出现像素扭曲,保持我们的单位像素
另外我们还要添加一个工具Cinemachine Confiner2D(帮助我们限定一定的区域来控制摄像机的移动范围)
在Bounding Shape 2D中要添加一个Collider2D类型的变量,而这个变量可以通过我们整个场景当中的大型的collider绘制来限制相机的可移动的范围,所以接下来我们来实现一下摄像机的限制范围
在Hierarchy窗口创建一个新的game object(Creat Empty),命名为Bounds边界范围;
为Bounds添加一个Polygon Collider 2D的组件
勾选Is Trigger(不然的话,就会把我们的人物以及其他所有的碰撞体相关的内容都会被弹开)
设置一下边界位置,可以设置任意的形状,然后摄像机将只在这个形状范围之内产生跟随的移动
先随便设置一个样子,绿色的线框
我们点击Virtual Camera,将Bounds拖拽进来给到我们的Boounding Shape 2D
点击运行测试,接下来我们就可以看到摄像机只能在我们绘制的区域内移动了,摄像机有了一定的区域边框的限制
这样我们就实现了场景当中摄像机的边界,可以自己修改一下,将边界调整一下
调整规整边界,我们可以删除它多余的points,一半用四个点做一个矩形就可以了,可以通过实际的设置数值来调整点的位置
这样我们就绘制好边框了,可以运行测试一下
可以看到整个摄像机只能在绿色的范围内进行移动
这样我们就成功实现了边框的内容
以后我们还会有多个场景,从一个场景转换到另外一个场景,我们如何获得当前场景的这个摄像机的边框位置,我们要在代码当中来进行实现
我们通过代码的方式,每次切换场景,要获得一下这个摄像机的边界,所有我们为了方便查找,先为这个边界添加一个标签
选择Bounds,Tag--Add Tag 起名为Bounds
重新选择这个物体,为他选择这个标签
我们希望我的Virtual Camera可以获得我的Cinemachine Confined 2D这个组件,然后我们通过修改这个变量值,每次切换场景之后,都去找一下这个被标记了Bounds的这个物体,将他身上的Collider添加到Cinemachine Confined 2D的Bounding Shape 2D
接下来来创建代码
选择Script文件夹,选择Utillitiles下创建一个c#脚本,叫做CameraControl(摄像机控制脚本)
将这个代码挂载在virtual Camera上
打开代码
Cinemachine是一个插件,我们想要在代码当中调用Cinemachine的话,要先调用它的命名空间using Cinemachine
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Cinemachine;
然后接下来我们要获得confined 2D的组件
在Awake当中Getcomponent
public class CameraControl : MonoBehaviour
{private CinemachineConfiner2D confiner2D;private void Awake(){confiner2D = GetComponent<CinemachineConfiner2D>();}
}
接下来我们就可以通过我们的代码获得我们的Shape 2D
创建一个函数方法GetNewCameraBounds,以后在做场景切换的时候,可以直接用这个函数的方法来调用,获得我们所有场景当中的Bounds
private void GetNewCameraBounds(){}
创建一个临时变量obj在找到这个挂载了Bounds的物体,用GameObject的方法,在这里面unity帮我们内置了一个查找的方法,输入find,会提示给我们非常多的方法来使用,刚才我们标记了标签,所以在此处我们使用FindGameObjectWithTag
private void GetNewCameraBounds()
{var obj = GameObject.FindGameObjectWithTag("Bounds");
}
找到这个东西以后,我们要确定一下场景当中到底有没有;我们需要加上一个约束判断,如果OBJ为空,return不需要执行其他的内容,帮助我们停止代码;如果不为空,我们才执行下边的内容。
我们既然获得了物体,我们就可以获得他身上Collider2D的组件,然后给到我们的Confiner2D;Confiner2D的BoundingShape2D=obj上get的Confiner2D类型
Collider2D是一个基类,所有的Box Collider都基于我们的Collider2D,所以我们用Collider2D就可以代表所有类型的碰撞体了
private void GetNewCameraBounds(){var obj = GameObject.FindGameObjectWithTag("Bounds");if (obj == null)return;confiner2D.m_BoundingShape2D = obj.GetComponent<Collider2D>();}
保持代码返回unity
我们看一下我们Virtual Camera--Cinemachine Confiner,在我们获得了我们的碰撞体之后,下面有个按钮叫做Invalidate Cache(强制我们清理一下缓存,下边有详细的说明,如果一旦从一个场景切换到另一个场景,从一个边界切换到另一个边界,需要通过这个默认的函数方法,来清理掉之前的形状缓存,这样才能让他应用于新的这个形状),所以我们可以直接调用这个函数Invalidate Cache,我们在代码当中把它添加进去
打开CameraControl代码,
获得一个新的图形之后,要调用Invalidate Cache
这样我们就成功的在游戏一开始,获得的标签身上找到Collider给到我们的赋值,然后清一下缓存
public class CameraControl : MonoBehaviour
{private CinemachineConfiner2D confiner2D;private void Awake(){confiner2D = GetComponent<CinemachineConfiner2D>();}private void GetNewCameraBounds(){var obj = GameObject.FindGameObjectWithTag("Bounds");if (obj == null)return;confiner2D.m_BoundingShape2D = obj.GetComponent<Collider2D>();confiner2D.InvalidateCache();}
}
那我们在说明时候来调用呢,在游戏的一开始,我们暂时先写在这儿调用这个函数方法得到新的摄像机范围
在场景切换之后,我们还要在里面做一些更改和修整;目前暂时在游戏开始时获得这个边界
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Cinemachine;public class CameraControl : MonoBehaviour
{private CinemachineConfiner2D confiner2D;private void Awake(){confiner2D = GetComponent<CinemachineConfiner2D>();}//TODO:场景切换后的更改private void Start(){GetNewCameraBounds();}private void GetNewCameraBounds(){var obj = GameObject.FindGameObjectWithTag("Bounds");if (obj == null)return;confiner2D.m_BoundingShape2D = obj.GetComponent<Collider2D>();confiner2D.InvalidateCache();}
}
保存代码,返回unity
我们来看一下,我们先将Virtual Camera的Collider删掉,让这个component没有获得任何变量,运行游戏,查看他能不能得到我们的Bounds,然后赋值给我们的Virtual Camera;
可以看到它自动就获得了这个边界,场景当中自动绘制好
这样我们就成功实现了获取当前场景标签Bounds的这个物体,用这个物体来控制我的摄像机的边界
接下来我们要来添加一个摄像机的震动反馈;每次攻击的时候,屏幕上都会有一些震动,这样会让我们的动作游戏更有打击感的效果
在VirtualCamera的Cinemachine当中来添加另外一个extension扩展的工具,点击下拉菜单,选择CinemachineImpulseListener监听摄像机的震动(impulse突然的冲击力“在添加Rigidbody刚体的时候使用过”)
也可以看一下这里面的说明,这个ImpulseListener他会根据我们发送的信号,来产生实际的摄像机的震动,所以它需要一个震动源,就是CinemachineImpulseSource;有了这个震动源,他才会监听每一次的震动事件
另外下边也有选项,默认情况下Unity都是3D,我们摄像机距离背景z轴为10,有10的距离;震动的方向也是3D的,当前我们的是2D游戏,所以只要有上下左右的震动就好了,所以勾选 Use 2D Distan,
我们有了震动的监听,接下来我们需要一个震动源
我们在Hierarchy窗口创建一个新的空的GameObject(Create Empty)命名为Camera Shake摄像机震动,调整窗口画面,我们把摄像机都放在上面
在这个上面我们来添加震动源
Add Component搜索Cinemachine Impulse Source,
这就是一个震动源,这个震动源里面帮助我们提供了很多预制的震动效果
它下面为我们提供了一个Test with Force,也就是我们可以模拟测试一下,不过他只有在运行的时候才可以测试,点Invoke测试
我们点击运行,来看一下这个摄像机的震动效果,根据效果来修改我们想要的数值
Impulse Shape:Bump(上下震动),下拉菜单还要其他的效果;当前这些震动是有固定的波幅的,可以打开下拉菜单,查看它的波幅的样子,希望x,y方向都有震动,我们可以在下面添加一下数值
如果我们希望震动不是那么强烈的话,可以将振幅稍微改小一点
注意,要在停止运行的时候重新修改数值,确定数值修改好了
类型我们选择Explosion爆炸类型
我们希望在每次攻击的时候,启动一下这个震动
接下来到代码当中来实现(通过代码来实现,在这个效果框中,有提示内容,如果想要调用这个组件,我们要使用的ApI的函数方法是GenerateImpulse),所以我们调用这个组件,然后使用这个函数方法GenerateImpulse,就会播放震动
我们要在Virtual Camera当中获得一下这个Cinemachine Impulse Source;
打开CameraControl代码,当中创建一个变量impulseSource来获得它
public class CameraControl : MonoBehaviour
{private CinemachineConfiner2D confiner2D;public CinemachineImpulseSource impulseSource;
保存返回unity
进行拖拽赋值
现在我们有了CameraShake之后,我们就可以直接用GenerateImpulse来执行震动了
当人物和敌人攻击或者受伤的时候才播放震动,
用事件监听的方式;接下来我们再创建一个ScriptableObject的事件方法,在ScriptableObject文件夹下创建,这次我们不用传递参数,起名为VoidEventSO(没有参数的事件)只要不传递任何变量参数的,我们都可以用同一个SO去创建实际上的Ass文件,来实现我们想要的事件
打开代码,进行编写
快速创建
using UnityEngine;[CreateAssetMenu(menuName ="Event/VoidEventSO")]
public class VoidEventSO:ScriptableObject
{}
保存代码,返回unity,确定我们可以成功创建
在Project的Events文件夹下,点击加号选择Event--VoidEventSO,创建,
起名为CameraShake Event
打开代码,继续编写
在上面添加调用命名空间
我们既然不传递任何的参数,我们就使用最基本的方法,在下面创建UnityAction
不需要传递任何参数
using UnityEngine;
using UnityEngine.Events;[CreateAssetMenu(menuName ="Event/VoidEventSO")]
public class VoidEventSO:ScriptableObject
{public UnityAction OnEventRaised;public void RaiseEvent(){OnEventRaised?.Invoke();}
}
保存返回unity
选择Player,在Player受伤的基础只是,我们班这个CameraShake这个事件拖拽进来
只要人物受伤,就呼叫一下,广播人物受伤,摄像机实现震动
所以我们每一个敌人在受到伤害或者死亡的时候,我们都要添加这个方法,
这样广播就已经都设置好了
接下来要实现监听,监听我们就要到Virtual Camera,在CameraControl当中去进行监听了
打开代码,因为我们已经获得了这个组件impulse;所以我们要监听一下这个事件
public class CameraControl : MonoBehaviour
{private CinemachineConfiner2D confiner2D;public CinemachineImpulseSource impulseSource;public VoidEventSO cameraShakeEvent;
然后添加一下注册的方法,取名为OnCameraShakeEvent
private void OnEnable(){cameraShakeEvent.OnEventRaised += OnCameraShakeEvent;}private void OnDisable(){cameraShakeEvent.OnEventRaised -= OnCameraShakeEvent;}
然后让编译器帮助我们自动生成函数OnCameraShakeEvent,在这个函数中执行摄像机的震动
private void OnCameraShakeEvent()
{impulseSource.GenerateImpulse();
}
保存代码,返回unity,
先把事件拖拽到VirtualCamera
测试一下
这样我们就成功实现了摄像机的震动,受到攻击的摄像机反馈震动,整个游戏的打击感就有了很大的提升
在CameraShake当中我们也可以自己创建震动效果,波幅的大小也可以使用震动文件,来产生屏幕上实际的震动效果