Bootstrap

【OpenGL/C++】面向对象扩展——测试环境


在使用各种类对OpenGL的流程和API进行抽象和封装后,为了测试方便,还可以写一些类来对其进行进一步的封装。我写这个封装是为了方便测试使用。

封装的功能:

  • 固定渲染流程,无需重复书写
  • 外部调用Run来执行渲染代码
  • 继承类重写Init和RenderLoop函数来执行渲染初始化和渲染循环中的操作
  • 继承类重写UseCameraControl函数并修改返回值来确定是否开启摄像机操作

注意事项:

  • 考虑实现的简单性,摄像机操作功能写在了类里并使用静态单例来注册到glfw上,并没有分离出去,我个人在使用时直接在main函数中替换具体类名来切换不同的测试示例。
  • 类并不负责创建窗口等最开始的操作,下述有完整的代码展示。

RunBase类展示:

#pragma once
#include <GL/glew.h>
#include <GLFW/glfw3.h>

#include "Shader.h"
#include "Camera.h"

#include <iostream>

#include <glm/glm.hpp>

using std::cout;
using namespace glm;

class RunBase
{
public:
    //单例
    static RunBase* Instance;
    
    // Camera attributes
    Camera camera;
    bool keys[1024];
    GLfloat lastX = 400, lastY = 300;
    bool firstMouse = true;
    float deltaTime;

    // Light attributes
    glm::vec3 lightPos;

    //构造函数初始化参数
    RunBase()
    {
        camera.Position = { 0, 0, 3 };
        lightPos = { 1.2f, 1.0f, 2.0f };

        Instance = this;
    }

    //渲染流程,外部调用
    virtual void Run(GLFWwindow* window, float targetFrameTime)
    {
        deltaTime = targetFrameTime;

        //开启深度测试
        glEnable(GL_DEPTH_TEST);

        
        if (UseCameraControl())
        {
            //隐藏指针
            glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_DISABLED);

            //摄像机按键绑定
            glfwSetKeyCallback(window, key_callback);
            glfwSetCursorPosCallback(window, mouse_callback);
            glfwSetScrollCallback(window, scroll_callback);
        }

        Init(window, targetFrameTime);

        //在渲染循环中绘制指定的顶点数组对象,索引缓存,着色器
        while (!glfwWindowShouldClose(window))
        {
            CheckFrameTimeOut(targetFrameTime);

            RenderLoop(window, targetFrameTime);

            glfwSwapBuffers(window);

            if (UseCameraControl())
                DoMovement();
            glfwPollEvents();
        }
    }

    //渲染前的准备,类可重写
    virtual void Init(GLFWwindow* window, float targetFrameTime)
    {

    }

    //渲染中操作,类可重写
    virtual void RenderLoop(GLFWwindow* window, float targetFrameTime)
    {

    }

    //是否开启相机控制
    virtual bool UseCameraControl()
    {
        return true;
    }

    //移动函数
    virtual void DoMovement()
    {
        if (keys[GLFW_KEY_W])
            camera.ProcessKeyboard(FORWARD, deltaTime);
        if (keys[GLFW_KEY_S])
            camera.ProcessKeyboard(BACKWARD, deltaTime);
        if (keys[GLFW_KEY_A])
            camera.ProcessKeyboard(LEFT, deltaTime);
        if (keys[GLFW_KEY_D])
            camera.ProcessKeyboard(RIGHT, deltaTime);
    }

    void CheckFrameTimeOut(float targetFrameTime)
    {
        // 获取当前时间
        double currentTime = glfwGetTime();

        // 计算帧间隔
        static double lastFrameTime = currentTime;
        double frameDelta = currentTime - lastFrameTime;

        // 计算帧时间
        double frameTime = targetFrameTime - frameDelta;

        // 等待剩余的帧时间
        if (frameTime > 0.0)
        {
            glfwWaitEventsTimeout(frameTime);
        }
    }

    static void key_callback(GLFWwindow* window, int key, int scancode, int action, int mode)
    {
        if (key == GLFW_KEY_ESCAPE && action == GLFW_PRESS)
            glfwSetWindowShouldClose(window, GL_TRUE);

        if (key >= 0 && key < 1024)
        {
            if (action == GLFW_PRESS)
                Instance->keys[key] = true;
            else if (action == GLFW_RELEASE)
                Instance->keys[key] = false;
        }
    }

    static void mouse_callback(GLFWwindow* window, double xpos, double ypos)
    {
        if (Instance->firstMouse)
        {
            Instance->lastX = xpos;
            Instance->lastY = ypos;
            Instance->firstMouse = false;
        }

        GLfloat xoffset = xpos - Instance->lastX;
        GLfloat yoffset = Instance->lastY - ypos;  // Reversed since y-coordinates go from bottom to left

        Instance->lastX = xpos;
        Instance->lastY = ypos;

        Instance->camera.ProcessMouseMovement(xoffset, yoffset);
    }

    static void scroll_callback(GLFWwindow* window, double xoffset, double yoffset)
    {
        Instance->camera.ProcessMouseScroll(yoffset);
    }
};

RunBase* RunBase::Instance = nullptr;

RunBase继承类展示:

LearnOpenGL的教程示例,十个箱子在固定位置旋转,并有各自的贴图,在制作了RunBase类后,使用时只需要专注于教程和测试的图形学部分。无需重复复制和束缚在其他C++代码上。

#pragma once
#include <GL/glew.h>
#include <GLFW/glfw3.h>

#include "Renderer.h"
#include "VertexBuffer.h"
#include "VertexArray.h"
#include "VertexBufferLayout.h"
#include "Texture.h"
#include "Shader.h"
#include "Camera.h"

#include <iostream>
#include "RunBase.hpp"

#include <glm/gtc/type_ptr.hpp>

using std::cout;
using namespace glm;

class MyTest : public RunBase
{
public:
    VertexArray vao;
    VertexBuffer vbo;
    VertexBufferLayout layout;

    Shader shader;

    Renderer renderer;

    Texture containerTexture;

    glm::vec3* cubePositions;

    MyTest() : RunBase()
    {
        shader.Init("res/shaders/Learn2.shader");
    }

    virtual void Init(GLFWwindow* window, float targetFrameTime) override
    {
        GLfloat vertices[] = {
            -0.5f, -0.5f, -0.5f,  0.0f, 0.0f,
             0.5f, -0.5f, -0.5f,  1.0f, 0.0f,
             0.5f,  0.5f, -0.5f,  1.0f, 1.0f,
             0.5f,  0.5f, -0.5f,  1.0f, 1.0f,
            -0.5f,  0.5f, -0.5f,  0.0f, 1.0f,
            -0.5f, -0.5f, -0.5f,  0.0f, 0.0f,

            -0.5f, -0.5f,  0.5f,  0.0f, 0.0f,
             0.5f, -0.5f,  0.5f,  1.0f, 0.0f,
             0.5f,  0.5f,  0.5f,  1.0f, 1.0f,
             0.5f,  0.5f,  0.5f,  1.0f, 1.0f,
            -0.5f,  0.5f,  0.5f,  0.0f, 1.0f,
            -0.5f, -0.5f,  0.5f,  0.0f, 0.0f,

            -0.5f,  0.5f,  0.5f,  1.0f, 0.0f,
            -0.5f,  0.5f, -0.5f,  1.0f, 1.0f,
            -0.5f, -0.5f, -0.5f,  0.0f, 1.0f,
            -0.5f, -0.5f, -0.5f,  0.0f, 1.0f,
            -0.5f, -0.5f,  0.5f,  0.0f, 0.0f,
            -0.5f,  0.5f,  0.5f,  1.0f, 0.0f,

             0.5f,  0.5f,  0.5f,  1.0f, 0.0f,
             0.5f,  0.5f, -0.5f,  1.0f, 1.0f,
             0.5f, -0.5f, -0.5f,  0.0f, 1.0f,
             0.5f, -0.5f, -0.5f,  0.0f, 1.0f,
             0.5f, -0.5f,  0.5f,  0.0f, 0.0f,
             0.5f,  0.5f,  0.5f,  1.0f, 0.0f,

            -0.5f, -0.5f, -0.5f,  0.0f, 1.0f,
             0.5f, -0.5f, -0.5f,  1.0f, 1.0f,
             0.5f, -0.5f,  0.5f,  1.0f, 0.0f,
             0.5f, -0.5f,  0.5f,  1.0f, 0.0f,
            -0.5f, -0.5f,  0.5f,  0.0f, 0.0f,
            -0.5f, -0.5f, -0.5f,  0.0f, 1.0f,

            -0.5f,  0.5f, -0.5f,  0.0f, 1.0f,
             0.5f,  0.5f, -0.5f,  1.0f, 1.0f,
             0.5f,  0.5f,  0.5f,  1.0f, 0.0f,
             0.5f,  0.5f,  0.5f,  1.0f, 0.0f,
            -0.5f,  0.5f,  0.5f,  0.0f, 0.0f,
            -0.5f,  0.5f, -0.5f,  0.0f, 1.0f
        };

        cubePositions = new vec3[]{
            glm::vec3(0.0f,  0.0f,  0.0f),
            glm::vec3(2.0f,  5.0f, -15.0f),
            glm::vec3(-1.5f, -2.2f, -2.5f),
            glm::vec3(-3.8f, -2.0f, -12.3f),
            glm::vec3(2.4f, -0.4f, -3.5f),
            glm::vec3(-1.7f,  3.0f, -7.5f),
            glm::vec3(1.3f, -2.0f, -2.5f),
            glm::vec3(1.5f,  2.0f, -2.5f),
            glm::vec3(1.5f,  0.2f, -1.5f),
            glm::vec3(-1.3f,  1.0f, -1.5f)
        };
		vbo.Init(vertices, sizeof(vertices));
        
        //创建顶点缓存布局,并链接到顶点数组上
        layout.Push<float>(3);
        layout.Push<float>(2);

        //缓存和布局绑定到vao上
        vao.AddBuffer(vbo, layout);

        //设置纹理
        containerTexture.Init("res/textures/container2.png");
        containerTexture.Bind();
        shader.Bind();
        shader.SetUniform1i("u_Texture", 0);
    }

    virtual void RenderLoop(GLFWwindow* window, float targetFrameTime) override
    {
        renderer.Clear();

        glm::mat4 view(1), proj(1);

        //视图矩阵
        view = camera.GetViewMatrix();
        shader.SetUniformMat4f("view", glm::value_ptr(view));
		//投影矩阵
        proj = glm::perspective<GLfloat>(camera.Zoom, 640 / 480, .1f, 100.f);
        shader.SetUniformMat4f("projection", glm::value_ptr(proj));

        //模型矩阵(模型位置)
        for (GLuint i = 0; i < 10; i++)
        {
            glm::mat4 model(1);
            model = glm::translate(model, cubePositions[i]);
            GLfloat angle = 20.0f * i;
            model = glm::rotate(model, angle + ((GLfloat)glfwGetTime() * (i % 2 ? 1 : -1)), glm::vec3(1.0f, 0.3f, 0.5f));
            shader.SetUniformMat4f("model", glm::value_ptr(model));

            renderer.Draw(vao, 36, shader);
        }
    }

    virtual bool UseCameraControl() override
    {
        return true;
    }
};

Main函数内容(使用示例):

在制作该类前,我一直直接通过使用全局的函数名来调用不同的渲染代码,后来在使用glfw的按键绑定相关代码时出现了重定义的错误,后来想直接把这些东西放在类里,最后干脆再封装一下渲染流程,最后就成了这个RunBase类。

#define GLEW_STATIC

#include <GL/glew.h>
#include <GLFW/glfw3.h>
#include <iostream>

#include "RunBase.hpp"
#include "Test1.hpp"
#include "learn1.hpp"
#include "learn2.hpp"
#include "light_learn_1.hpp"
#include "MyTest.hpp"

using std::cout;
using std::endl;

#define TargetFrameRate 60


int main(void)
{
	/* Initialize the library */
	if (!glfwInit())
		return -1;


	glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
	glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
	glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);


	/* Create a windowed mode window and its OpenGL context */
	GLFWwindow* window = glfwCreateWindow(640, 480, "Hello World", NULL, NULL);
	if (!window)
	{
		glfwTerminate();
		return -1;
	}

	/* Make the window's context current */
	glfwMakeContextCurrent(window);

	//这样做能让GLEW在管理OpenGL的函数指针时更多地使用现代化的技术
	glewExperimental = GL_TRUE;

	//初始化GLEW(要在初始化了OpenGL上下文之后)
	if (glewInit() != GLEW_OK)
		return -1;

	//目标帧时间
	const double targetFrameTime = 1.0 / TargetFrameRate;

	//课程代码
	//Test1(window);
	//Learn1(window, targetFrameTime);
	//Learn2(window, targetFrameTime);

	RunBase* runable = new MyTest();
	runable->Run(window, targetFrameTime);

	glfwTerminate();
	return 0;
}
;