OpenGL基本了解(十一) 投影变换及坐标系统

投影变换及坐标系统 官方说明数据   https://www.opengl.org/wiki/Main_Page

在OPENGL中采用的是三维笛卡尔坐标系。

PC版本OpenGl支持多种多边形,但是在OpenGL ES下只支持三角形。

也就是说,所有图形都由三角形构建,类似积分的原理。

在计算机图形学中有两种类型的投影方式:透视投影与正交投影。采用透视投影,物体越远越小,这样更接具真实性。正交投影往往用于2D绘制

投影变换 在OpenGL中,如果想对模型进行操作,就要对这个模型的状态(当前的矩阵)乘上这个操作对应的一个矩阵.

如果乘以变换矩阵(平移, 缩放, 旋转), 那相乘之后, 模型的位置被变换;

如果乘以投影矩阵(将3D物体投影到2D平面), 相乘后, 模型的投影方式被设置;

如果乘以纹理矩阵(), 模型的纹理方式被设置. 而用来指定乘以什么类型的矩阵, 就是glMatriMode(GLenum mode);

glMatrixMode有3种模式: GL_PROJECTION 投影, GL_MODELVIEW 模型视图, GL_TEXTURE 纹理. 所以,在操作投影矩阵以前,需要调用函数:

glMatrixMode(GL_PROJECTION); //将当前矩阵指定为投影矩阵

然后把矩阵设为单位矩阵:

glLoadIdentity();

然后调用glFrustum()或gluPerspective(),它们生成的矩阵会与当前的矩阵相乘,生成透视的效果;

有两种投影(透视投影及正射投影)。投影要在创建的时候设置。

透视投影 一个函数:

void glFrustum(GLdouble left,GLdouble Right,GLdouble bottom,GLdouble top,GLdouble near,GLdouble far);

Ø  前四个参数是摄像头视口。正常情况是成比例的。

Ø  第五个是近平面:可以认为近平面就对应着屏幕视口,不过近平面是Y向上,屏幕视口是Y向下。

Ø  第六个是远平面:远平面的一个作用是做裁切用。只有在近平面,元平面之间的物体才被显示,超过的就裁切掉。

创建一个透视型的视景体。其操作是创建一个透视投影的矩阵,并且用这个矩阵乘以当前矩阵。这个函数的参数只定义近裁剪平面的左下角点和右上角点的三维空间 坐标,即(left,bottom,-near)和(right,top,-near);最后一个参数far是远裁剪平面的离视点的距离值,其左下角点和 右上角点空间坐标由函数根据透视投影原理自动生成。near和far表示离视点的远近,它们总为正值(near/far 必须>0)。

源代码: https://www.opengl.org/wiki/GluPerspective_code

void glhFrustumf2(float *matrix, float left, float right, float bottom, float top,
                  float znear, float zfar)
{
    float temp, temp2, temp3, temp4;
    temp = 2.0 * znear;
    temp2 = right - left;
    temp3 = top - bottom;
    temp4 = zfar - znear;
    matrix[0] = temp / temp2;
    matrix[1] = 0.0;
    matrix[2] = 0.0;
    matrix[3] = 0.0;
    matrix[4] = 0.0;
    matrix[5] = temp / temp3;
    matrix[6] = 0.0;
    matrix[7] = 0.0;
    matrix[8] = (right + left) / temp2;
    matrix[9] = (top + bottom) / temp3;
    matrix[10] = (-zfar - znear) / temp4;
    matrix[11] = -1.0;
    matrix[12] = 0.0;
    matrix[13] = 0.0;
    matrix[14] = (-temp * zfar) / temp4;
    matrix[15] = 0.0;
}

另一个函数:

void gluPerspective(GLdouble fovy,GLdouble aspect,GLdouble zNear, GLdouble zFar);

Ø  参数fovy定义视野在X-Z平面的角度,范围是[0.0, 180.0];

Ø  参数aspect是投影平面宽度与高度的比率;

Ø  参数zNear和Far分别是远近裁剪面沿Z负轴到视点的距离,它们总为正值。

OpenGL的视角设定可以直接调用gluPerspective函数,但是OpenGL ES版本并不支持这个用法。ES的版本中只能支持glFrustum的函数调用。

创建一个对称的透视型视景体,但它的参数定义于前面的不同,如图。其操作是
创建一个对称的透视投影矩阵,并且用这个矩阵乘以当前矩阵。参数fovy定义视 野在Y-Z平面的角度,范围是[0.0, 180.0];参数aspect是投影平面宽度与高度的比率;参数Near和Far分别是近远裁剪面到视点(沿Z负轴)的距离,它们总为正值。

  以上两个函数缺省时,视点都在原点,视线沿Z轴指向负方向。

void gluPerspective(
  GLdouble fovy, //角度
  GLdouble aspect,//视景体的宽高比
  GLdouble zNear,//沿z轴方向的两裁面之间的距离的近处
  GLdouble zFar //沿z轴方向的两裁面之间的距离的远处
  );

在OPENGL中我们可以使用gluPerspective来设置视椎体。PC版的gluPerspective在OPENGL3.0以后就好像不见了。

但是在OpenGLES中却没有提供这样的实用库支持,其实我们可以自己来完成这个函数的功能。代码如下:

//matrix will receive the calculated perspective matrix.
//You would have to upload to your shader
// or use glLoadMatrixf if you aren't using shaders.
void glhPerspectivef2(float *matrix, float fovyInDegrees, float aspectRatio,
                      float znear, float zfar)
{
    float ymax, xmax;
    float temp, temp2, temp3, temp4;
    ymax = znear * tanf(fovyInDegrees * M_PI / 360.0);
    //ymin = -ymax;
    //xmin = -ymax * aspectRatio;
    xmax = ymax * aspectRatio;
    glhFrustumf2(matrix, -xmax, xmax, -ymax, ymax, znear, zfar);
}

正射投影 正射投影的最大一个特点是无论物体距离相机多远,投影后的物体大小尺寸不变。

函数:

void glOrtho(GLdouble left,GLdouble right,GLdouble bottom,GLdouble top, GLdouble near,GLdouble far)

六个参数, 前两个是x轴最小坐标和最大坐标,中间两个是y轴,最后两个是z轴值

它创建一个平行视景体(就是一个长方体空间区域)。

实际上这个函数的操作是创建一个正射投影矩阵,并且用这个矩阵乘以当前矩阵。

其中近裁剪平面是一个矩形,矩形左下角点三维空间坐标是(left,bottom,-near),

右上角点是(right,top,-near);远裁剪平面也是一个矩形,左下角点空间坐标是(left,bottom,-far),右上角点是(right,top,-far)。

注意,所有的near和far值同时为正或同时为负, 值不能相同。如果没有其他变换,正射投影的方向平行于Z轴,且视点朝向Z负轴。这意味着物体在视点前面时far和near都为负值,物体在视点后面时far和near都为正值。

只有在视景体里的物体才能显示出来。

如果最后两个值是(0,0),也就是near和far值相同了,视景体深度没有了,整个视景体都被压成个平面了,就会显示不正确。

设置视图变换(Setting the View Transform) 设置视图矩阵最简单的方法就是用LookAt方法,它并不是OpenGL ES的内置函数,但是可以自已快速实现。它有三个参数:相机位置,目标位置,一个”up”向量表示相机朝向>

通过三个向量的传入,LookAt就可以生成一个变换矩阵,否则就得用基本的变换(缩放,移动,旋转)来生成。

一个参考的 LookAt函数

mat4 LookAt(const vec3& eye, const vec3& target, const vec3& up)

{

vec3 z = (eye – target).Normalized(); vec3 x = up.Cross(z).Normalized(); vec3 y = z.Cross(x).Normalized();

mat4 m;

m.x = vec4(x, 0);

m.y = vec4(y, 0);

m.z = vec4(z, 0);

m.w = vec4(0, 0, 0, 1);

vec4 eyePrime = m * -eye; m = m.Transposed();

m.w = eyePrime;

return m;

}

一旦设置了设影矩阵, 就设定了视野。视锥表示眼在金字塔顶部的一个锥体。

基于金字塔的顶点(称为视野)的角度,可以计算一个视锥。开发者认为这样比指定六面更加直观。示例2.2中方法有四个参数:视角,金字塔宽高比,远与近裁剪面。

示例方法有四个参数:视角,金字塔宽高比,远与近裁剪面。

void VerticalFieldOfView(float degrees, float aspectRatio, float near, float far)

{

     float top = near * std::tan(degrees * Pi / 360.0f);

     float bottom = -top;

     float left = bottom * aspectRatio;

     float right = top * aspectRatio;

    

     glFrustum(left, right, bottom, top, near, far);

}

转自:http://iteches.com/archives/44016

时间: 2024-09-20 06:03:55

OpenGL基本了解(十一) 投影变换及坐标系统的相关文章

Swift游戏开发之俄罗斯方块:No.2 准备工作

如果你已经成功建好了工程,运行之后你会发现是个小飞机的程序,如果我没记错的: 这个程序是spin-the-bottle:Space Edition,但是可惜的是,我们并不需要这些东西,我们需要一个干净的模板,所以,我们需要首先清理下战场,然后把我们需要用到的一些资源放进来. 打开项目导航栏,如下图中圆圈内的图标,或者点击⌘ + 1: 右键点击GameScene.sks,选择delete 当系统询问的时候,选择move to trash 接下来点击image.xcassets,然后删除spaces

基于VC++的OpenGL编程讲座之坐标变换

OpenGL通过相机模拟.可以实现计算机图形学中最基本的三维变换,即几何变换.投影变换.裁剪变换.视口变换等,同时,OpenGL还实现了矩阵堆栈等.理解掌握了有关坐标变换的内容,就算真正走进了精彩地三维世界. 一.OpenGL中的三维物体的显示 (一)坐标系统 在现实世界中,所有的物体都具有三维特征,但计算机本身只能处理数字,显示二维的图形,将三维物体及二维数据联系在一起的唯一纽带就是坐标. 为了使被显示的三维物体数字化,要在被显示的物体所在的空间中定义一个坐标系.这个坐标系的长度单位和坐标轴的

076_《OpenGL图形程序开发实务》

<OpenGL图形程序开发实务> Delphi 教程 系列书籍 (076) <OpenGL图形程序开发实务> 网友(邦)整理 EMail: shuaihj@163.com 下载地址: Pdf 作者: 薛惠锋 吴慧欣 解丹蕊 出版社:西北工业大学出版社 ISBN:7561219393 上架时间:2006-9-6 出版日期:2005 年8月 开本:16开 页码:293 版次:1-1 内容简介 OpenGL作为一种图形工业标准,在实际工作中应用广泛并占有非常重要的地位.本书采用在Delp

OpenGL入门介绍

1.OpenGL简介 OpenGL是近几年发展起来的一个性能卓越的三维图形标准,它是在SGI等多家世界闻名的计算机公司的倡导下,以SGI的GL三维图形库为基础制定的一个通用共享的开放式三维图形标准.目前,包括Microsoft.SGI.IBM.DEC.SUN.HP等大公司都采用了OpenGL做为三维图形标准,许多软件厂商也纷纷以OpenGL为基础开发出自己的产品,其中比较著名的产品包括动画制作软件Soft Image和3D Studio MAX.仿真软件Open Inventor.VR软件Wor

浅述Delphi下的OpenGL图形开发

OpenGL最初是由Silicon图形公司开发的底层图形库规范.你的系统中准确实现这个规范的部分,通常被称为OpenGL驱动,它允许你使用几何集合(点,线,多边形,图像等等)来描述你希望表现的场景.让肉眼观察起来较为舒适的中等规模场景,通常在毫秒级的速度上实现,这意味着该库文件有足够的能力来支持你创建一个生机勃勃的虚拟世界. OpenGL驱动一般以二进制库文件的形式提供.它能够动态的连接到你的程序中.在Windows平台上,它将是成为DLL的形式(在你的系统目录下检查opengl.dll).自从

OpenGL开发库的介绍[转载]

开发基于 OpenGL 的应用程序,必须先了解 OpenGL 的库函数.它采用 C 语言风格,提供大量的函数来进行图形的处理和显示. OpenGL 库函数的命名方式非常有规律.所有 OpenGL 函数采用了以下格式:  < 库前缀 >< 根命令 >< 可选的参数个数 >< 可选的参数类型 >  库前缀有 gl . glu . aux . glut . wgl . glx . agl 等等,分别表示该函数属于 OpenGL 那个开发库,从函数名后面中还可以看出

从零开始学习OpenGL ES之六 – 纹理及纹理映射

转自:http://bbs.ityran.com/article-9-1.html [转载]从零开始学习OpenGL ES之六 – 纹理及纹理映射 2011-9-14 23:49| 发布者: Iven| 查看: 9206| 评论: 15 摘要: 图形图像, 编程,编程, OpenGL ES, 教程,OpenGL ES 3D 在OpenGL ES中另一种为多边形定义颜色创建材质的方法是将纹理映射到多边形.这是一种很实用的方法,它可以产生很漂亮的外观并节省大量的处理器时间.比如说,你想在游戏中造一个

(译)OpenGL ES2.0 – Iphone开发指引

纯的不能再纯的转载,原译者说别人是牛人,不过在我看来,他也挺牛的了. 刚接触OpenGL ES时,这篇文章不知看了多少遍了,只能照本宣科,不知其然与其所以然. 后决定放弃这种不负责任的工作方式,虽然不是我的方式,但我又必须按要求做事! 按自已的想法,找到罗朝辉博客上最全面的那篇,把例子下载下来,准备花长些时间,耐下心来,一点一点地看. 到目前,从几何顶点,到灯光,到着色器,再到模型加载,最后是纹理,中间回到几何顶点几次,轻轻地,仅是轻轻地扒了一小层皮儿而已,现在再回头看这篇,感觉明白了许多. 不

《OpenGL编程指南(原书第9版)》——1.5 第一个程序:深入分析

1.5 第一个程序:深入分析 现在我们来深入探讨一下之前的第一个程序.1.5.1 进入main()函数 为了了解示例程序从一开始是如何运行的,首先了解一下main()函数当中都发生了什么.前面的6行使用GLFW设置和打开了一个渲染用的窗口.这方面的详细介绍可以参见附录A,这里只介绍每一行的执行结果. 第一个函数glfwtInit()负责初始化GLFW库.它会处理向程序输入的命令行参数,并且移除其中与控制GLFW如何操作相关的部分(例如设置窗口的大小).glfwtInit()必须是应用程序调用的第