前面已经建立了 OpenGL 框架,加载了 3D 模型,但是还没有在场景中漫游的功能。为了展示 3D 模型,我只是简单地利用变换视图矩阵的方式使模型在视野中旋转。同时,之前的程序连最简单的改变窗口大小的功能都没有,不能放大窗口而观察模型的更多细节。从这一节开始,我要实现在场景中漫游的功能。

功能的设计很简单,就像所有的 FPS 游戏一样,按A W S D进行前进后退和左右移动,使用鼠标控制方向,为了简单起见,暂时只考虑左右转动,不实现上下转动的功能。

改变窗口大小的功能很简单,添加一个 static 的 onWindowSize() 函数就可以了,然后调用 glfwSetWindowSizeCallback() 注册这个回调函数。添加这个功能后,我们就可以把窗口放大到全屏了,如下图:

前面一直使用的是线框模型,这里可以设置按M键来切换线框模式和填充模式。这里可以先编写一个 onKey() 方法,然后使用 glfwSetSetKeyCallback() 来设置回调。

这时不能使用 glfwSetSetKeyCallback() 来设置回调,因为 onKey() 方法只在每次按键的时候调用一次,即使按着键不动,它也不会连续调用,不符合我们的要求。这时,需要在每一帧的绘图函数里面调用 processInput() 方法,并在 processInput() 方法里面调用 glfwGetKey() 来实现这个效果。

另外,我们的视图矩阵要改了。我们可以在 App 类里面设置三个变量,cameraPosition、cameraFront、cameraUp,分别代表摄像机的位置、前方、上方,然后使用 GLM 的 lookAt() 函数来设置视图矩阵。

根据 3D 场景的复杂程度不同,其渲染速度也会不同,为了保证我们移动速度的一致性,我这里顺便搞一个计算帧率的功能。

使用 GLFW 的鼠标回调函数,可以很方便地得到鼠标的 X 坐标和 Y 坐标,因此实现左右转动视角的功能非常方便。我没有使用很复杂的三角函数计算,只是利用 GLM 的 rotate() 函数对 cameraFront 向量进行旋转就可以了。我们还可以充分利用 GLFW 捕获鼠标功能,设计为在窗口中点击鼠标后捕获鼠标指针,按ESC键后释放鼠标捕获,只有在捕获鼠标指针的状态下才能够左右旋转视角。这时主要用到的 API 是 glfwSetCursorPosCallback() 和 glfwSetMouseButtonCallback()。

经过修改后的 app.hpp 完整代码如下:

  1. #ifndef __APP_HPP__
  2. #define __APP_HPP__
  3. #include <GL/glew.h>
  4. #include <GLFW/glfw3.h>
  5. #include <iostream>
  6. #include <glm/glm.hpp>
  7. #include <glm/gtc/matrix_transform.hpp>
  8. class App
  9. {
  10. private:
  11. const int SCR_WIDTH = 1920;
  12. const int SCR_HEIGHT = 1080;
  13. public:
  14. static App *the_app;
  15. float aspect;
  16. glm::vec3 cameraPosition;
  17. glm::vec3 cameraFront;
  18. glm::vec3 cameraUp;
  19. float cameraSpeed;
  20. double timeLastFrame;
  21. double timeThisFrame;
  22. double timeAccumulate;
  23. int countFrames;
  24. bool showFps;
  25. bool firstMouse;
  26. double lastX;
  27. bool captureCursor;
  28. App()
  29. {
  30. aspect = (float)SCR_WIDTH / (float)SCR_HEIGHT;
  31. cameraPosition = glm::vec3(0.0f, 0.0f, 0.0f);
  32. cameraFront = glm::vec3(0.0f, 0.0f, -1.0f);
  33. cameraUp = glm::vec3(0.0f, 1.0f, 0.0f);
  34. firstMouse = true;
  35. }
  36. static void onWindowSize(GLFWwindow *window, int width, int height)
  37. {
  38. glViewport(0, 0, width, height);
  39. the_app->aspect = (float)width / (float)height;
  40. }
  41. static void onKey(GLFWwindow *window, int key, int scancode, int action, int mods)
  42. {
  43. if (action == GLFW_PRESS)
  44. {
  45. switch (key)
  46. {
  47. case GLFW_KEY_M: //切换线框模式和填充模式
  48. {
  49. static GLenum mode = GL_FILL;
  50. mode = (mode == GL_FILL ? GL_LINE : GL_FILL);
  51. glPolygonMode(GL_FRONT_AND_BACK, mode);
  52. return;
  53. }
  54. case GLFW_KEY_ESCAPE: //停止鼠标捕获,主要是应付鼠标被捕获的情况
  55. {
  56. if (the_app->captureCursor)
  57. {
  58. glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
  59. the_app->captureCursor = false;
  60. }
  61. return;
  62. }
  63. case GLFW_KEY_F: //打开和关闭输出fps的功能,输出到控制台
  64. {
  65. the_app->showFps = (the_app->showFps == false ? true : false);
  66. return;
  67. }
  68. }
  69. }
  70. }
  71. virtual void processInput(GLFWwindow *window)
  72. {
  73. if (glfwGetKey(window, GLFW_KEY_W) == GLFW_PRESS)
  74. {
  75. cameraPosition += cameraSpeed * cameraFront;
  76. }
  77. if (glfwGetKey(window, GLFW_KEY_S) == GLFW_PRESS)
  78. {
  79. cameraPosition -= cameraSpeed * cameraFront;
  80. }
  81. if (glfwGetKey(window, GLFW_KEY_A) == GLFW_PRESS)
  82. {
  83. cameraPosition += cameraSpeed * glm::normalize(glm::cross(cameraFront, cameraUp));
  84. }
  85. if (glfwGetKey(window, GLFW_KEY_D) == GLFW_PRESS)
  86. {
  87. cameraPosition -= cameraSpeed * glm::normalize(glm::cross(cameraFront, cameraUp));
  88. }
  89. }
  90. static void onMouseMove(GLFWwindow *window, double xpos, double ypos)
  91. {
  92. //std::cout << "xpos:" << xpos << " ypos:" << ypos << std::endl;
  93. if (!the_app->captureCursor)
  94. {
  95. return;
  96. }
  97. if (the_app->firstMouse)
  98. {
  99. the_app->lastX = xpos;
  100. the_app->firstMouse = false;
  101. return;
  102. }
  103. double xoffset = xpos - the_app->lastX;
  104. the_app->lastX = xpos;
  105. double sensitivity = 0.005f; //灵敏度
  106. xoffset *= sensitivity;
  107. glm::mat4 I(1.0f);
  108. glm::vec3 Y(0.0f, 1.0f, 0.0f);
  109. the_app->cameraFront = glm::vec3(glm::vec4(the_app->cameraFront, 1.0f) * glm::rotate(I, (float)xoffset, Y));
  110. }
  111. static void onMouseButton(GLFWwindow *window, int button, int action, int mods)
  112. {
  113. if (action == GLFW_PRESS)
  114. {
  115. switch (button)
  116. {
  117. case GLFW_MOUSE_BUTTON_LEFT:
  118. glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_DISABLED);
  119. the_app->captureCursor = true;
  120. return;
  121. }
  122. }
  123. }
  124. virtual void init()
  125. {
  126. }
  127. virtual void display()
  128. {
  129. }
  130. virtual void run(App *app)
  131. {
  132. if (the_app != NULL)
  133. { //同一时刻,只能有一个App运行
  134. std::cerr << "The the_app is already run." << std::endl;
  135. return;
  136. }
  137. the_app = app;
  138. glfwInit();
  139. GLFWwindow *window = glfwCreateWindow(SCR_WIDTH, SCR_HEIGHT, "StudyOpenGL", NULL, NULL);
  140. if (window == NULL)
  141. {
  142. std::cerr << "Failed to create GLFW window" << std::endl;
  143. glfwTerminate();
  144. return;
  145. }
  146. glfwMakeContextCurrent(window);
  147. glfwSetWindowSizeCallback(window, onWindowSize);
  148. glfwSetKeyCallback(window, onKey);
  149. glfwSetCursorPosCallback(window, onMouseMove);
  150. glfwSetMouseButtonCallback(window, onMouseButton);
  151. if (glewInit() != GLEW_OK)
  152. {
  153. std::cerr << "Failed to initalize GLEW" << std::endl;
  154. return;
  155. }
  156. init(); //Init主要是用来创建VAO、VBO等,并准备要各种数据
  157. while (!glfwWindowShouldClose(window))
  158. {
  159. display(); //这里才是渲染图形的主战场
  160. timeThisFrame = glfwGetTime();
  161. //记录帧渲染之后的时间,并计算帧率,如果输出帧率,则每一秒输出一次(主要是标准输出太慢),同时计算cameraSpeed;
  162. double timeInterval = timeThisFrame - timeLastFrame;
  163. timeLastFrame = timeThisFrame;
  164. if (showFps)
  165. {
  166. if (timeAccumulate < 1.0)
  167. {
  168. countFrames++;
  169. timeAccumulate += timeInterval;
  170. }
  171. else
  172. {
  173. std::cout << "FPS: " << countFrames << std::endl;
  174. countFrames = 0;
  175. timeAccumulate = 0;
  176. }
  177. }
  178. cameraSpeed = 2.5f * (float)timeInterval;
  179. glfwSwapBuffers(window);
  180. processInput(window);
  181. glfwPollEvents();
  182. }
  183. glfwDestroyWindow(window);
  184. glfwTerminate();
  185. return;
  186. }
  187. };
  188. App *App::the_app = NULL;
  189. #define DECLARE_MAIN(a) \
  190. int main(int argc, const char **argv) \
  191. { \
  192. a *app = new a; \
  193. app->run(app); \
  194. delete app; \
  195. return 0; \
  196. }
  197. #endif

然后,我们的 WanderInScene.cpp 的完整内容如下:

  1. #include "../include/app.hpp"
  2. #include "../include/shader.hpp"
  3. #include "../include/model.hpp"
  4. #include <glm/glm.hpp>
  5. #include <glm/gtc/matrix_transform.hpp>
  6. class MyApp : public App {
  7. private:
  8. const GLfloat clearColor[4] = {0.2f, 0.3f, 0.3f, 1.0f};
  9. Model lita;
  10. Shader* simpleShader;
  11. public:
  12. void init(){
  13. ShaderInfo shaders[] = {
  14. {GL_VERTEX_SHADER, "simpleShader.vert"},
  15. {GL_FRAGMENT_SHADER, "simpleShader.frag"},
  16. {GL_NONE, ""}
  17. };
  18. simpleShader = new Shader(shaders);
  19. lita.loadModel("lita.obj");
  20. glEnable(GL_DEPTH_TEST);
  21. glDepthFunc(GL_LEQUAL);
  22. glPolygonMode( GL_FRONT_AND_BACK, GL_FILL );
  23. }
  24. void display(){
  25. glClearBufferfv(GL_COLOR, 0, clearColor);
  26. glClear(GL_DEPTH_BUFFER_BIT);
  27. glm::mat4 I(1.0f);
  28. glm::vec3 X(1.0f, 0.0f, 0.0f);
  29. glm::vec3 Y(0.0f, 1.0f, 0.0f);
  30. glm::vec3 Z(0.0f, 0.0f, 1.0f);
  31. glm::mat4 view_matrix = glm::lookAt(cameraPosition, cameraPosition + cameraFront, cameraUp);
  32. glm::mat4 projection_matrix = glm::perspective(glm::radians(45.0f), aspect, 1.0f, 100.0f);
  33. glm::mat4 allis_model_matrix = glm::translate(I, glm::vec3(0.0f, 2.0f, 0.0f))
  34. * glm::scale(I, glm::vec3(0.8f, 0.8f, 0.8f)) * glm::rotate(I, glm::radians(0.0f), X);
  35. simpleShader->setModelMatrix(allis_model_matrix);
  36. simpleShader->setViewMatrix(view_matrix);
  37. simpleShader->setProjectionMatrix(projection_matrix);
  38. simpleShader->setCurrent();
  39. lita.render();
  40. }
  41. ~MyApp(){
  42. if(simpleShader != NULL){
  43. delete simpleShader;
  44. }
  45. }
  46. };
  47. DECLARE_MAIN(MyApp)

编译运行的命令如下:

  1. g++ -o WanderInScene WanderInScene.cpp -lGL -lglfw -lGLEW -lassimp
  2. ./WanderInScene

就可以看到程序运行的效果了,我们可以很方便地从不同角度、不同距离观察 3D 模型,如下图:

该随笔由京山游侠在2021年08月09日发布于博客园,引用请注明出处,转载或出版请联系博主。QQ邮箱:1841079@qq.com

版权声明:本文为youxia原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/youxia/p/cg006.html