正交投影
viod glOrtho(GLdouble left,GLdouble right,GLdouble bottom,GLdouble top,GLdouble near,GLdouble far)
//设定一个正交投影矩阵,并定义了一个形状为直平行六面体的视域体。
void glutWireCube(GLdouble size) //生成一个线框立方体,中心位于原点,边长为size
void glutSolidCube(GLdouble size) //生成一个实立方体
一个简单的画三维立方体的例子:
#include <gl/glut.h>
#include <math.h>
#include <vector>
#include <iostream>
using namespace std;
void init()
{
glClearColor(0.0,0.0,0.0,0.0); //指定屏幕背景为黑色
glColor3f(1.0,1.0,1.0); //设置绘制颜色为白色
glShadeModel(GL_FLAT); //设置颜色插值为平面模式
}
void display()
{
glClear(GL_COLOR_BUFFER_BIT); //清理屏幕颜色为我们指定的颜色
glutSolidCube(0.5);
glFlush(); //强制以上绘图操作执行
}
void reshape(int w,int h)
{
glMatrixMode(GL_PROJECTION); //设置为投影模式
glLoadIdentity();
glOrtho(-1.0,1.0,-1.0,1.0,-1.0,1.0);
glViewport(0,0,(GLsizei)w,(GLsizei)h);
}
int main(int argc,char**argv)
{
glutInit(&argc,argv); //初始化glut
glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB); //设置窗口模式为单缓冲和RGB模式
glutInitWindowSize(500,500); //设置窗口大小
init();
glutCreateWindow("test"); //设置窗口标题
glutDisplayFunc(display); //设置绘图回调函数
glutReshapeFunc(reshape); //设置窗口回调函数
glutMainLoop(); //开始循环,等待响应
return 0;
}
运行得到:
摄影机的定位
上面的例子存在一个问题,即摄像机只朝向该立方体的一个面,所以我们也只能看到该立方体的一个面。如果想看到其他面,必须移动摄像机或者被观察对象。而在OpenGL中,这两个操作时等价的。由视图矩阵来实现。
为了获得所期望的视图,我们可以在对象坐标系中设置摄像机的位置和朝向。我们可以决定摄像机的位置并通过该摄像机所指向的点来指定。其次还需指定一个向上向量。为方便起见,通常将向量取为(0,1,0)或者坐标系中的y轴方向
void gluLookAt(GLdouble eyex,GLdouble eyey,GLdouble eyez,
GLdouble atx,GLdouble aty,GLdouble atz,
GLdouble upx,GLdouble upy,GLdouble upz)
//确定了一个可用于为摄像机定位和定向的矩阵,涉及的参数包括摄像机的位置(视点eye),被观察点(点at)以及所期望的up方向。
再更改上述代码可以得到
#include <gl/glut.h>
#include <math.h>
#include <vector>
#include <iostream>
using namespace std;
void init()
{
glClearColor(0.0,0.0,0.0,0.0); //指定屏幕背景为黑色
glColor3f(1.0,1.0,1.0); //设置绘制颜色为白色
glShadeModel(GL_FLAT); //设置颜色插值为平面模式
}
void display()
{
glClear(GL_COLOR_BUFFER_BIT); //清理屏幕颜色为我们指定的颜色
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
gluLookAt(1.0,1.0,1.0,0.0,0.0,0.0,0.0,1.0,0.0);
glutWireCube(0.5);
glFlush(); //强制以上绘图操作执行
}
void reshape(int w,int h)
{
glMatrixMode(GL_PROJECTION); //设置为投影模式
glLoadIdentity();
glOrtho(-2.0,2.0,-2.0,2.0,-2.0,2.0);
glViewport(0,0,(GLsizei)w,(GLsizei)h);
}
int main(int argc,char**argv)
{
glutInit(&argc,argv); //初始化glut
glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB); //设置窗口模式为单缓冲和RGB模式
glutInitWindowSize(500,500); //设置窗口大小
init();
glutCreateWindow("test"); //设置窗口标题
glutDisplayFunc(display); //设置绘图回调函数
glutReshapeFunc(reshape); //设置窗口回调函数
glutMainLoop(); //开始循环,等待响应
return 0;
}
运行得到:
因为把摄像机的位置放在了(1,1,1)那点,所以得到这样的图像。
顶点数组
void glEnableClientState(GLenum array)
void glDisableClientState(GLenum array)
//启用或禁用数组类型GL_VERTEX_ARRAY、GL_COLOR_ARRAY、GL_INDEX_ARRAY、GL_NORMAL_ARRAY、 GL_TEXTURE_COORD_ARRAY、GL_EDGE_FLAG_ARRAY.
void glVertexPointer(GLint dim,GLenum type,GLsizei stride,GLvoid* array)
void glColorPointer(GLint dim,GLenum type,GLsizei stride,GLvoid* array)
//提供数组中的信息。数据存储在数组array中,dim为数组的维数(2,3,4),type表明数据的存储类 型(GL_SHORT、GL_INT、GL_FLOAT、GL_DOUBLE),stride是相邻两个数据之间的字节数(0意味着时 紧密排列的)
void glDrawElements(GLenum mode,GLsizei n,GLenum type,void* indices)
//使用数组indices中的n个索引来绘制类型为mode的元素,该数组可以为一下类型 (GL_UNSIGEND_BYTE、GL_UNSIGEND_SHORT、GL_UNSIGEND_INT)
消隐
绘制三维图形时,如不使用消隐,即使后面物体被前面物体挡住,OpenGL也会绘制被挡住的物体。
因此OpenGL利用深度缓存算法开启消隐功能.
首先在创建窗口的时候得请求一个深度缓存
glutInitDiaplayMode(GLUT_RGB | GLUT_DOUBLE | GLUT_DEPTH)
然后开启消隐功能
glEnable(GL_DEPTH_TEST);
清空颜色缓存时,通常将深度缓存清空:
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
下面看一个绘制立方体的例子:
#include <gl/glut.h>
GLfloat vertices[]={
-1.0,-1.0,1.0,
-1.0,1.0,1.0,
1.0,1.0,1.0,
1.0,-1.0,1.0,
-1.0,-1.0,-1.0,
-1.0,1.0,-1.0,
1.0,1.0,-1.0,
1.0,-1.0,-1.0
}; //定义立方体的8个顶点
GLint index[]={
0,3,2,1,
2,3,7,6,
3,0,4,7,
1,2,6,5,
4,5,6,7,
5,4,0,1
}; //定义每个面所需要那几个顶点
GLfloat colors[]={
1.0,0.0,0.0,
0.0,1.0,1.0,
1.0,1.0,0.0,
0.0,1.0,0.0,
0.0,0.0,1.0,
1.0,0.0,1.0,
1.0,1.0,0.0,
0.0,1.0,1.0
}; //定义六个面的颜色
void init()
{
glClearColor(0.0,0.0,0.0,0.0); //指定屏幕背景为黑色
glColor3f(1.0,1.0,1.0); //设置绘制颜色为白色
glShadeModel(GL_FLAT); //设置颜色插值为平面模式
}
void display()
{
glEnable(GL_DEPTH_TEST);
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); //清理屏幕颜色为我们指定的颜色
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
gluLookAt(0.5,0.5,0.5,0.0,0.0,0.0,0.0,1.0,0.0);
glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_COLOR_ARRAY);
glVertexPointer(3,GL_FLOAT,0,vertices);
glColorPointer(3,GL_FLOAT,0,colors);
glDrawElements(GL_QUADS,24,GL_UNSIGNED_INT,index);
glFlush(); //强制以上绘图操作执行
}
void reshape(int w,int h)
{
glMatrixMode(GL_PROJECTION); //设置为投影模式
glLoadIdentity();
glOrtho(-2.0,2.0,-2.0,2.0,-2.0,2.0);
glViewport(0,0,(GLsizei)w,(GLsizei)h);
}
int main(int argc,char**argv)
{
glutInit(&argc,argv); //初始化glut
glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB | GLUT_DEPTH); //设置窗口模式为单缓冲和RGB模式
glutInitWindowSize(500,500); //设置窗口大小
init();
glutCreateWindow("test"); //设置窗口标题
glutDisplayFunc(display); //设置绘图回调函数
glutReshapeFunc(reshape); //设置窗口回调函数
glutMainLoop(); //开始循环,等待响应
return 0;
}
运行得到如下结果:
转载于:https://blog.51cto.com/6996127/1546600