[游戏开发-学习笔记]菜鸟慢慢飞(四)-Camera

游戏开发中,主相机应该是最重要的GameObject之一,毕竟游戏呈现给玩家,就是通过它。
相机的使用,在不同的游戏中,有很大的不同。这里总结一下自己学到的一些相关知识。

固定位置-游戏过程中相机的Transform属性不改变。

  • 调整好位置后就不要动了,一般使用正交相机,即Camera-Projection选择Orthographic。Unity Manual-Camera
    适用:2D游戏。比如飞机大战,消消乐。
  • 游戏开始后,相机追踪某一物体,然后固定不动。
    游戏开始后,我们才能确定追踪物体的位置,先看代码:

 1 using UnityEngine;
 2
 3 public class CameraController : MonoBehaviour
 4 {
 5     public GameObject player;
 6     private Vector3 offset;
 7     void Start ()
 8     {
 9         offset = transform.position - player.transform.position;
10     }
11     void LateUpdate ()
12     {
13         transform.position = player.transform.position + offset;
14     }
15 }

适用:固定视角的3D游戏。

3D视角-围绕一个中心随意旋转

先看代码:

using UnityEngine;

public class CameraContorller : MonoBehaviour
{
    /// <summary>
    /// 追踪目标
    /// </summary>
    public Transform target;
    /// <summary>
    /// 旋转速度
    /// </summary>
    public float xSpeed = 200, ySpeed = 200;
    /// <summary>
    /// 缩放速度
    /// </summary>
    public float mSpeed = 10;
    /// <summary>
    /// 最小/最大限制角度
    /// </summary>
    public float yMinLimit = -50, yMaxLimit = 50;
    /// <summary>
    /// 是否使用插值运算
    /// </summary>
    public bool needDamping = true;
    /// <summary>
    /// 速度
    /// </summary>
    public float damping = 5.0f;

    /// <summary>
    /// 观察距离
    /// </summary>
    private float distance = 2;
    /// <summary>
    /// 最小/最大观察距离
    /// </summary>
    private float minDistance = 2, maxDistance = 30;
    /// <summary>
    /// 旋转角度
    /// </summary>
    private float x = 0.0f, y = 0.0f;

    void Start()
    {
        Vector3 angles = transform.eulerAngles;
        x = angles.y;
        y = angles.x;
        //根据相机的初始位置初始化距离,最大距离,最小距离;
        Vector3 offset = transform.position - target.position;
        distance = offset.magnitude;
        minDistance = distance / 5;
        maxDistance = distance * 3;
    }

    void LateUpdate()
    {
        if (target)
        {
            if (Input.GetMouseButton(1))//0-左键,1-右键,2-中键
            {
                x += Input.GetAxis("Mouse X") * xSpeed * 0.02f;
                y -= Input.GetAxis("Mouse Y") * ySpeed * 0.02f;

                y = ClampAngle(y, yMinLimit, yMaxLimit);
            }
            distance -= Input.GetAxis("Mouse ScrollWheel") * mSpeed;
            distance = Mathf.Clamp(distance, minDistance, maxDistance);

            Quaternion rotation = Quaternion.Euler(y, x, 0.0f);
            Vector3 disVector = new Vector3(0.0f, 0.0f, -distance);
            Vector3 position = rotation * disVector + target.position;
            if (needDamping)
            {
                transform.rotation = Quaternion.Lerp(transform.rotation, rotation, Time.deltaTime * damping);
                Vector3 transformTemp = Vector3.Lerp(transform.position, position, Time.deltaTime * damping);
                if (transformTemp.y <= 0)//如果相机位置在“水平面”以下,强制设置到水平面。
                {
                    transform.position = new Vector3(transformTemp.x, 0, transformTemp.z);
                }
                else
                {
                    transform.position = transformTemp;
                }
            }
            else
            {
                transform.rotation = rotation;
                if (position.y <= 0)//如果相机位置在“水平面”以下,强制设置到水平面。
                {
                    transform.position = new Vector3(position.x, 0, position.z);
                }
                else
                {
                    transform.position = position;
                }
            }
        }
    }

    static float ClampAngle(float angle, float min, float max)
    {
        if (angle < -360)
            angle += 360;
        if (angle > 360)
            angle -= 360;
        return Mathf.Clamp(angle, min, max);
    }
}

大部分3D游戏中,我这样理解:主角看作球心,球半径玩家可以自己改变,相机的位置就在“地面”以上的半球球体上。
上面这段代码基本做到了这个要求,并在相机到“地面”的时候有抬头看的效果。
代码有点问题,开始游戏后相机会跳动两次,暂时放到自己的问题清单中。

其他用法

官方教程Tanks tutorial 中有个不常见的用法。
不管两个Tank如何移动,相机通过前后缩放,大小缩放保证两个Tank肯定在相机视野内。
主要代码:(官方代码写得好啊)

private void Move ()
        {
            // Find the average position of the targets.
            FindAveragePosition ();

            // Smoothly transition to that position.
            transform.position = Vector3.SmoothDamp(transform.position, m_DesiredPosition, ref m_MoveVelocity, m_DampTime);
        }

        private void FindAveragePosition ()
        {
            Vector3 averagePos = new Vector3 ();
            int numTargets = 0;

            // Go through all the targets and add their positions together.
            for (int i = 0; i < m_Targets.Length; i++)
            {
                // If the target isn‘t active, go on to the next one.
                if (!m_Targets[i].gameObject.activeSelf)
                    continue;

                // Add to the average and increment the number of targets in the average.
                averagePos += m_Targets[i].position;
                numTargets++;
            }

            // If there are targets divide the sum of the positions by the number of them to find the average.
            if (numTargets > 0)
                averagePos /= numTargets;

            // Keep the same y value.
            averagePos.y = transform.position.y;

            // The desired position is the average position;
            m_DesiredPosition = averagePos;
        }

        private void Zoom ()
        {
            // Find the required size based on the desired position and smoothly transition to that size.
            float requiredSize = FindRequiredSize();
            m_Camera.orthographicSize = Mathf.SmoothDamp (m_Camera.orthographicSize, requiredSize, ref m_ZoomSpeed, m_DampTime);
        }

        private float FindRequiredSize ()
        {
            // Find the position the camera rig is moving towards in its local space.
            Vector3 desiredLocalPos = transform.InverseTransformPoint(m_DesiredPosition);

            // Start the camera‘s size calculation at zero.
            float size = 0f;

            // Go through all the targets...
            for (int i = 0; i < m_Targets.Length; i++)
            {
                // ... and if they aren‘t active continue on to the next target.
                if (!m_Targets[i].gameObject.activeSelf)
                    continue;

                // Otherwise, find the position of the target in the camera‘s local space.
                Vector3 targetLocalPos = transform.InverseTransformPoint(m_Targets[i].position);

                // Find the position of the target from the desired position of the camera‘s local space.
                Vector3 desiredPosToTarget = targetLocalPos - desiredLocalPos;

                // Choose the largest out of the current size and the distance of the tank ‘up‘ or ‘down‘ from the camera.
                size = Mathf.Max(size, Mathf.Abs(desiredPosToTarget.y));

                // Choose the largest out of the current size and the calculated size based on the tank being to the left or right of the camera.
                size = Mathf.Max(size, Mathf.Abs(desiredPosToTarget.x) / m_Camera.aspect);
            }

            // Add the edge buffer to the size.
            size += m_ScreenEdgeBuffer;

            // Make sure the camera‘s size isn‘t below the minimum.
            size = Mathf.Max (size, m_MinSize);

            return size;
        }

暂时放下的内容

Unity的相机相关:

时间: 2024-08-01 21:48:40

[游戏开发-学习笔记]菜鸟慢慢飞(四)-Camera的相关文章

[游戏开发-学习笔记]菜鸟慢慢飞(一)

"菜鸟"就是我自己. 分享一下我从零开始学习游戏开发的过程,心得什么的.本篇先扯个开头,说说个人一些乱七八糟的想法. 从0开始 其实,只要有心,不管干哪一行,都不能算从零开始.总会有办法了解到行业的信息.我之前是干船舶电气的,一路过来,最大的感觉是:不怕你找不到资料,就怕你没时间.我要做的是分清楚自己想学什么,怎么样去提高自己,然后沿着自己路线"慢慢"朝前飞.比如:知乎就是个很好的地方,我最早接触游戏开发,是在这里.决定学这个,也是在知乎喝了鸡汤. 游戏开发 对我来

[游戏开发-学习笔记]菜鸟慢慢飞(二)-迷宫

简介:练手Demo,<走出迷宫>,文章主要说说如何创建迷宫. 学习Unity3D有一段时间了,自己想了一个项目来练手手.然后就有了这篇. 一.固定的格数,开局后随机生成. 说明:这个迷宫10*10,开始后随机生成,四周留下一个空做出口. 先说如何实现: 主要准备了三个Prefab:横墙,竖墙,柱子,墙高度是10,宽度是10,厚度是1,柱子高度是10.宽度和厚度都是10. 手动按照10*10排列(参考下图) 脚本 #region //初始化游戏 #endregion using System.C

[游戏开发-学习笔记]菜鸟慢慢飞(九)- NGUI- UIPanel(官方说明翻译)

我自己笔记是做在OneNote上,直接复制粘贴过来变成图片了,效果好像还可以. 机器翻译,我自己看了一下,改了一部分.

[游戏开发-学习笔记]菜鸟慢慢飞(七)-&#160;迷宫更新

随机迷宫,留下一个问题: "Quaternion这个API看不懂,先放到学习清单中,如果会这个,应该就可以横墙,竖墙只用一个." 查看了下官方API. 只用一面墙,效果是一样的. 更新一下代码: using UnityEngine; /// <summary> /// 初始化迷宫 /// </summary> public class InitializeMazerev1 : MonoBehaviour { public GameObject WallHS; p

[游戏开发-学习笔记]菜鸟慢慢飞(五)-你怎么做笔记?

做笔记?做个笔记呗?做个笔记吧? 我参考的 知乎-如何构建自己的笔记系统? 中票数最高的 "INK笔记法". 工具:OnNote. - InBox,看视频,看书,来灵感了.记下来 - Note 整理过后的知识点.    - Knowledge 总结,成篇成章. 归纳,总结,也是再学一遍的过程,很有必要.

[游戏开发-学习笔记]菜鸟慢慢飞(三)-官方教程学习小心得

自己的事情自己做 举例:官方教程<Tanks tutorial>中,小坦克:移动,移动的声音,射击,生命值的管理,等Component都挂载在GameObject坦克自己的身上.炮弹,则管理自己的爆炸等. 好处不少: ~开发维护的时候更加方便 ~符合"面对对象"的思想 一个脚本做一件事情 举例:官方教程<Tanks tutorial>中,小坦克:c#脚本分为三个,移动,生命管理,射击. 好处很多: ~在炮弹的爆炸脚本可以单独调用生命管理去更改生命值. ~代码更加

[游戏开发-学习笔记]菜鸟慢慢飞(11)- Unity3D中的Json

关键词:Json,C#,LitJson,Unity3D 内容:Unity3D中,Json的一些常规用法. 定义:(必应词典) Json(JavaScript Object Notation)是一种由道格拉斯·克罗克福特构想设计.轻量级的数据交换语言,以文字为基础,且易于让人阅读.尽管JSON是Javascript的一个子集,但JSON是独立于语言的文本格式,并且采用了类似于C语言家族的一些习惯. Json可以分为两个部分: Json Object(A collection of name/val

[游戏开发-学习笔记]菜鸟慢慢飞(六)-&#160;冒泡排序

1 /// <summary> 2 /// 冒泡排序算法 3 /// </summary> 4 /// <param name="array"></param> 5 public static void sort(int[] arr) 6 { 7 int temp = 0; 8 for (int i = 0; i < arr.Length - 1; i++) 9 { 10 for (int j = 0; j < arr.Len

[游戏开发-学习笔记]菜鸟慢慢飞(八)-&#160;插入排序

static void InsertSort(int[] array) { for (int i = 1; i < array.Length; i++) { int temp = array[i]; int j = i; while (j > 0 && array[j - 1] > temp) { array[j] = array[j - 1]; j--; } array[j] = temp; } }