当前位置: 首页 > news >正文

“摄像机”跟随及攻击抖动实现

学习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当中我们也可以自己创建震动效果,波幅的大小也可以使用震动文件,来产生屏幕上实际的震动效果


http://www.mrgr.cn/news/57307.html

相关文章:

  • 力扣71~75题
  • w~自动驾驶合集9
  • 调查显示软件供应链攻击增加
  • Matplotlib学习Day19
  • LLaMA Factory环境配置
  • 提高采购效率:掌握采购订单跟踪管理的关键工具
  • Linux基础IO
  • Android Handler(Looper.getMainLooper()),Kotlin
  • priority_queue (优先级队列的使用和模拟实现)
  • K折交叉验证代码实现——详细注释版
  • IPC 信号-Signal Linux环境
  • 栈的顺序存储总览
  • 关于风险系统解读最全最专业文章:一篇文章讲透风险,跨学科搞懂风险游戏规则,风险信任风险主观性客观性风险本质人格特质与风险态度技术风险系统风险社会新产品风险
  • 栈和队列代码
  • ARM/Linux嵌入式面经(五二):华为
  • Spring 设计模式之单例模式
  • C++新基础类型(C++11~C++20)
  • ECharts图表图例11
  • 解决cad找不到vcruntime140_1.dll,无法继续执行代码的6种方法
  • 《YOLO 目标检测》—— YOLO v3 详细介绍
  • 拟态UI3.0个人页
  • django模板相关配置
  • 一个将.Geojson文件转成shapefile和kml文件的在线页面工具(续)
  • hive数据库,表操作
  • 前缀和 有图文 超详细整理通俗易懂
  • OpenEular + KVM + virt-manager 笔记