10 相机控制

本节主要学习如何控制相机自动调整位置和焦距,使两个坦克一直同时在视野内.

 
image

在Hierarchy点击右键

 
image

点击 Create Empty,生成空对象,改名为CameraRig

 
image

设置CameraRig rotation为(40,60,0)

 
image

将MainCamera拖拽到CameraRig中,使之成为CameraRig的子对象

设置MainCamera的position为(0,0,-65),rotation为(0,0,0)

 
image

Orthographic Camera

 
image

Orthographic size

 
image

通过设置size实现缩放

Aspect宽高比

 
image

相机要做到事情有两个:跟随坦克和通过缩放实现坦克一直在屏幕内.

在wm/Scripts文件夹新建脚本CameraRig ,挂载到CameraRig上面

下面我们首先实现跟随坦克

 
image

跟随坦克首先找到坦克坐标的中间坐标,然后把cameraRig移动过去

先计算中间坐标

public Transform[] targets; // tanks

private Vector3 targetPos; // 计算出来的目标点

void FindAveragePos(){ // 计算平均坐标

    Vector3 avg = new Vector3 (); // 计算平均坐标的临时变量

    for (int i = 0; i < targets.Length; i++) { // 所有tank遍历一遍

        avg += targets [i].position; // 所有坐标都加起来

    }

    avg /= targets.Length; // 除以坦克个数,得到坐标的平均值

    avg.y = transform.position.y; // 坐标的y轴保持不变

    targetPos = avg; // 更新目标点

}

然后使用 Vector3.SmoothDamp 平滑阻尼是相机移动过去

还需要声明两个变量

private Vector3 moveSpeed; // 移动速度,SmoothDamp里面使用的

private float dampTime = 0.2f;// 到达目标的大约时间,SmoothDamp里面使用的

然后再Update()里面添加

transform.position = Vector3.SmoothDamp (transform.position, targetPos, moveSpeed, dampTime);

就可以缓动运动过去了.

 
image

CameraRig代码:

using UnityEngine;

using System.Collections;

public class CameraRig: MonoBehaviour {

public Transform[] targets; // tanks

private Vector3 targetPos; // 计算出来的目标点

private Vector3 moveSpeed; // 移动速度,SmoothDamp里面使用的

private float dampTime = 0.2f;// 到达目标的大约时间,SmoothDamp里面使用的

// Use this for initialization

void Start () {

} 

// Update is called once per frame

void Update () {

    FindAveragePos (); // 计算平均坐标

    transform.position = Vector3.SmoothDamp (transform.position, targetPos,ref moveSpeed, dampTime);// 缓动效果

}

void FindAveragePos(){ // 计算平均坐标

    Vector3 avg = new Vector3 (); // 计算平均坐标的临时变量

    for (int i = 0; i < targets.Length; i++) { // 所有tank遍历一遍

        avg += targets [i].position; // 所有坐标都加起来

    }

    avg /= targets.Length; // 除以坦克个数,得到坐标的平均值

    avg.y = transform.position.y; // 坐标的y轴保持不变

    targetPos = avg; // 更新目标点

}

}

回到unity,将两个坦克放到Targets里面

 
image

点击Play运行,就可以看到无论两个坦克怎么移动,此时相机一直会在两个坦克之间.

 
image

但是不一定总能看到两个坦克.

下面来计算一下size实现正确的缩放.

 
image

中间的点为targetPos,就是我们上面计算出来的那个坐标.

Y轴 size = distance.y

X轴 size = distance.x / aspect

取大的

 
image

最终版本代码截图

 
image

CameraRig最终代码:

using UnityEngine;

using System.Collections;

public class CameraRig: MonoBehaviour {

public Transform[] targets; // tanks

private Vector3 targetPos; // 计算出来的目标点

private Vector3 moveSpeed; // 移动速度,SmoothDamp里面使用的

private float dampTime = 0.2f;// 到达目标的大约时间,SmoothDamp里面使用的

private Camera camera; // 相机

private float screenEdgeBuffer = 4f; // 边缘缓冲

private float minZoom = 6.5f; // 最小缩放值

private float zoomSpeed ;// 缩放速度 SmoothDamp里面使用

// Use this for initialization

void Start () {

    camera = GetComponentInChildren<Camera> ();

} 

// Update is called once per frame

void Update () {

    FindAveragePos (); // 计算平均坐标

    transform.position = Vector3.SmoothDamp (transform.position, targetPos,ref moveSpeed, dampTime);// 缓动移动

    float zoom = FindZoomSize (); // 计算缩放值

    camera.orthographicSize = Mathf.SmoothDamp (camera.orthographicSize, zoom, ref zoomSpeed, dampTime);// 缓动缩放

}

void FindAveragePos(){ // 计算平均坐标

    Vector3 avg = new Vector3 (); // 计算平均坐标的临时变量

    for (int i = 0; i < targets.Length; i++) { // 所有tank遍历一遍

        avg += targets [i].position; // 所有坐标都加起来

    }

    avg /= targets.Length; // 除以坦克个数,得到坐标的平均值

    avg.y = transform.position.y; // 坐标的y轴保持不变

    targetPos = avg; // 更新目标点

}

float FindZoomSize(){ // 计算缩放大小

    float size = 0; // 临时缩放值变量

    Vector3 localTargetPos = transform.InverseTransformPoint (targetPos);// 将目标点转换为相对于当前transform的本地坐标

    for (int i = 0; i < targets.Length; i++) { // 变量所有坦克

        Vector3 tankLocalPos = transform.InverseTransformPoint ( targets [i].position); // 坦克本地坐标

        Vector3 distance = targetPos - tankLocalPos; // 和当前坦克的距离

        size = Mathf.Max (size, Mathf.Abs (distance.y)); // y轴距离,取大的

        size = Mathf.Max (size, Mathf.Abs (distance.x)  / camera.aspect ) ; // x轴距离/aspect,取大的

    }

    size += screenEdgeBuffer; // 加上屏幕边缘缓冲值

    size = Mathf.Max (size, minZoom); // 不能小于最小缩放值

    return size;

}

}

---------------------------我是目录分割线---------------------------

《杜增强讲Unity之Tanks坦克大战》1-准备工作

《杜增强讲Unity之Tanks坦克大战》2-场景设置

《杜增强讲Unity之Tanks坦克大战》3-添加坦克

《杜增强讲Unity之Tanks坦克大战》4-坦克的移动和旋转

《杜增强讲Unity之Tanks坦克大战》5-子弹

《杜增强讲Unity之Tanks坦克大战》6-发射子弹

《杜增强讲Unity之Tanks坦克大战》7-坦克血条

《杜增强讲Unity之Tanks坦克大战》8-子弹碰撞处理

《杜增强讲Unity之Tanks坦克大战》9-发射子弹时蓄力

《杜增强讲Unity之Tanks坦克大战》10-相机控制

《杜增强讲Unity之Tanks坦克大战》11-游戏流程控制

---------------------------我是目录分割线---------------------------

《杜增强讲Unity之Tanks坦克大战》10-相机控制的更多相关文章

  1. 《杜增强讲Unity之Tanks坦克大战》11-游戏流程控制

    11 游戏流程控制 使用协程来控制游戏流程 11.1 添加MessageText 首先添加一个Text来显示文字   image 设置GameMgr   image 11.2 游戏整体流程 下面Gam ...

  2. 《杜增强讲Unity之Tanks坦克大战》9-发射子弹时蓄力

    9 发射子弹时蓄力 实现效果如下   image 按下开火键(坦克1为空格键)重置力为最小力,一直按着的时候蓄力,抬起的时候发射.如果按着的时候蓄力到最大,则自动发射,此时在抬起则不会重复发射. 首先 ...

  3. 《杜增强讲Unity之Tanks坦克大战》4-坦克的移动和旋转

    4 坦克移动和旋转 本节课的目标是实现同时wsad和上下左右控制两个坦克分别移动和旋转 4.1 本节代码预览   image 将上节课场景s2另存为s3. 4.2 添加车轮扬沙效果 从Prefabs里 ...

  4. 《杜增强讲Unity之Tanks坦克大战》1-准备工作

    0.案例介绍 0.1开始界面   点击Play Now 进入游戏界面   左边的坦克使用ws控制前后移动,ad键左右旋转,空格键开火   右边的坦克使用方向键上下控制前后移动,方向键左右键实现左右旋转 ...

  5. 《杜增强讲Unity之Tanks坦克大战》2-场景设置

    2  场景设置 2.1 本节效果预览   2.2 项目目录设置 点击Project面板的Create按钮,在根目录下面新建wm文件夹   Wm文件夹用于存放我们自己生成的Prefab和脚本等其他资源, ...

  6. 《杜增强讲Unity之Tanks坦克大战》3-添加坦克

    3 添加坦克 3.1 本节效果预览   3.2 另存新场景 首先打开上次的场景s1,另存为s2,放到同一个文件夹下面.   3.3 添加坦克模型 在Model文件夹下面找到Tank模型   将Tank ...

  7. 《杜增强讲Unity之Tanks坦克大战》5-子弹

    5 子弹 本节的任务是创建子弹的Prefab   image 首先从Model/Shell找到子弹的模型,拖入Hierarchy中,添加刚体组件,所有属性默认值. 添加Capsule Collider ...

  8. 《杜增强讲Unity之Tanks坦克大战》6-发射子弹

    6 发射子弹 本节完成发射子弹的功能,最终代码如下:   image 首先,发射子弹得确定发射的位置和方向,还有发射的初始速度.具体的发射速度和按下发射按键的时间长短有关,这个关于子弹的蓄力我们在第九 ...

  9. 《杜增强讲Unity之Tanks坦克大战》7-坦克血条

    7 坦克血条 点击菜单GameObject->UI->Slider创建Slider   选中EventSystem,设置Horizontal Axis为HorzontalUI,Vertic ...

随机推荐

  1. 【转】Spring学习---SpringIOC容器的初始化过程

    [原文]https://www.toutiao.com/i6594400249429623304/ SpringIOC容器的初始化过程 简单来说,IoC容器的初始化是由refresh()方法来启动的, ...

  2. 关于使用python的open函数时报No Such File or DIr的错误

    我写的代码如下: def createFileWithFileName(localPathParam,fileName): totalPath=local_url+'\\'+fileName if n ...

  3. 开源作业调度框架 - Quartz.NET - Cron表达式测试

    昨天简单写了一下如何使用Quzrtz.NET. 那么问题来了,我设置了Cron表达式之后如何知道是表达式是否按照预期的时间执行了呢? 我找到了些Cron表达式工具生成了表达式,确发现它们基本上没有进行 ...

  4. Symbol Tables

    符号表 符号表是键值对的集合,支持给定键查找值的操作,有很多应用: API put() 和 get() 是最基础的两个操作,为了保证代码的一致性,简洁性和实用性,先说下具体实现中的几个设计选择. 泛型 ...

  5. 设置webstorm支持ES6语法

    1.  点击File目录下的Default Settings 2.  再依次点击Languages & Frameworks  ----->  JaveScript  ----> ...

  6. web.xml配置遇到的问题

    web.xml<listener>            <listener-class>org.springframework.web.context.ContextLoad ...

  7. (三) DRF 序列化

    一.单表的GET和POST: 使用serializers序列化,针对每一个表,需要单独写函数.一般会写在views.py里面,但是这样做,会导致整个文件代码过长.需要分离出来! 在app01(应用名) ...

  8. js判断文本是否溢出容器

    onShowNameTipsMouseenter: function(e) { var target = e.target; var containerLength = $(target).width ...

  9. python利用imap实现伪“无痕”取信

    所谓无痕取信,目前主要是指从邮箱中把信件收取后,邮箱内状态不发生任何改变.这里的状态主要是指两部分,一部分是邮件状态不变,即已读与未读状态不变,另一部分是指邮箱记录的登陆IP不发生改变.本文中所说的伪 ...

  10. [Lydsy1805月赛]口算训练 BZOJ5358

    分析: 没想到这道题还能二分查找... 这题主席树的话,裸的很显然...我们将每一个数分解质因数,之后建一个可持久化权值线段树维护[L,R]区间内的每一种质因子的个数,分解质因数的话,可以选择用线筛, ...