OpenGL坐标系之间的转换 http://blog.csdn.net/sac761/article/details/52179585

时间:2022-09-06 22:51:05

1. OpenGL 渲染管线

OpenGL渲染管线分为两大部分,模型观测变换(ModelView Transformation)投影变换(Projection Transformation)。做个比喻,计算机图形开发就像我们照相一样,目的就是把真实的场景在一张照相纸上表现出来。那么观测变换的过程就像是我们摆设相机的位置,选择好要照的物体,摆好物体的造型。而投影变换就像相机把真实的三维场景显示在相纸上一样。下面就分别详细的讲一下这两个过程。

1.1模型观测变换

让我们先来弄清楚OpenGL中的渲染管线。管线是一个抽象的概念,之所以称之为管线是因为显卡在处理数据的时候是按照一个固定的顺序来的,而且严格按照这个顺序。就像水从一根管子的一端流到另一端,这个顺序是不能打破的。先来看看下面的图1:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585

  图1 OPENGL渲染管线

图中显示了OpenGL图形管线的主要部分,也是我们在进行图形编程的时候常常要用到的部分。一个顶点数据从图的左上角(MC)进入管线,最后从图的右下角(DC)输出MC是Model Coordinate的简写,表示模型坐标DCDevice Coordinate的简写,表示设备坐标。当然DC有很多了,什么显示器,打印机等等。这里DC我们就理解成常说的屏幕坐标好了。MC当然就是3D坐标了(注意:我说的3D坐标,而不是世界坐标),这个3D坐标就是模型坐标,也说成本地坐标(相对于世界坐标)。MC要经过模型变换(Modeling Transformation)才变换到世界坐标,图2:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图2 世界坐标系和模型坐标系

变换到世界坐标WC(World Coordinate)说简单点就是如何用世界坐标系来表示本地坐标系中的坐标。为了讲得更清楚一些,这里举个2D的例子。如图3:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585

 

图3 世界坐标系和模型坐标系的计算

图中红色坐标系是世界坐标系WC,绿色的是模型坐标系MC。现在有一个顶点,在模型坐标系中的坐标为(1,1),现在要把这个模型坐标转换到世界坐标中来表示。从图中可以看出,点(1,1)在世界坐标系中的坐标为(3,4),现在我们来通过计算得到我们希望的结果。首先我们要把模型坐标系MC在世界坐标系中表示出来,使用齐次坐标(Homogeneous Coordinate )可以表示为矩阵(注意,本教程中使用的矩阵都是以列向量组成):OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585其中,矩阵的第一列为MC中x轴在WC中的向量表示第二列为MC中y轴WC中的向量表示第三列为MC中的原点在WC中的坐标。对齐次坐标系不了解的同学,请先学习游戏数学方面的知识。有了这个模型变换矩阵后,用这个矩阵乘以在MC中表示的坐标就可以得到该坐标在世界坐标系中的坐标。所以该矩阵和MC中的坐标(1,1)相乘有:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585这也正是我们需要的结果。现在让我们把相机坐标也加进去,相机坐标也称为观测坐标(View Coordinate),如图4和图5。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图4 ModelView变换的三个坐标系

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图5 ModelView变换计算

来看看MC坐标中的点(1,1)如何在相机坐标中表示。从图5中可以直接看出MC中的点(1,1)在相机坐标系VC中为(-2,-2)。和上面同样的道理,我们可以写出相机坐标系VC在世界标系WC中可以表示为:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585那么世界坐标系中的点转换为相机坐标系中的点我们就需求VC的逆矩阵:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585那么世界坐标系WC中的点(3,4)在相机坐标系VC中坐标为:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585上面的变换过程,就是可以把模型坐标变换为相机坐标。在OpenGL中,当我们申明顶点的时候,有时候说的是世界坐标,这是因为初始化的时候世界坐标系、模型坐标系和相机坐标系是一样的,重合在一起的。所以OpenGL中提供了模型观测变换,它是把模型坐标系直接转换为相机坐标系,如图4。现在我们已经计算得到了VC-1和MC,如果把VC-1和MC相乘,就可以得到模型坐标在相机坐标中的表示。为了得到模型坐标系中的坐标在相机坐标系中的表示,这就是OpenGL中的ModelView变换矩阵。这也是ModelView变换的名字的由来,它是通过了上面两个步骤得到的。那么这里,ModelView变换矩阵M为:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585现在只要用上面的模型观测矩阵M乘以模型坐标系MC中的坐标就可以得到相机坐标系中的坐标了。模型观测变换的关键就是要得到相机坐标系中的坐标,因为光照等计算都是在这个这个坐标系中完成的。下面我们实际OpenGL程序中检查一下。在程序中,为了计算方便,我们使用图6中的模型。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图6 ModelView变换计算模型

根据图中的数据,我们分别可以写出对应MC和VC-1,从而求得观测变换矩阵M

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585现在程序中用glGetFloatv()这个函数来获得当前矩阵数据来检查一下。

  1. float m[16] = {0}; //用来保存当前矩阵数据
  2. glMatrixMode(GL_MODELVIEW);
  3. glLoadIdentity();
  4. glGetFloatv(GL_MODELVIEW_MATRIX, m);
  5. //相机设置,View 变换
  6. gluLookAt(0.0, 0.0, 5.0,
  7. 0.0, 0.0, 0.0,
  8. 0.0, 1.0, 0.0);
  9. glGetFloatv(GL_MODELVIEW_MATRIX, m);
  10. //投影设置
  11. glMatrixMode(GL_PROJECTION);
  12. glLoadIdentity();
  13. glOrtho(-10,10,-10,10,-10,10);
  14. glMatrixMode(GL_MODELVIEW);
  15. //Modeling变换
  16. glTranslatef(0, 0, -3);
  17. glGetFloatv(GL_MODELVIEW_MATRIX, m);
  18. glBegin(GL_POINTS);
  19. glVertex3f(1,1,0);
  20. glEnd();

如果在上面程序段中最后一个glGetFloatv(GL_MODELVIEW_MATRIX, m)处设定断点的话,就可以看到图7所显示的数据。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图7 ModelView变换矩阵数据

到这里,整个ModelView变换就完成了。通过ModelView变换后得到是相机坐标系内的坐标。在这个坐标系内典型的计算就是法线了。现在再来看看后面一个阶段。

//////////////////////////////////////////////////////////////

我的理解:ModelView 变换矩阵,就是完成从模型坐标到View坐标的转换,是坐标系之间的大变换。注意,modelview 既有model,也有view。不只是一个model的矩阵。

只是对model的进行平移或旋转的函数为  glTranslatef等函数,称作模型变换!它的坐标是基于模型本身的,即位于模型坐标系类,比如glTranslatef(0, 0, -3);的3个坐标值。

只是针对view进行设置的函数为  gluLookAt,它的坐标系是view坐标系,比如

  1. gluLookAt(0.0, 0.0, 5.0,
  2. 0.0, 0.0, 0.0,
  3. 0.0, 1.0, 0.0);

它里面的坐标的原点位于相机坐标系的原点。参看下面的投影变换。

/////////////////////////////////////////////////////////////////

1.2投影变换

先还是复习一下OpenGL的渲染管线。图1中可以看到,在投影变换(Projection Transformation)中也分为两个部分,第一个部分是将上个阶段得到的坐标转换为平面坐标,第二个部分是将转换后的平面坐标在进行归一化并进行剪裁。一般地,将三维坐标转换为平面坐标有两种投影方式:正交投影(Orthogonal Projection)和透视投影(Perspective Projection)

1.2.1 正交投影

正交投影很简单,如图8,对于三维空间中的坐标点和一个二维平面,要在对应的平面上投影,只需将非该平面上的点的坐标分量改为该平面上的坐标值,其余坐标不变。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图8 正交投影

比如将点(1,1,5)正交投影到z=0的平面上,那么投影后的坐标为(1,1,0)。在openGL中,设置正交投影可以使用函数:

  1. glOrtho (GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar)

该函数可以设置正交投影的投影空间,在该空间以外的坐标点就不会被投影到投影平面上。函数中的六个参数分是投影空间六个平面,如图9:

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585

图9 OpenGL正交投影空间和投影变换

在图9中,大的投影空间是根据这六个参数设置的投影空间,OpenGL会自动将该空间归一化,也就是将该空间或立方体转化为变长为1的正六面体投影空间,并且该证六面体的中心在相机坐标系的原点。一旦设置使用glortho函数设置投影空间,OpenGL会生成投影矩阵。这个矩阵的作用就是将坐标进行正交投影并且将投影后的坐标正规化(转换到-1到1之间)。要注意的是,生成该矩阵的时候,OpenGL会把右手坐标系转换为左手坐标系。原因很简单,右手坐标系的Z轴向平面外的,这样不符合我们的习惯。该矩阵的矩阵推导这里就不详细说明了,不了解的同学可以参考游戏数学方面资料,这里只给出正交投影矩阵。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585这个矩阵看来很复杂,其实计算很简单。举个例子,现在设置了这样的正交投影空间glOrtho(-10,10,-10,10,-10,10),这是个正六面体空间,变长为10。把这些参数带入上面的矩阵可以得到

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585现在还是在OpenGL程序中来检查一下。在OpenGL程序中添加下面代码段:

  1. //投影设置
  2. glMatrixMode(GL_PROJECTION);
  3. glLoadIdentity();
  4. glOrtho(-10,10,-10,10,-10,10);
  5. glMatrixMode(GL_MODELVIEW);
  6. glGetFloatv(GL_PROJECTION_MATRIX,m)
glGetFloatv(GL_PROJECTION_MATRIX,m)处设定断点就可以看到图10中所显示的信息。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图10 正交变换矩阵数据 

1.2.2透视投影

透视投影和正交投影最大的区别就是透视投影具有远近感。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585

图11 透视投影

透视投影采用了图11中的模型,这样的模型就是保证远的物体看起来小,近的物体看起来大。 在OpenGL中设置透视投影可以使用函数:

  1. void APIENTRY gluPerspective (GLdouble fovy, GLdouble aspect, GLdouble zNear, GLdouble zFar);

该函数也会根据给定的参数生成一个投影空间。如图11中,该投影空间是一个截头体。同样地,OpenGL会自动生成透视投影矩阵,该矩阵也会让3D坐标投影在投影平面上,并且将投影后的坐标也进行正规化。下面也直接给出OpenGL中使用的透视投影矩阵。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585下面在OpenGL中添加下面代码段:

  1. //投影设置
  2. glMatrixMode(GL_PROJECTION);
  3. glLoadIdentity();
  4. gluPerspective(45, 1.0, 1.0, 100);
  5. glMatrixMode(GL_MODELVIEW);
  6. glGetFloatv(GL_PROJECTION_MATRIX,m)

设置断点后,我们可以看到图12中显示的数据。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图12
透视变换矩阵数据

到此为止,整个投影变换就完成了。透过投影变换后得到的是正规化的投影平面坐标。这为下一个阶段的视口变换(View port Transformation)做好了准备。

1.3视口变换

现在到了最后一个阶段了。这个阶段叫做视口变换,它把上个阶段得到的正规化的投影坐标转化为windows 窗口坐标。视口变换会将投影平面上的画面映射到窗口上。在OpenGL中可以使用函数

  1. GLAPI void GLAPIENTRY glViewport (GLint x, GLint y, GLsizei width, GLsizei height);

来进行对窗口的映射,如图13。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585
图13 视口变换glViewport(width/2, 0, width/2, height/2)

举个例子说明,比如上个阶段中得到了一个顶点的坐标为(0,0,0.5,1),根据这个坐标,该顶点位于投影平面的正中间。如果将该点映射到大小为50*50的窗口上时,那么它应该位于屏幕的中间,坐标为(25,25,

0.5,1)。当然这里深度值0.5是不会改变的。有的同学肯定有疑问了,既然投影到了窗口上,那么还要深度值0.5干什么?这里要注意的是,虽然在窗口上显示时只需要x,y坐标就够了,但是要在2D窗口上显示3D图形时深度值是不可少的。这里的深度值不是用于显示,而是用于在光栅化的时候进行深度测试。

OpenGL也会根据glViewport函数提供的参数值生成一个视口变换矩阵

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585该矩阵把上个阶段得到的正规化坐标映射到窗口上,并且将正规化坐标中的深度值在转换到0到1之间。所以在深度缓冲中最大值为1,最小值为0。视口变换结束后,OpenGL中主要的图形管线阶段就算完成了,后面就是光栅化等等操作。再来回顾一下图1,现在相信大家对这个渲染管线有了一定的认识了,也明白了每一个阶段对应的变换矩阵以及如何进行坐标之间的转换的。

2. 屏幕坐标转换为世界坐标

通过前面的教程,以及现在大家对OpenGL整个渲染管线理解后,现在要将屏幕上一点坐标转换为世界坐标就比较容易了。从图形管线的开始到结束,一个模型坐标系中的坐标被转化为了屏幕坐标,那么现在把整个过程倒过来的话,屏幕上一点坐标也可以转为为世界坐标。只要在对应的阶段求得对应变换矩阵的逆矩阵,就可以得到前一个阶段的坐标。这整个过程可以用图14表示。

OpenGL坐标系之间的转换   http://blog.csdn.net/sac761/article/details/52179585图14屏幕坐标转换为世界坐标

图中显示的过程完全就是OpenGL渲染管线的逆过程,通过这个过程,屏幕上的点就可以转化为世界坐标系中的点了。可能又有的同学要问,当鼠标点击屏幕上一点的时候并没有深度信息,转换的时候要怎么办呢?这个时候可以使用OpenGL函数

  1. void glReadPixels (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLvoid *pixels);

该函数能够获得屏幕上一点对应像素的深度信息。有了这个深度信息,就可以利用上面过程把屏幕上一点转换为世界坐标了。在OpenGL中,上面的过程其实已经有现成的函数可以使用,那就是

  1. int APIENTRY gluUnProject (
  2. GLdouble  winx, GLdouble  winy,
  3. GLdouble  winz,
  4. const GLdouble modelMatrix[16],
  5. const GLdouble projMatrix[16],
  6. const GLint    viewport[4],
  7. GLdouble  *objx,  GLdouble  *objy,
  8. GLdouble       *objz);

该函数直接将屏幕上一点转换对应的世界坐标,该函数的内部实现其实还是上面的那么逆过程。下面给出利用该函数获取世界坐标的代码段。

  1. GVector screen2world(int x, int y)
  2. {
  3. GLint viewport[4];
  4. GLdouble modelview[16];
  5. GLdouble projection[16];
  6. GLfloat winX, winY, winZ;
  7. GLdouble posX, posY, posZ;
  8. glGetDoublev(GL_MODELVIEW_MATRIX, modelview);
  9. glGetDoublev(GL_PROJECTION_MATRIX, projection);
  10. glGetIntegerv(GL_VIEWPORT, viewport);
  11. winX = (float)x;
  12. winY = (float)viewport[3] - (float)y;
  13. glReadPixels(x, int(winY), 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT, &winZ);
  14. gluUnProject(winX, winY, winZ, modelview, projection, viewport, &posX, &posY, &posZ);
  15. GVector v(4, posX, posY, posZ, 1.0);
  16. return v;
  17. }
代码中函数返回类型GVector是用户定义的向量类,返回的是齐次坐标。

OpenGL坐标系之间的转换 http://blog.csdn.net/sac761/article/details/52179585的更多相关文章

  1. [转帖] select、poll、epoll之间的区别总结[整理] + 知乎大神解答 https://blog.csdn.net/qq546770908/article/details/53082870 不过图都裂了.

    select.poll.epoll之间的区别总结[整理] + 知乎大神解答 2016年11月08日 15:37:15 阅读数:2569 http://www.cnblogs.com/Anker/p/3 ...

  2. http://blog.csdn.net/pizi0475/article/details/48286579 -------------(Collada 快速入门)

    http://blog.csdn.net/zhouhangjay/article/details/8469085 说明:Collada的文件格式,中文版的很少,在csdn上看到了一个Sleepy的,感 ...

  3. 使用C++扩展Python的功能 转自:http://blog.csdn.net/magictong/article/details/8897568#comments

    使用C++扩展Python的功能 环境 VS2005Python2.5.4 Windows7(32位) 简介 长话短说,这里说的扩展Python功能与直接用其它语言写一个动态链接库,然后让Python ...

  4. 转:Java面试题集(51-70) http://blog.csdn.net/jackfrued/article/details/17403101

    Java面试题集(51-70) Java程序员面试题集(51-70) http://blog.csdn.net/jackfrued/article/details/17403101 摘要:这一部分主要 ...

  5. 学习mongoDB的一些感受(转自:http://blog.csdn.net/liusong0605/article/details/11581019)

    曾经使用过mongoDB来保存文件,最一开始,只是想总结一下在开发中如何实现文件与mongoDB之间的交互.在此之前,并没有系统的了解过mongoDB,虽然知道我们用它来存储文件这些非结构化数据,但是 ...

  6. 软件需求规格说明书(转自http://blog.csdn.net/li_canhui/article/details/6927540)

    转自http://blog.csdn.net/li_canhui/article/details/6927540 1概述 1.1编写目的 指出编写<需求规格说明书>的目的.下面是示例: 编 ...

  7. http&colon;&sol;&sol;blog&period;csdn&period;net&sol;v&lowbar;july&lowbar;v&sol;article&sol;details&sol;6543438

    本文转载至: http://blog.csdn.net/v_july_v/article/details/6543438 算法 程序员面试.算法研究.编程艺术.红黑树.数据挖掘5大经典原创系列集锦与总 ...

  8. http&colon;&sol;&sol;blog&period;csdn&period;net&sol;LANGXINLEN&sol;article&sol;details&sol;50421988

    GitHub上史上最全的Android开源项目分类汇总 今天在看博客的时候,无意中发现了 @Trinea在GitHub上的一个项目 Android开源项目分类汇总, 由于类容太多了,我没有一个个完整地 ...

  9. 线段树详解 (原理,实现与应用)(转载自:http&colon;&sol;&sol;blog&period;csdn&period;net&sol;zearot&sol;article&sol;details&sol;48299459)

    原文地址:http://blog.csdn.net/zearot/article/details/48299459(如有侵权,请联系博主,立即删除.) 线段树详解    By 岩之痕 目录: 一:综述 ...

随机推荐

  1. Linux下安装jetty服务器

    jetty和我们通常使用的tomcat一样,是一个开源的servlet容器,特点是轻量易部署,一方面jetty可以作为web容器使用,另一方面也是最一般的方式是jetty以一组jar包的形式发布,所以 ...

  2. 提高性能:用RequireJS优化Wijmo Web页面

    上周Wijmo 2014 V2版本刚刚发布(下载地址),  有网友下载后发现仅仅使用了40个Widgets的一小部分,还需要加载全部的jquery.wijmo-pro.all.3.20142.45.m ...

  3. codeforces629C Famil Door and Brackets (dp)

    题意:给你一个长度为n的括号匹配串(不一定恰好匹配),让你在这个串的前面加p串和后面加上q串,使得这个括号串平衡(平衡的含义是对于任意位置的括号前缀和大于等于0,且最后的前缀和为0). 思路:枚举这个 ...

  4. 转&colon; Transact-sql游标使用详解~~很详细

    /*原理:游标就是把数据按照指定要求提取出相应的数据集,然后逐条进行数据处理.1.1游标的概念 游标(Cursor)它使用户可逐行访问由SQL Server返回的结果集. 使用游标(cursor)的一 ...

  5. 实现标签的添加与删除&lpar;tags&rpar;

    在项目中会遇到,标签(tags)的添加与去除的需求 demo:我们有 tags   '专利','商标','版权','域名' demand:在发布内容的时候,要求可以添加tag,(实现tag的增加与删除 ...

  6. Python 调用让系统自动调用默认程序打开文件?

    windows上可以使用os.startfile os.startfile(file) linux上可以使用xdg-open subprocess.call(["xdg-open" ...

  7. HDU 2084 数塔&lpar;简单DP入门&rpar;

    数塔 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submiss ...

  8. MySQL如何正确查询字段为NULL的字段

    //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ ...

  9. linux文件命令汇总

    查看文件内容命令 cat 直接输出在命令行,适用于很少内容时候的输出, vim .vi  通过 编辑命令的只读模式进行文件内容的查看(翻页快捷键好像是 ctrl + F 下一页, ctrl + B 上 ...

  10. Vue前端框架面试问题

    1.active-class是哪个组件的属性?嵌套路由怎么定义? 答:vue-router模块的router-link组件. 2.怎么定义vue-router的动态路由?怎么获取传过来的动态参数? 答 ...