1. 从官方说明如下

  https://www.opengl.org/sdk/docs/man2/xhtml/gluProject.xml

  Name  

    gluProject   — map object coordinates to window coordinatesC,匹配“物体坐标系”“Win窗口坐标系(坐标原点在左下方)”

  Specification

    GLint  gluProject(GLdouble objX,  GLdouble objY,  GLdouble objZ,  const GLdouble * model,  const GLdouble * proj,  const GLint * view,  GLdouble* winX,  GLdouble* winY,  GLdouble* winZ);

  Parameters  

    objX, objY, objZ
                    Specify the object coordinates.
                model
                    Specifies the current modelview matrix (as from a glGetDoublev call).
                proj
                    Specifies the current projection matrix (as from a glGetDoublev call).
                view
                    Specifies the current viewport (as from a glGetIntegerv call).
                winX, winY, winZ
                    Return the computed window coordinates.

  Description
            gluProject transforms the specified object coordinates into window coordinates using model, proj, and view. The result is stored in winX, winY, and winZ. A return value of GLU_TRUE indicates success, a return value of GLU_FALSE indicates failure.

2. 使用介绍

  把空间中的一个三维坐标转换成二维坐标,用这个gluProject函数

3. 原码分析

  原码的写法很好 nice, 先通过看代码,来一步一步分析它的数学原理吧!既学习代码,同时也要掌握数学原理, 在这里才是成长的关键所在!

static void transform_point(GLdouble out[], const GLdouble m[], const GLdouble in[])
{
#define M(row,col) m[col*4+row]
out[] = M(, ) * in[] + M(, ) * in[] + M(, ) * in[] + M(, ) * in[];
out[] = M(, ) * in[] + M(, ) * in[] + M(, ) * in[] + M(, ) * in[];
out[] = M(, ) * in[] + M(, ) * in[] + M(, ) * in[] + M(, ) * in[];
out[] = M(, ) * in[] + M(, ) * in[] + M(, ) * in[] + M(, ) * in[];
#undef M
}
// gluProject source code (说明见OpenGL API文档)
GLint gluProject(GLdouble objx, GLdouble objy, GLdouble objz,
const GLdouble modelMatrix[], const GLdouble projMatrix[],
const GLint viewport[], GLdouble *winx, GLdouble *winy, GLdouble *winz)
{
// matrice transformation
GLdouble in[], out[];
//initialize matrice and column vector as a transformer
in[] = objx;
in[] = objy;
in[] = objz;
in[] = 1.0; transform_point(out, modelMatrix, in); //乘以模型视图矩阵
transform_point(in, projMatrix, out); //乘以投影矩阵
//齐次向量的第四项不能为0
if(in[] == 0.0)
return GL_FALSE;
//向量齐次化标准化
in[] /= in[];
in[] /= in[];
in[] /= in[]; //视口向量的作用
*winx = viewport[] + (1 + in[0]) * viewport[2] / 2;
*winy = viewport[] + ( + in[]) * viewport[] / ;
*winz = ( + in[]) / ;
return GL_TRUE;
}

  上述原码可以看出:

  (1)观察transform_point函数中矩阵元素和列向量的相乘过程可知,这个矩阵是被转置后再和列向量相乘的。这就是为什么说OpenGL的矩阵相乘是遵循列主元的,而我们使用gluProject函数的时候输入的矩阵参数却是按照行主元的方式。

  (2)从列向量被变换的顺序可以看出,一个齐次表示的3D点是先经过模型矩阵变换(蕴含照相机的位移、旋转、缩放等几何变换),在进行投影变换(蕴含照相机的内参数,即本身的参数),最后使用视口向量将齐次表示的2D点限定在当前的视口内。

  (3)根据函数的输入输出可以看出,输入的3D点是通过直接将向量的第四个元素in[3]赋值为1而齐次化的,输出的齐次表示其实是(winx,winy,1)。winz并不是投影在视口内的2D坐 标的齐次表示的第三个元素,它携带的含义是多方面的:表面上看来,它表示了在视口上的深度值;实际上,它是由于照相机参数的不精确性(所有照相机模型都是 对实际照相机的模拟,因而不可能完全精确的表达照相机本身)和矩阵相乘过程中的计算误差所造成的;因此假如照相机参数对照相机的建模是完美的,并且计算过 程是没有误差的,那么winz的最终结果应该是0,在实际的计算中此参数所返回的值也应该是一个逼近于0的数,这个数的大小反映了误差的总量。所以我们在使用gluProject时通常不必关心winz返回的参数。

  (4)对于返回的(winx,winy),论坛上曾有朋友发问说为什么得到的投影点和实际的点是关于图像上下倒置的。这个是因为OpenGL窗口坐标系统默认(0,0)点是窗口的左下角,而通常Windows窗口和一般图像坐标系的(0,0)点默认在左上角,因此,当我们在这种不一致的坐标系之间进行点的运算时,要先进行一步winy = height – winy; 的转换。

  另一种原码解释

GLint gluProject(GLdouble objx, GLdouble objy, GLdouble objz
, const GLdouble model[], const GLdouble proj[], const GLint viewport[]
, GLdouble * winx, GLdouble * winy, GLdouble * winz)
{
/* transformation matrix */
GLdouble objCoor[];
GLdouble objProj[], objModel[]; /* initilise matrix and vector transform */
// 4x4 matrix must be multi to a 4 dimension vector( it a 1 x 4 matrix)
// so we need to put the original vertex to a 4D vector
objCoor[] = objx;
objCoor[] = objy;
objCoor[] = objz;
objCoor[] = 1.0; // 由于原来的向量位于标准基向量(1, 0, 0), (0, 1, 0), (0, 0, 1)中,所以需要先转换到当前的模型矩阵中
transform_point(objModel, model, objCoor); // 然后将模型矩阵中的顶点转换到投影矩阵所在坐标系的矩阵中
transform_point(objProj, proj, objModel); // scale matrix
/*
GLdouble scaleMat[4][4] =
{
{0.5, 0, 0, objPr0j[3]},
{0, 0.5, 0, objProj[3]},
{0, 0, 0.5, objProj[3]},
{1, 1, 1, 1}
}; GLdouble objProjTemp[4];
memcpy(objProjTemp, objProj, sizeof(objProjTemp);
transfrom_point(objProj, scaleMat, objProjTemp);
*/ /* or the result of normalized between -1 and 1 */
if (objProj[] == 0.0)
return GL_FALSE; objProj[] /= objProj[];
objProj[] /= objProj[];
objProj[] /= objProj[]; /* in screen coordinates */
// 由于投影矩阵投影在[-1, 1]之间,所以需要将转换后的投影坐标放置到[0, 1]之间
// 最后再在一个offset 矩形中转换为屏幕坐标就可以了(viewport[4]可以简单的认为一个offset矩形) #define SCALE_FROM_0_TO_1(_pt) (((_pt) + 1)/2)
objProj[] = SCALE_FROM_0_TO_1(objProj[]);
objProj[] = SCALE_FROM_0_TO_1(objProj[]);
objProj[] = SCALE_FROM_0_TO_1(objProj[]);
#undef SCALE_FROM_0_TO_1 *winx = viewport[] + objProj[] * viewport[];
*winy = viewport[] + objProj[] * viewport[]; /* between 0 and 1 */
*winz = objProj[];
return GL_TRUE;
}

基本的思路就是:

  (1)将输入的顶点,通过模型视图矩阵,变换到模型视图矩阵的坐标系中;

  (2)将模型视图矩阵中的顶点,再变换到投影矩阵中;

  (3)将顶点缩放到[0, 1]的映射区间中;应该是[-1, 1]

  (4)通过视口的位置和大小,计算出当前3D顶点中的屏幕坐标(2D坐标);

  其中,因为顶点缩放到[0, 1]映射后,(1 + in[0]) * viewport[2] / 2; 计算出窗口坐标中的宽度的像素,一种可能的解释是:顶点坐标缩放到[-1, 1]的范围。

  其实gluUnproject和gluProject是非常类似的, 其实就是gluPorject反过来的过程,只是有一些数学运算要注意一下:

  (1)首先,需要将输入的顶点,通过视口变换到[0, 1]之间;

  (2)然后将顶点缩放到[-1, 1]之间,就是上面代码中的scaleMat矩阵的逆矩阵

  (3)然后乘上投影矩阵的逆矩阵;

  (4)最后就是乘上模型视图矩阵的逆矩阵;

  

  

Opengl中的gluProject函数认识的更多相关文章

  1. OpenGL中常用的函数

    OPengl的官方文档如下:https://www.opengl.org/sdk/docs/man4/ void glGetIntegerv(   GLenum pname,      GLint * ...

  2. opengl中对glOrtho()函数的理解

    glOrtho是创建一个正交平行的视景体. 一般用于物体不会因为离屏幕的远近而产生大小的变换的情况.比如,常用的工程中的制图等.需要比较精确的显示. 而作为它的对立情况, glFrustum则产生一个 ...

  3. OpenGL中glRotatef()函数究竟对矩阵做了什么

    OpenGL中glRotatef()函数究竟对矩阵做了什么 我们知道OpenGL中维持着两套矩阵,一个是模型视图矩阵(model view matrix),另一个是投影矩阵(projection ma ...

  4. openGL中的gl,glu,glut

    OpenGL函数库相关的API有核心库(gl).实用库(glu).辅助库(aux).实用工具库(glut).窗口库(glx.agl.wgl)和扩展函数库等.gl是核心,glu是对gl的部分封装.glx ...

  5. CSharpGL(26)在opengl中实现控件布局/渲染文字

    CSharpGL(26)在opengl中实现控件布局/渲染文字 效果图 如图所示,可以将文字.坐标轴固定在窗口的一角. 下载 CSharpGL已在GitHub开源,欢迎对OpenGL有兴趣的同学加入( ...

  6. OpenGL中坐标系的理解(一)

    在OpenGL中,存在着至少存在着三种矩阵,对应着函数glMatrixMode()的三个参数:GL_MODELVIEW,GL_PROJECTION,GL_TEXTURE. 以下主要描述GL_MODEL ...

  7. CSharpGL(6)在OpenGL中绘制UI元素

    CSharpGL(6)在OpenGL中绘制UI元素 2016-08-13 由于CSharpGL一直在更新,现在这个教程已经不适用最新的代码了.CSharpGL源码中包含10多个独立的Demo,更适合入 ...

  8. OpenGL中glVertex、显示列表(glCallList)、顶点数组(Vertex array)、VBO及VAO区别

    OpenGL中glVertex.显示列表(glCallList).顶点数组(Vertex array).VBO及VAO区别 1.glVertex 最原始的设置顶点方法,在glBegin和glEnd之间 ...

  9. OpenGL中实现双缓冲技术

    在OpenGL中实现双缓冲技术的一种简单方法: 1.在调用glutInitDisplayMode函数时, 开启GLUT_DOUBLE,即glutInitDisplayMode(GLUT_RGB | G ...

随机推荐

  1. scala --操作符和运算

    基本类型和操作 scala 的基本类型包括如下几种 数类型 ​ 整数类型 :Byte Short Int Long ​ 小数类型: Float Double 字符类型:Char 用'' 单引号包裹,是 ...

  2. swagger ui

    You can pull a pre-built docker image of the swagger-ui directly from Dockerhub: docker pull swagger ...

  3. git pull 免密

    linux下 在~/下, touch创建文件 .git-credentials, 用vim编辑此文件,输入内容格式: touch .git-credentials vim .git-credentia ...

  4. 结对项目3-bug的三种状态

    这周和小伙伴结对构造程序,来深刻理解软件测试中,bug发现的三种状态. 1:不能触发Fault 2:触发Fault,但是不能触发Error 3:触发Error,但是不能产生Failure 我们完成的代 ...

  5. The value for the useBean class attribute is invalid.

    报错如下: The value for the useBean class attribute com.JavaBeanTest is invalid. 解决方法: 在JavaBean代码中加入无参数 ...

  6. vs code 插件推荐

    通用插件 HTML Snippets 超级实用且初级的 H5代码片段以及提示 HTML CSS Support 让 html 标签上写class 智能提示当前项目所支持的样式新版已经支持scss文件检 ...

  7. 如何在ecplise中配置maven以及ecplise访问本地仓库

    1.m2e的插件 因为使用ecplise版本比较高,所以它自带了maven的插件,但是我们希望可以使用我们自己指定的maven.配置步骤如下: ecplise--->preperences下,点 ...

  8. Spring的2.5版本中提供了一种:p名称空间的注入(了解)

    1. 步骤一:需要先引入 p 名称空间 * 在schema的名称空间中加入该行:xmlns:p="http://www.springframework.org/schema/p"( ...

  9. CFR - another java decompiler批量反编译jar文件

    jd-gui众所周知,业界公认的反编译必备工具. 笔者目前遇到一个java项目,社区版,想做一个本地化的版本,询问官方,官方说闭源,无奈之下只能反编译了. 面对那么多jar,jd-gui一个个去反编译 ...

  10. windows常用运行命令总结

    开始→运行→命令集锦 winver---------检查Windows版本 wmimgmt.msc----打开windows管理体系结构(WMI) wupdmgr--------windows更新程序 ...