OpenGL初探

第十三节—GLTools自带图形绘制和视变换

2020-08-29  本文已影响0人  L_Ares

本文为L_Ares个人写作,包括图片皆为个人亲自操作,以任何形式转载请表明原文出处。

OpenGL的一些库中是有一些自带图形的,用于一些简单的图形绘制是很好用的,另外本节代码也会为下一节做一下铺垫。

直接上代码,代码中都有非常详细的注释,而且都是经过测试可以跑起来的。


//
//  main.cpp
//  07矩阵堆栈
//
//  Created by EasonLi on 2020/8/29.
//  Copyright © 2020 EasonLi. All rights reserved.
//

#include <stdio.h>

#pragma mark - 引用类

//着色器管理器。负责管理和创建着色器管理器。里面有一些固定的着色器,可用于初级操作
#include "GLShaderManager.h"
//GLTool工具类,里面包含了GLTool大多数类似C语言的函数
#include "GLTools.h"
//矩阵堆栈类。包含了矩阵堆栈的操作
#include "GLMatrixStack.h"
//管道类,用于更加方便的管理矩阵
#include "GLGeometryTransform.h"
//用于设置观察者或者物体的位置
#include "GLFrame.h"
//投影矩阵获取
#include "GLFrustum.h"
//三角形批次类
#include "GLBatch.h"
//3d数学类
#include "math3d.h"

//宏定义判断系统,引入GLUT
#ifdef __APPLE__
#include <glut/glut.h>
#else
#define FREEGLUT_STATIC
#include <GL/glut.h>
#endif

#pragma mark - 公共变量

//着色器管理器
GLShaderManager shaderManager;
//几何变换管道
GLGeometryTransform transformPipeline;

//模型视图矩阵
GLMatrixStack modelViewMatrix;
GLMatrixStack projectionMatrix;

//观察者(相机)位置
GLFrame cameraFrame;
//物体世界坐标位置
GLFrame objectFrame;

//视景体,用来构造投影矩阵
GLFrustum viewFrustum;

//三角形批次类
//球
GLTriangleBatch sphereBatch;
//环
GLTriangleBatch torusBatch;
//圆柱
GLTriangleBatch cylinderBatch;
//锥
GLTriangleBatch coneBatch;
//磁盘
GLTriangleBatch diskBatch;

//颜色
GLfloat vRed[] = {1.f,0.f,0.f,1.f};
GLfloat vBlack[] = {0.f,0.f,0.f,1.f};

//记录点击空格的次数
int step = 0;

#pragma mark - 函数

/**
 设置窗口
 */
void ChangeSize(int w,int h)
{
    
    if (h == 0) {
        h = 1;
    }
    
    //设置视口
    glViewport(0, 0, w, h);
    
    //设置透视投影
    //1.观察者垂直视角度数。2.宽高比。3.视点与近剪裁面距离。4.视点与远剪裁面距离
    viewFrustum.SetPerspective(36.f, float(w)/float(h), 1.f, 500.f);
    
    //projectionMatrix矩阵堆栈加载透视投影矩阵
    projectionMatrix.LoadMatrix(viewFrustum.GetProjectionMatrix());
    
    //modelViewMatrix堆栈矩阵加载一个单元矩阵。其实不加载也没事,源码中可以看到,已经帮忙加载了一个了。
    modelViewMatrix.LoadIdentity();
    
    //通过GLGeometryTransform管理矩阵堆栈(模型视图矩阵堆栈、投影矩阵堆栈)
    transformPipeline.SetMatrixStacks(modelViewMatrix, projectionMatrix);
    
    
}

/**
 设置渲染环境
 */
void SetUpRC ()
{
    
    //设置清屏颜色
    glClearColor(0.3f, 0.3f, 0.3f, 1.f);
    //着色器管理器初始化
    shaderManager.InitializeStockShaders();
    
    //开启需要用到的环境
    //开启深度测试
    glEnable(GL_DEPTH_TEST);
    
    //设置观察者和物体的距离,保证物体都能看得到
    //这里有两种方式(1)(2)
    //(1)将观察者(摄像机)位置沿其Z轴向内部移动15个单位。负数往里面移动,正数向外面移动。
    //cameraFrame.MoveForward(-15.f);
    
    //(2)将物体移动15个单位,观察者(摄像机)不动
    objectFrame.MoveForward(15.f);
    
    //这里直接使用一下GLTools里面给过的图形
    /**
     球:
     gltMakeSphere(GLTriangleBatch& sphereBatch, GLfloat fRadius, GLint iSlices, GLint iStacks);
     参数:
     (1)sphereBatch:三角形批次类对象
     (2)fRadius:球体半径
     (3)iSlices:从球体底部堆叠到球体顶部的三角形带的数量。可以看出球体是一圈圈的三角形带组成的。
     (4)iStacks:围绕球体一圈排列的三角形片段对数
     建议:一个对称性较好的球体一般满足片段数量是三角形带数量的2倍。iSlices * 2 = iStacks
     绘制球体都是围绕Z轴的,因为这样Z轴的正方向(+)就是球体顶部,Z轴负方向(-)就是球体底部。
     */
    gltMakeSphere(sphereBatch, 3.f, 10, 20);
    
    /**
     环:
     gltMakeTorus(GLTriangleBatch& torusBatch, GLfloat majorRadius, GLfloat minorRadius, GLint numMajor, GLint numMinor);
     参数:
     (1)torusBatch:三角形批次类对象
     (2)majorRadius:圆环中心到外边缘的半径(大圆半径)
     (3)minorRadius:圆环中心到内边缘的半径(小圆半径)
     (4)numMajor:大圆的三角形数量
     (5)numMinor:小圆的三角形数量
     */
    gltMakeTorus(torusBatch, 3.f, 1, 16, 16);
    
    /**
     圆柱:
     gltMakeCylinder(GLTriangleBatch& cylinderBatch, GLfloat baseRadius, GLfloat topRadius, GLfloat fLength, GLint numSlices, GLint numStacks);
     参数:
     (1)cylinderBatch:三角形批次类对象
     (2)baseRadius:底部圆半径
     (3)topRadius:顶部圆半径
     (4)fLength:圆柱体的长度(高度)
     (5)numSlices:围绕Z轴的三角形带的对数
     (6)numStacks:从圆柱底到圆柱顶的三角形带的数量
     */
    gltMakeCylinder(cylinderBatch, 3.f, 3.f, 3.f, 16, 3);
    
    /**
     锥:特殊的圆柱
     gltMakeCylinder(GLTriangleBatch& cylinderBatch, GLfloat baseRadius, GLfloat topRadius, GLfloat fLength, GLint numSlices, GLint numStacks);
     参数:
     (1)cylinderBatch:三角形批次类对象
     (2)baseRadius:底部圆的半径
     (3)topRadius:顶部圆半径,锥体直接给0就行了
     (4)fLength:锥体长度(高度)
     (5)numSlices:围绕Z轴的三角形带的对数
     (6)numStacks:从圆柱底到圆柱顶的三角形带的数量
     */
    gltMakeCylinder(coneBatch, 3.f, 0.f, 3.f, 16, 3);
    
    /**
     磁盘:
     gltMakeDisk(GLTriangleBatch& diskBatch, GLfloat innerRadius, GLfloat outerRadius, GLint nSlices, GLint nStacks);
     参数:
     (1)diskBatch:三角形批次类对象
     (2)innerRadius:内部圆半径
     (3)outerRadius:外部圆半径
     (4)nSlices:围绕Z轴的三角形对的数量
     (5)nStacks:从内边缘到外边缘三角形带的对数
     */
    gltMakeDisk(diskBatch, 1.f, 3.f, 16, 4);
    
}

/**
 利用三角形批次类进行绘制
 */
void DrawBatch(GLTriangleBatch *dBatch)
{
    
    //绘制图形
    shaderManager.UseStockShader(GLT_SHADER_FLAT,transformPipeline.GetModelViewProjectionMatrix(),vRed);
    
    //这里用的是指针,看清楚上面,不能用点方法
    dBatch->Draw();
    
    //绘制黑色线条
    //这里就需要开启颜色混合、多边形偏移、抗锯齿等功能
    
    //开启多边形偏移
    glEnable(GL_POLYGON_OFFSET_LINE);
    //设置偏移的面和模式
    glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
    //指定偏移量
    glPolygonOffset(-1.f, -1.f);
    
    //开启混合
    glEnable(GL_BLEND);
    //设置混合因子
    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
    //开启线段模式抗锯齿(简单的抗锯齿)
    glEnable(GL_LINE_SMOOTH);
    
    //可以设置一下线段的宽度
    glLineWidth(3.f);
    
    //利用平面着色器画黑线
    shaderManager.UseStockShader(GLT_SHADER_FLAT,transformPipeline.GetModelViewProjectionMatrix(),vBlack);
    dBatch->Draw();
    
    //绘制完成记得关闭之前开启的功能,回复图形填充方式,并恢复线段宽度
    glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
    glDisable(GL_POLYGON_OFFSET_LINE);
    glDisable(GL_BLEND);
    glDisable(GL_LINE_SMOOTH);
    glLineWidth(1.f);
    
}

//观察者移动(摄像机),物体不动
void CameraMoveObjNo()
{
    //复制栈顶的值,并且压入栈顶
    modelViewMatrix.PushMatrix();
    
    //获取观察者(摄像头)矩阵
    //先定义一个矩阵,方便一会存储获取到的观察者(摄像头)矩阵
    M3DMatrix44f mCamera;
    //从cameraFrame中获取它栈顶的矩阵,然后赋值给mCamera,好用来做矩阵变换,这样做能保证多图形的情况下,不改变原
    //cameraFrame的栈顶矩阵,而用一份复制出来的操作,不会影响到其他的图形绘制
    cameraFrame.GetCameraMatrix(mCamera);
    //让modelViewMatrix模型视图矩阵和观察者(摄像头)矩阵相乘,发生第一步变化,相乘的结果会存储到
    //modelViewMatrix栈顶里面
    modelViewMatrix.MultMatrix(mCamera);
    
    //获取物体的矩阵
    //先定义一个矩阵,也是方便存储获取到的物体仿射变换后的矩阵
    M3DMatrix44f mObject;
    //从objectFrame中获取其栈顶的矩阵,并且赋值给mObject,原理和上面的一样
    objectFrame.GetMatrix(mObject);
    //让modelViewMatrix模型视图矩阵和mObject相乘,结果矩阵依然存储在modelViewMatrix栈顶
    modelViewMatrix.MultMatrix(mObject);
    
}

//物体移动,观察者(摄像机)不动
void ObjMoveCameraNo()
{
    
    modelViewMatrix.PushMatrix(objectFrame);
    
}

/**
 渲染
 */
void RenderScene()
{
    
    //清空缓冲区
    //虽然模版缓冲区都没用到,但是养成全部清空的好习惯肯定不会错
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
    
    //当观察者移动(摄像机),而物体不移动的时候,对应了SetUpRC中的(1)
    //CameraMoveObjNo();
    
    //当观察者不动,物体移动的时候,对应了SetUpRC中的(2)
    ObjMoveCameraNo();
    
    
    //到上面这一步,矩阵的变换已经都做完了,下面要绘制了
    //判断空格点击次数,更换不同的图形
    switch (step) {
            
        case 0:
            DrawBatch(&sphereBatch);
            break;
            
        case 1:
            DrawBatch(&torusBatch);
            break;
            
        case 2:
            DrawBatch(&cylinderBatch);
            break;
        
        case 3:
            DrawBatch(&coneBatch);
            break;
            
        case 4:
            DrawBatch(&diskBatch);
            break;
            
    }
    
    //绘制完成一个图形,就把栈顶的矩阵可以Pop出去了,不要影响下一个图形的绘制
    modelViewMatrix.PopMatrix();
    
    //交换缓冲区
    glutSwapBuffers();
    
}

/**
 设置特殊键位(上下左右)
 */
void SpecialKeys(int key,int x,int y)
{
    
    //这里还是移动世界坐标系,因为画的东西都有太多的顶点了,移动物体的成本太高,效果是一样的。
    switch (key) {
            
        case GLUT_KEY_UP:
            objectFrame.RotateWorld(m3dDegToRad(-5.f), 1.f, 0.f, 0.f);
            break;
            
        case GLUT_KEY_DOWN:
            objectFrame.RotateWorld(m3dDegToRad(5.f), 1.f, 0.f, 0.f);
            break;
            
        case GLUT_KEY_LEFT:
            objectFrame.RotateWorld(m3dDegToRad(-5.f), 0.f, 1.f, 0.f);
            break;
            
        case GLUT_KEY_RIGHT:
            objectFrame.RotateWorld(m3dDegToRad(5.f), 0.f, 1.f, 0.f);
            break;
            
    }
    
    //移动完了要发送渲染信号
    glutPostRedisplay();
    
}

/**
 设置键盘键位(空格)
 */
void SpaceKey(unsigned char key,int x,int y)
{
    
    //拿空格键的ASCII码来判定点击的是键盘上的空格键
    if (key == 32) {
        step++;
        if(step > 4) step = 0;
    }
    
    switch (step) {
        case 0:
            glutSetWindowTitle("Sphere球");
            break;
        case 1:
            glutSetWindowTitle("Torus圆环");
            break;
        case 2:
            glutSetWindowTitle("Cylinder圆柱");
            break;
        case 3:
            glutSetWindowTitle("Cone锥");
            break;
        case 4:
            glutSetWindowTitle("Disk磁盘");
            break;
    }
    //发送重新渲染的信号
    glutPostRedisplay();
    
}

#pragma mark - main

int main(int argc,char *argv[])
{
    
    //设置工作目录和项目目录都在/Resouce下面,GLUT优先级做过,但是手动确保一定
    gltSetWorkingDirectory(argv[0]);
    
    //GLUT初始化
    glutInit(&argc, argv);
    
    //设置显示模式
    glutInitDisplayMode(GLUT_DOUBLE | GLUT_DEPTH | GLUT_RGBA | GLUT_STENCIL);
    
    //设置窗口大小
    glutInitWindowSize(600, 600);
    
    //创建窗口并命名
    glutCreateWindow("Sphere球");
    
    //注册函数
    //注册窗口
    glutReshapeFunc(ChangeSize);
    //注册渲染
    glutDisplayFunc(RenderScene);
    //注册特殊键位
    glutSpecialFunc(SpecialKeys);
    //注册空格键位
    glutKeyboardFunc(SpaceKey);
    
    //初始化Glew库
    GLenum status = glewInit();
    if (status != GLEW_OK) {
        printf("glew init error : %s \n",glewGetErrorString(status));
    }
    
    //设置渲染环境
    SetUpRC();
    
    //构建本地循环
    glutMainLoop();
    
    return 0;
    
}

显示效果我就不贴图了,有问题的话可以留言。

这上面void SetUpRC()void RenderScene()中的(1)(2)的变换,就是视变换的结果,说的通俗点,就是山不向你走来,你就向山走去。

上一篇 下一篇

猜你喜欢

热点阅读