Camera定义

游戏中,Camera用来向用户展示场景,Camera就像一个摄像机,摄像机里面的景象就是Camera的展示范围,如下图所示:

在3D空间中Camera被定义为一个位置,有一个单位“方向”向量和一个“向上”的单位向量组成,方向和向上向量告诉OpenGL 当前Camera如何定向。

视景体

Camera的另一个重要特性是视觉平截体(View Frustrum),在上面的图片中可以看到一个被砍去顶尖的角锥体,这就是一个视觉平截体,所以视觉平截体内部的东西都可以在屏幕上看到,视觉平截体被六个剖面确定,near、far、left、right、top、bottom。near剖面还有一个特殊的性质:你可以认为它是相机照出的照片中的平面。这个由3D转换为2D的过程,称为投影,通常有两种投影:orthographic projection(正交投影) 和 perspective projection(透视投影、中心投影),正交投影在2D的图形中经常使用,无论物体离camera远近,物体在屏幕上的大小始终一样。透视投影是现实世界中的成像原理:物体离眼睛越远,物体的就越小。不同的投射类型就是改变视景体的形状,透视投影的视觉平截体形状就是上图的角锥体形状,正交投影的视觉平截体是一个长方体,投影的过程简单来说:对于物体上的每一个点,计算这个点到camera之间的连线是否与near剖面相交。下图是正交投影与透视投影的图片示例:


 由此可以看出为什么在透视投影下,屏幕上越远的物体会越小,而正交投影下,物体保持大小不变。OpenGL一般使用3D空间展示,无论你是绘图还是输出字符串,当你使用正交投影时,可以假设z轴不存在,下图就是一个2D精灵在3D空间中的样子:

因此,在这里可以忽略2D和3D的不同之处,下图是一个在未设置任何矩阵(matrix)下使用SpriteBatch绘图的正交视景体


我们所做的就是让我们的精灵在x/y平面上移动,忽略z轴,保持我们一直工作在2D空间的假象

透视投影使用两个属性来定义它的投影规则:广角和纵横比(长宽比),广角是一个角度值,用来定义视场有多大”开口“,如下图

纵横比是视口的宽度和高度比。视口是一个长方形,即camera用来进行渲染图像的地方,一个480*320像素的窗口,它的纵横比是480/320。

Camera相关类

Camera:基础类

OrthographicCamera:正交Camera 继承自Camera

PerspectiveCamera:投影Camera继承在Camera

首先看Camera类

  1. public abstract class Camera {
  2. public final Vector3 position = new Vector3();
  3. public final Vector3 direction = new Vector3(0, 0, -1);
  4. public final Vector3 up = new Vector3(0, 1, 0);

三个公共成员分别是Camera的位置、方向、up向量 ,采用的是默认属性,因此camera默认位于原点,z轴默认朝上。

  1. public final Matrix4 projection = new Matrix4();
  2. public final Matrix4 view = new Matrix4();
  3. public final Matrix4 combined = new Matrix4();
  4. public final Matrix4 invProjectionView = new Matrix4();

这是一系列的在OpenGL ES 2.0中用到的矩阵,前两个是投影和模型视图矩阵,第三个是前两个的结合,第四个是第三个的翻转,经常用来做提取登时,这些都不会经常使用到

  1. public float near = 1;
  2. public float far = 100;
  3. public float viewportWidth = 0;
  4. public float viewportHeight = 0;

这是near和far剖面距离camera的距离,以及视口的宽度和高度,near和far必须遵循0<=near<far的公式。默认情况情况下near剖面距离camera是1个单位,对于正交投影,near经常设置为0,width和height用来计算透视投影的纵横比或者正交投影的视景体长方体

  1. public final Frustum frustum = new Frustum();

最后的成员是Frustum,有6个平面组成,Frustum可以用来进行裁剪:检测一个物体是否在Frustum内,如果不在则不绘制它。Frustum有一些的方法用来检测一个 盒子、球体或者一个点是否在Frustum中。

  1. public abstract void update();
  2. public void apply(GL10 gl);

这是两个常用方法,update方法用来重新计算camera的举证,当camera的属性发生改变时进行调用。apply方法用来设置camera的GL_PROJECTION或者GL_MODELVIEW矩阵,不过这个方法在OpenGL ES 2.0上不可用。

  1. public void unproject(Vector3 vec);
  2. public void project(Vector3 vec);
  3. public Ray getPickRay(float x, float y);

最后是几个你可能用到额高级方法。unproject方法用来从屏幕坐标中取得一个点,并产生这个点的三位坐标。与原有类的OrthographicCamera.screenToWorld()左右相同,也等同于gluUnproject方法。x、y坐标是可触坐标,参数中的z坐标是个0和1之间的值,0表示点位于near平面,1表示点位于far平面;

project方法正好相反,从3D世界中取得一个点,并转换成屏幕上的2d点;

getPickRay方法会返回一个直线,可以想象成3D世界中,从camera位置到x、y的一条直线,用来检测这个直线是否与其他图形或者物体接触到。

  1. public class OrthographicCamera extends Camera {
  2. public float zoom = 1;
  3. public OrthographicCamera(float viewportWidth, float viewportHeight);

OrthographicCamera有一个额外成员zoom用来定义缩放因子。构造参数定义了camera的视口width和高度,如果想要最佳的利用机器像素,只需要在这里定义GRaphics.getWidth()/getHeight(),也可以使用不同的单位,比如厘米。

  1. public class PerspectiveCamera extends Camera {
  2. public float fieldOfView = 67;
  3. public PerspectiveCamera(float fieldOfView, float viewportWidth, float viewportHeight);

PerspectiveCamera 也有一个额外的成员,广角。

文章翻译完,对Camera的理解又进了一层。

转一篇讲camera的 mb好多年不搞3d 都忘光了的更多相关文章

  1. 在知乎上看到 Web Socket这篇文章讲得确实挺好,从头看到尾都非常形象生动,一口气看完,没有半点模糊,非常不错

    在知乎上看到这篇文章讲得确实挺好,从头看到尾都非常形象生动,一口气看完,没有半点模糊,非常不错,所以推荐给大家,非常值得一读. 作者:Ovear链接:https://www.zhihu.com/que ...

  2. 对于DQN的三大改进 - 这篇讲的好些

    可以看这篇文章: https://blog.csdn.net/u013236946/article/details/73161586 这篇也讲的不错: https://www.cnblogs.com/ ...

  3. 推荐一篇讲arm架构gcc内联汇编的文章

    这是来自ethernut网站的一篇文章,原文链接: http://www.ethernut.de/en/documents/arm-inline-asm.html 另外,据说nut/os是个不错的开源 ...

  4. 觉得一篇讲SPFA还不错的文章

    我觉得他整理的有一些乱,我都改成插入代码了,看的顺眼一些 转载自http://blog.csdn.net/juststeps/article/details/8772755 下面的都是原文: 最短路径 ...

  5. 复盘一篇讲sklearn库学习文章(上)

    认识 sklearn 官网地址: https://scikit-learn.gor/stable/ 从2007年发布以来, scikit-learn已成为重要的Python机器学习库, 简称sklea ...

  6. 一篇讲Java指令重排和内存可见性的好文

    在这里: http://tech.meituan.com/java-memory-reordering.html 指令重排和内存可见性(缓存不一致)是两个不同的问题. volatile关键字太强,即阻 ...

  7. 很好的一篇讲LTP在编解码中的作用的文章

    原文链接 LONG-TERM PREDICTION by: Adit Aviv       Kfir Grichman introduction: The speech signal has been ...

  8. Android扩展 - 拍照篇(Camera)

    1.调用系统摄像头 1.声明常量和变量 2.按钮点击事件,打开系统摄像头 3.重写onActivityResult事件接收拍照返回 4.生成文件名返回路径 5.保存图片 private static  ...

  9. linux动态链接库---一篇讲尽

    一般我们在Linux下执行某些外部程序的时候可能会提示找不到共享库的错误, 比如: tmux: error while loading shared libraries: libevent-1.4.s ...

随机推荐

  1. 忘记windows的登陆密码

    http://user.qzone.qq.com/372806800/blog/1342261571

  2. Linux workqueue工作原理 【转】

    转自:http://blog.chinaunix.net/uid-21977330-id-3754719.html 转自:http://bgutech.blog.163.com/blog/static ...

  3. win10启动文件夹:

    win10启动文件夹: C:\ProgramData\Microsoft\Windows\Start Menu\Programs\StartUp

  4. [ Laravel 5.3 文档 ] 安全 ―― API认证(Passport)保障安全性。

    1.简介 Laravel通过传统的登录表单已经让用户认证变得很简单,但是API怎么办?API通常使用token进行认证并且在请求之间不维护session状态.Laravel使用LaravelPassp ...

  5. hdwiki中model模块的应用

    control中调用model原则是这样的,如果你的这个model在本control中大部分方法中都要用到,那么,就写在构造函数里面.例如,名字为doc的control的构造函数如下: functio ...

  6. sql 语法

    CASE ISNULL(b.enddate , '2000-1-1') WHEN '2000-1-1' THEN '未发稿' ELSE '已经发稿' END 如果时间为空,则显示为值‘200-1-1’ ...

  7. c# web 缓存管理

    using System; using System.Collections; using System.Text.RegularExpressions; using System.Web; usin ...

  8. c# UDP通信 列子

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  9. winform中利用反射实现泛型数据访问对象基类(1)

    考虑到软件使用在客户端,同时想简化代码的实现,就写了一个泛型的数据访问对象基类,并不是特别健全,按道理应该参数化的方式实现insert和update,暂未使用参数化,抽时间改进. /// <su ...

  10. ssh 配置自动登录

    假定 机器A 连接至 机器B . 1. 在机器A上,生成RSA秘钥对 ssh-keygen -t rsa 期间passphrase不输入密码.默认生成文件至 ~/.ssh/ -rw------- we ...