10.10节书中给出了一个程序示例,有一个填充正方形,从侧面的角度观察并画到屏幕上。

  图0

  这里进一步画出一个立方体,将相机放入立方体中心,旋转相机,达到在立方体中旋转看到不同画面的效果。

步骤:

  1 使用的是4.9节中的OpenGL顶点数组方法。创建一个立方体100*100*100,坐标范围(0, 0, 0)到(100, 100, 100)。

  2 立方体各面使用不同的颜色,调整顶点顺序以确保相机看到的都是填充面而不是线框图。

  3 将投影观察点(即观察系原点)设置在矩形中心P0 = (50, 50, 50)。

  4 视点Pref为相机朝向的点(注意,这个不是观察点),初始设置为Pref =(0, 50, 0)。

  5 修改Pref的坐标,使其绕x=z=50进行逆时针旋转(与y轴平行的旋转公式(9.9)),最终实现相机在立方体中逆时针旋转,看到不同面的效果。

备注:

  1 与y轴平行的旋转公式,先将旋转轴平移到与y轴重合,再进行旋转:z' = z * cosθ - x * sinθ, x' = z * sinθ + x * cosθ, y'=y;

  2 近平面距离投影观察点不能为0,近平面和远平面的坐标会影响画面的效果,调整范围以确保画面不太离谱(并不真实)

  3 之前直接在displayFcn中调用cube,修改使用display list后可以提高cpu的效率,从2%降到1%。

问题:

  1 调整dnear和dfar时,会影响投影效果。dnear离P0越近,投影效果强度更大,离得远的点看着更小。见图2。不真实。

  2 当调整近裁剪面大小且当立方体旋转时,投影画面会改变宽度,看起来会更奇怪。见图3。不真实。

  3 在idleFcn中使用glutPostRedisplay替换直接调用displayFcn,效率有降低,不懂为什么。

 #include <GLUT/GLUT.h>
#include <math.h> GLint winWidth = , winHeight = ;
GLfloat x0 = 50.0, y00 = 50.0, z0 = 50.0; // 这是观察参考点,与观察系原点重合
GLfloat xref = 0.0, yref = 50.0, zref = 0.0; // camera要瞄准的点,不是观察参考点
GLfloat Vx = 0.0, Vy = 1.0, Vz = 0.0;
GLfloat xwMin = -40.0, ywMin = -40.0, xwMax = 40.0, ywMax = 40.0;
GLfloat dnear = , dfar = ;
GLdouble radianAngle = 1.0/360.0 * 3.14159;
GLfloat cos1 = cos(radianAngle);
GLfloat sin1 = sin(radianAngle); typedef GLint vertex3 [];
vertex3 pt [] = {{, , }, {, , }, {, , }, {, , },
{, , }, {, , }, {, , }, {, , }}; void init (void)
{
glClearColor(1.0, 1.0, 1.0, 0.0); glMatrixMode(GL_MODELVIEW);
gluLookAt(x0, y00, z0, xref, yref, zref, Vx, Vy, Vz); glMatrixMode(GL_PROJECTION);
glFrustum(xwMin, xwMax, ywMin, ywMax, dnear, dfar); void cube();
cube();
} void quad (GLint n1, GLint n2, GLint n3, GLint n4)
{
glBegin(GL_QUADS);
glVertex3iv(pt[n1]);
glVertex3iv(pt[n2]);
glVertex3iv(pt[n3]);
glVertex3iv(pt[n4]);
glEnd();
} GLuint regHex; void cube ()
{
regHex = glGenLists();
glNewList(regHex, GL_COMPILE); glColor3f(0.0, 1.0, 0.0); // back
quad(, , , );
glColor3f(1.0, 0.0, 0.0); // left
quad(, , , );
glColor3f(0.0, 0.0, 1.0); // top
quad(, , , );
glColor3f(1.0, 1.0, 0.0); // front
quad(, , , );
glColor3f(0.0, 1.0, 1.0); // right
quad(, , , );
glColor3f(1.0, 0.0, 1.0); // bottom
quad(, , , ); glEndList();
} void displayFcn (void)
{
glClear(GL_COLOR_BUFFER_BIT); glColor3f(0.0, 1.0, 0.0);
glPolygonMode(GL_FRONT, GL_FILL);
glPolygonMode(GL_BACK, GL_LINE);
// glFrontFace(GL_CW); // show the back glCallList(regHex); glutSwapBuffers();
} void reshapeFcn (GLint newWidth, GLint newHeight)
{
glViewport(, , newWidth, newHeight); winWidth = newWidth;
winHeight = newHeight;
} void idleFcn (void)
{
GLfloat xref1, zref1;
xref = xref - x0;
zref = zref - z0;
zref1 = zref * cos1 - xref * sin1 + z0;
xref1 = zref * sin1 + xref * cos1 + x0;
xref = xref1;
zref = zref1; glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
gluLookAt(x0, y00, z0, xref, yref, zref, Vx, Vy, Vz); glutPostRedisplay();
} int main(int argc, char * argv[])
{
glutInit(&argc, argv);
glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE);
glutInitWindowSize(winWidth, winHeight);
glutInitWindowPosition(, );
glutCreateWindow("Perspective View of A Square"); init();
glutDisplayFunc(displayFcn);
glutReshapeFunc(reshapeFcn);
glutIdleFunc(idleFcn);
glutMainLoop(); return ;
}

图1  图2  图3

[图形学] Chp10 OpenGL三维观察程序示例的更多相关文章

  1. 开源免费跨平台opengl opencv webgl gtk blender, opengl贴图程序

    三维图形的这是opengl的强项,大型3D游戏都会把它作为首选.图像处理,是opencv的锁定的目标,大多都是C的api,也有少部分是C++的,工业图像表现,图像识别,都会考虑opencv的.webg ...

  2. 学习基于OpenGL的CAD程序的开发计划(一)

    本人目前从事的工作面对的客户中很多来自高端制造业,他们对CAD/CAE/CAM软件的应用比较多.公司现有的软件产品主要是用于渲染展示及交互,但面对诸如CAD方面的应用(比如基于约束的装配.制造工艺的流 ...

  3. 《Unix 网络编程》05:TCP C/S 程序示例

    TCP客户/服务器程序示例 系列文章导航:<Unix 网络编程>笔记 目标 ECHO-Application 结构如下: graph LR; A[标准输入/输出] --fgets--> ...

  4. 转载--Linux命令top动态观察程序的变化

    转载:http://www.cnblogs.com/allen8807/archive/2010/11/10/1874001.html top:动态观察程序的变化 [root@linux ~]# to ...

  5. Arduino 入门程序示例之一个 LED(2015-06-11)

    前言 答应了群主写一些示例程序,一直拖延拖延拖延唉.主要还是害怕在各大高手面前班门弄斧……(这也算是给拖延症找一个美好的理由吧),这几天终于下决心要写出来了,各位高手拍砖敬请轻拍啊. 示例程序 首先是 ...

  6. WinDBG调试.NET程序示例

    WinDBG调试.NET程序示例 好不容易把环境打好了,一定要试试牛刀.我创建了一个极其简单的程序(如下).让我们期待会有好的结果吧,阿门! using System; using System.Co ...

  7. 现代控制理论习题解答与Matlab程序示例

    现代控制理论习题解答与Matlab程序示例 现代控制理论 第三版 课后习题参考解答: http://download.csdn.net/detail/zhangrelay/9544934 下面给出部分 ...

  8. 第一个Mybatis程序示例 Mybatis简介(一)

    在JDBC小结中(可以参阅本人JDBC系列文章),介绍到了ORM,其中Mybatis就是一个不错的ORM框架 MyBatis由iBatis演化而来 iBATIS一词来源于“internet”和“aba ...

  9. map reduce程序示例

    map reduce程序示例 package test2; import org.apache.hadoop.conf.Configuration; import org.apache.hadoop. ...

随机推荐

  1. Java调用IDL出错处理

    之前有一个java调用idl的详细介绍http://www.cnblogs.com/lizhishan3380/p/4353286.html,里面有提到[需要先在java中加载IDL的java包(ja ...

  2. Java操作PDF之iText超入门

    iText是著名的开放项目,是用于生成PDF文档的一个java类库.通过iText不仅可以生成PDF或rtf的文档,而且可以将XML.Html文件转化为PDF文件. http://itextpdf.c ...

  3. SmartCoder每日站立会议05

    1.站立会议内容 经过四天的努力,我们的微信小程序有了很大的进展,首页的设计定了方案,API接地图正在试着把消息展示到其中,争取把地图信息做到直观形象. 站立会议照片: 2.任务展板 3.燃尽图

  4. 源码阅读—Iterator接口和LIstIterator接口

    在继续看ArrayList源码之前,先了解Iterator接口和ListIterator接口,下篇文章详细讲解ArrayList是如何实现它们的. 我们知道,接口只是一种规范,当继承接口并实现其中的方 ...

  5. 数据库连接之SQL JDBC

    数据库连接之SQL JDBC SQlServer的配置: 1).外围配置服务器中将远程连接设置为:同时使用TCP/IP和named pipes 2).点击该连接->属性->安全性-> ...

  6. 一天搞定CSS:字体font--04

    1.字体体系 2.字体各属性取值 说明: 每一个属性后面的分支是属性值,以及对属性值的说明. 比如font-weight- - - -有两个取值:bold,normal 3.演示代码 <!DOC ...

  7. linux升级openssh7.4sp1

    1.准备相关的包 openssh下载地址:http://mirror.internode.on.net/pub/OpenBSD/OpenSSH/portable/ openssl相关包下载:http: ...

  8. javaScript 设计模式系列之二:适配器模式

    介绍 适配器模式将一个类的接口转接成用户所期待的,有助于避免大规模改写现有客户代码. In software engineering, the adapter pattern is a softwar ...

  9. 使用r.js来打包模块化的javascript文件

    前面的话 r.js(下载)是requireJS的优化(Optimizer)工具,可以实现前端文件的压缩与合并,在requireJS异步按需加载的基础上进一步提供前端优化,减小前端文件大小.减少对服务器 ...

  10. 几个常用的linux命令(操作服务器时会用到)

    目录 tmux 背景 安装 使用 启动一个tmux session 暂时离开当前session 回到之前的session 重命名session 创建window 创建pane ps scp 参考 tm ...