OpenGL教程翻译 第十四课 相机控制(一)

OpenGL教程翻译 第十四课 相机控制(一)

原文地址:http://ogldev.atspace.co.uk/(源码请从原文主页下载)

Background

在之前的教程中我们学习了如何在三维场景中的任何地方放置相机。那么我们下一步就应该学着去控制这个相机。相机可以向任何方向自由移动。我们可以用鼠标和键盘控制相机——鼠标控制视口方向,键盘控制我们的位置。这些都和第一人称视角相似。这一章我们主要来学习鼠标和键盘的控制。

我们仍然使用上下左右四个方向键。记住,我们的相机的变换取决于位置、target
向量和up
向量。当我们使用键盘移动时,我们只改变了我们的位置信息。上下左右的操作不会导致相机倾斜或旋转,所以target vector
和 up vector都不受影响。

控制键盘我们用到另一个GLUT API:glutSpecialFunc()。当一个“特定”键被敲击的时候,这个函数就会注册一个因之触发的回调。这些特定的键包括:功能键、方向键和PAGE-UP/PAGE-DOWN/HOME/END/INSERT键。如果你想要捕获常规键(字符或者数字),那么请使用glutKeyboardFunc()。

Code Walkthru

相机的功能封装在相机类内。这个类存储相机的属性,并且可以通过接收到的移动事件改变这些属性。管线类获取这些属性并通过它们产生变换矩阵。

(Camera.h)

class Camera

{

public:

Camera();

Camera(const Vector3f& Pos, const Vector3f& Target, const Vector3f& Up);

bool OnKeyboard(int Key);

const Vector3f& GetPos() const

const Vector3f& GetTarget() const

const Vector3f& GetUp() const

private:

Vector3f m_pos;

Vector3f m_target;

Vector3f m_up;

};

这是相机类的声明。它存储着定义相机类的三个属性——位置向量、target
vector 和up vector。有两个构造函数。默认的那个构造函数简单地将相机摆放在从原点看向Z轴正方向,up
向量指向“天空”(0,1,0)点位置。我们还可以选择创建一个具有特定属性值的相机。OnKeyboard()函数给相机类提供了键盘事件。它返回一个布尔值用来显示该事件是否由相机类接受。如果点击的按键与我们定义的响应事件相关,则返回true,否则返回false。这样,你可以建立一条客户端链用来接收键盘事件,在获取响应特定事件的第一个客户端后停止。

(Camera.cpp:42)

bool Camera::OnKeyboard(int Key)

{

bool Ret = false;

switch (Key) {

case GLUT_KEY_UP:

{

m_pos += (m_target * StepSize);

Ret = true;

}

break;

case GLUT_KEY_DOWN:

{

m_pos -= (m_target * StepSize);

Ret = true;

}

break;

case GLUT_KEY_LEFT:

{

Vector3f Left = m_target.Cross(m_up);

Left.Normalize();

Left *= StepSize;

m_pos += Left;

Ret = true;

}

break;

case GLUT_KEY_RIGHT:

{

Vector3f Right = m_up.Cross(m_target);

Right.Normalize();

Right *= StepSize;

m_pos += Right;

Ret = true;

}

break;

}

return Ret;

}

这个函数根据键盘事件移动相机。GLUT定义了与方向键相应的宏命令,而上面的switch语句正是基于此。不幸的是,这些宏定义的类型都是只是“int”类型,而不是枚举类型。

向前向后的移动是最简单的。因为移动总是沿着target vector,我们就只需要让当前位置加上或者减去target vector就可以了。target
vector本身不改变。注意,在加上或者减去target vector之前,我们用一个叫做“StepSize”的常数来缩放它。对所有的方向键,我们都做如此处理。StepSize提供了一个改变速度的中心点(在后面我们也许将会把这个值添加到类属性中)。为使步长一致,我们要确保我们总是乘上单位长度的向量(即我们必须确保目标和up向量是单位长度的)。

向侧面移动有些复杂。这是一种沿着垂直于target vectors和
up vectors所确定的平面的矢量的移动。这个平面将三维空间分成了两个部分,并且有两个向量垂直于它,这两个向量方向相反。我们称其中的一个向量为“left”,另一个为“right”。这两个向量是由target
vectors和 up vectors的叉积的两种可能的组合分别产生的——target vectors×up
vectors和up target vectors×target 
vectors(叉积运算没有交换律——也就是说,在叉积运算中,改变参数的顺序会得到不同的结果)。得到left/right向量之后,我们用StepSize常量将其规范化,用StepSize放缩,最后添加到position(可以将相机向左右方向移动)上。此外,target
vectors和up vectors没有受到影响。

注意,这个函数的内部操作里使用了一些新的运算符,比如被添加到Vector3f
类中的“+=”和“-=”。

   (tutorial14.cpp:73)

static void SpecialKeyboardCB(int Key, int x, int y)

{

GameCamera.OnKeyboard(Key);

}

static void InitializeGlutCallbacks()

{

glutDisplayFunc(RenderSceneCB);

glutIdleFunc(RenderSceneCB);

glutSpecialFunc(SpecialKeyboardCB);

}

这里,我们注册一个新的回调函数来处理特殊的键盘事件。在按键被触发的时候,回调函数接收按键和鼠标的位置信息。我们忽略鼠标的位置,并且把这个事件传递给成相机类的一个实例,这个实例早已分配给该文件的全局部分。

(tutorial14.cpp:55)

p.SetCamera(GameCamera.GetPos(), GameCamera.GetTarget(), GameCamera.GetUp());

以前,我们在管线类中用硬编码的向量初始化相机参数。现在,我们不再使用这些向量,而是从Camera类直接获取相机属性。

时间: 2024-12-21 20:38:32

OpenGL教程翻译 第十四课 相机控制(一)的相关文章

OpenGL教程翻译 第十五课 相机控制(二)

OpenGL教程翻译 第十五课 相机控制(二) 原文地址:http://ogldev.atspace.co.uk/(源码请从原文主页下载) Background 在这一节中我们将使用鼠标来控制相机的方向,从而得我们的相机控制更加完善.相机有不同的自由程度,这与其设计有关.在本教程中我们将要实现的是与第一人称游戏中相似的相机控制(如枪战类游戏).这意味着我们将可以使相机完成360度的旋转(绕着Y轴),这与我们的头部向左转向右转.身体转一整圈类似.除此之外我们也能使相机向上或者向下倾斜以获得更好的向

OpenGL教程翻译 第十六课 基本的纹理贴图

OpenGL教程翻译 第十六课 基本的纹理贴图 原文地址:http://ogldev.atspace.co.uk/(源码请从原文主页下载) Background 纹理贴图就是将任意一种类型的图片应用到3D模型的一个或多个面.图片(也可以称之为纹理)内容可以是任何东西,但是他们一般都是一些比如砖,叶子,地面等的图案,纹理贴图增加了场景的真实性.例如,对比下面的两幅图片. 为了进行纹理贴图,你需要进行三个步骤:将图片加载到OpenGl中,定义模型顶点的纹理坐标(以对其进行贴图),用纹理坐标对图片进行

OpenGL教程翻译 第十八课 漫反射光(Diffuse Lighting)

OpenGL教程翻译 第十七课 环境光(Ambient Lighting) 原文地址:http://ogldev.atspace.co.uk/(源码请从原文主页下载) Background 环境光和漫反射光的主要不同是,漫反射光的计算需要依靠光线方向而环境光完全忽略了它!当只有环境光时整个场景被均等照亮.漫反射光会使物体面对光的部分比背对光的部分更加明亮. 此外漫反射光还增加了一点新的计算,光线的入射角决定了表面的亮度.通过下面的图片来演示这个概念: 让我们假设两条光线的强度是一样的,而唯一不一

NeHe OpenGL教程 第三十八课:资源文件

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第三十八课:资源文件 从资源文件中载入图像: 如何把图像数据保存到*.exe程序中,使用Windows的资源文件吧,它既简单又实用. 欢迎来到NeHe教程第38课.离上节课的写作已经有些时日了,加上写了一整天的code,也许笔头已经

NeHe OpenGL教程 第三十五课:播放AVI

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第三十五课:播放AVI 在OpenGL中播放AVI: 在OpenGL中如何播放AVI呢?利用Windows的API把每一帧作为纹理绑定到OpenGL中,虽然很慢,但它的效果不错.你可以试试. 首先我得说我非常喜欢这一章节.Jonat

NeHe OpenGL教程 第三十六课:从渲染到纹理

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第三十六课:从渲染到纹理 放射模糊和渲染到纹理: 如何实现放射状的滤镜效果呢,看上去很难,其实很简单.把渲染得图像作为纹理提取出来,在利用OpenGL本身自带的纹理过滤,就能实现这种效果,不信,你试试. 嗨,我是Dario Corn

NeHe OpenGL教程 第三十九课:物理模拟

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第三十九课:物理模拟 物理模拟简介: 还记得高中的物理吧,直线运动,自由落体运动,弹簧.在这一课里,我们将创造这一切. 物理模拟介绍 如果你很熟悉物理规律,并且想实现它,这篇文章很适合你. 在这篇教程里,你会创建一个非常简单的物理引

NeHe OpenGL教程 第四十四课:3D光晕

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第四十四课:3D光晕 3D 光晕 当镜头对准太阳的时候就会出现这种效果,模拟它非常的简单,一点数学和纹理贴图就够了.好好看看吧. 大家好,欢迎来到新的一课,在这一课中我们将扩展glCamera类,来实现镜头光晕的效果.在日常生活中,

NeHe OpenGL教程 第十四课:图形字体

转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线教程的编写,以及yarn的翻译整理表示感谢. NeHe OpenGL第十四课:图形字体 图形字体: 在一课我们将教你绘制3D的图形字体,它们可像一般的3D模型一样被变换. 这节课继续上一节课课的内容.在第13课我们学习了如何使用位图字体,这节课,我们将学习如何使用轮廓字体. 创建轮廓字体的方法类似于