main.cpp
上传用户:arsena_zhu
上传日期:2022-07-12
资源大小:399k
文件大小:11k
源码类别:

OpenGL

开发平台:

Visual C++

  1. #include "main.h"
  2. // Function prototypes...
  3. bool InitializeGL();
  4. void RenderScene();
  5. void GetInput();
  6. void Shutdown();
  7. // Width and height of the screen.
  8. #define ScreenWidth 1024
  9. #define ScreenHeight 750
  10. // Device context.
  11. HDC g_HDC;
  12. float spd = 0.1;
  13. CTerrain terrain;
  14. CTerrainData tdata;
  15. CTerrainEdit tedit;
  16. CCamera cam;
  17. void SetupPixelFormat(HDC hDC)
  18. {
  19. int nPixelFormat;
  20. static PIXELFORMATDESCRIPTOR pfd = {
  21. sizeof(PIXELFORMATDESCRIPTOR), // size of structure.
  22. 1,  // always 1.
  23. PFD_DRAW_TO_WINDOW |  // support window
  24. PFD_SUPPORT_OPENGL |  // support OpenGl
  25. PFD_DOUBLEBUFFER,  // support double buffering
  26. PFD_TYPE_RGBA,  // support RGBA
  27. 16, // bit color mode
  28. 0, 0, 0, 0, 0, 0,  // ignore color bits
  29. 0,  // no alpha buffer
  30. 0,  // ignore shift bit
  31. 0,  // no accumulation buffer
  32. 0, 0, 0, 0,  // ignore accumulation bits.
  33. 16, // number of depth buffer bits.
  34. 0,  // number of stencil buffer bits.
  35. 0,  // 0 means no auxiliary buffer
  36. PFD_MAIN_PLANE, // The main drawing plane
  37. 0,  // this is reserved
  38. 0, 0, 0 };   // layer masks ignored.
  39. // This chooses the best pixel format and returns index.
  40. nPixelFormat = ChoosePixelFormat(hDC, &pfd);
  41. // This set pixel format to device context.
  42. SetPixelFormat(hDC, nPixelFormat, &pfd);
  43. }
  44. LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
  45. {
  46. static HGLRC hRC;  // Rendering context.
  47. static HDC hDC; // Device context.
  48. int width, height; // The window width and height.
  49. switch(message)
  50. {
  51. case WM_CREATE: // Windows creation.
  52. hDC = GetDC(hwnd); // This gets the device context for our window.
  53. g_HDC = hDC; // Assigns the global device context to this one.
  54. SetupPixelFormat(hDC);   // Call the pixel format function.
  55. hRC = wglCreateContext(hDC);   // Creates the rendering context.
  56. wglMakeCurrent(hDC, hRC);   // Makes the rendering context.
  57. return 0;
  58. break;
  59. case WM_CLOSE:  // Close message.
  60. case WM_DESTROY:
  61. wglMakeCurrent(hDC, NULL);
  62. wglDeleteContext(hRC);   // Deletes the rendering context.
  63. PostQuitMessage(0);   // Says close the program.
  64. return 0;
  65. break;
  66. case WM_SIZE:   // re-size message.
  67. height = HIWORD(lParam); // This gets the height of the window.
  68. width = LOWORD(lParam);  // This gets the width of the window.
  69. if(height==0)   // we don't want it to be possible for a
  70. {   // height of 0.  If it is 0 me make it 1.
  71. height = 1;
  72. }
  73. glViewport(0, 0, width, height); // resets the viewport to new dimensions.
  74. glMatrixMode(GL_PROJECTION);   // Sets the projection matrix.
  75. glLoadIdentity();  // Reset the modelview matrix.
  76. // Calculate the aspect ratio of the window.
  77. gluPerspective(45.0f, (GLfloat)width/(GLfloat)height, 0.1f, 2000.0f);
  78. glMatrixMode(GL_MODELVIEW); // Sets the projection matrix.
  79. glLoadIdentity();  // Reset the modelview matrix.
  80. return 0;
  81. break;
  82. case WM_KEYDOWN:
  83. switch(wParam)
  84. // Close the app if the user presses escape.
  85. case VK_ESCAPE:
  86. PostQuitMessage(0);
  87. break;
  88. default:
  89. break;
  90. }
  91. break;
  92. default:
  93. break;
  94. }
  95. // What this does is pass all of the unhandled messages to DefWindowProc.
  96. return (DefWindowProc(hwnd, message, wParam, lParam));
  97. }
  98. int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd)
  99. {
  100. MSG msg;  // A message variable.
  101. WNDCLASSEX windowClass;  // Your Window class.
  102. HWND hwnd;   // The Window handle.
  103. bool isFinished;   // True then exit.
  104. // This is the Window class.
  105. windowClass.cbSize = sizeof(WNDCLASSEX);   // size of the WNDCLASSEX structure.
  106. windowClass.style = CS_HREDRAW | CS_VREDRAW;  // style of the window.
  107. windowClass.lpfnWndProc = WndProc;   // Address to the windows procedure.
  108. windowClass.cbClsExtra = 0; // Extra class information.
  109. windowClass.cbWndExtra = 0; // Extra window information.
  110. windowClass.hInstance = hInstance;   // Handle of application Instance.
  111. // Handle of application Icon.
  112. windowClass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
  113. windowClass.hCursor = LoadCursor(NULL, IDC_ARROW);// mouse cursor
  114. windowClass.hbrBackground = NULL; // background color.
  115. windowClass.lpszMenuName = NULL;  // name of the main menu.
  116. windowClass.lpszClassName = "UltimateGameProgrammingClass";// window class name.
  117. // icon when minimized.
  118. windowClass.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
  119. // You must register you class with Windows.
  120. if(!RegisterClassEx(&windowClass)) return 0;
  121. // Create window...
  122. hwnd = CreateWindowEx(0,// The extended window style.
  123. "UltimateGameProgrammingClass",// window Class name.
  124. "Octrees Demo - By the Programming Ace.",// window name.
  125. WS_OVERLAPPEDWINDOW | WS_VISIBLE |// The window style.
  126. WS_SYSMENU | WS_CLIPCHILDREN |// window style.
  127. WS_CLIPSIBLINGS,// window style.
  128. 0, 0,// window x, y coordinate.
  129. ScreenWidth, ScreenHeight,// window width and height.
  130. NULL,// handle to parent window.
  131. NULL,// menu.
  132. hInstance,// handle to app instance.
  133. 0); // pointer to window creation data.
  134. // If there was an error with creating the window, then close the program.
  135. if(!hwnd) return 0;
  136. ShowWindow(hwnd, SW_SHOW); // This shows the window.
  137. UpdateWindow(hwnd);   // This forces a paint message.
  138. isFinished = false;   // False = running, True = not running.
  139. // If initialize fail (return false), then we don't want the program to run.
  140. if(!InitializeGL()) isFinished = true;
  141. // This is the messsage loop.
  142. while(!isFinished)
  143. {
  144. if(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
  145. {
  146. // If a quit message then quit the app.
  147. if(msg.message == WM_QUIT)
  148. {
  149. isFinished = true;
  150. }
  151. TranslateMessage(&msg);
  152. DispatchMessage(&msg);
  153. }
  154. else
  155. {
  156. RenderScene();
  157. }
  158. }
  159. // Shutdown (free) all resources we used up.
  160. Shutdown();
  161. return (int)msg.wParam;
  162. }
  163. bool InitializeGL()
  164. {
  165. glClearColor(0.6f, 0.6f, 1.0f, 1.0f);
  166. glShadeModel(GL_SMOOTH);
  167. glEnable(GL_DEPTH_TEST);
  168. glEnable(GL_TEXTURE_2D);
  169. glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);
  170. if (!tdata.Init(&terrain)) return false;
  171. tedit.Init(&terrain);
  172. if (!tdata.Generate("grass.bmp", 32, "lowest.tga", "low.tga", "high.tga", "highest.tga", "detail.tga")) return false;
  173. // if (!tdata.Generate("grass.bmp", 32, "lowest.tga", "low.tga", "high.tga", "highest.tga", "detail.tga")) return false;
  174. cam.SetCamera(50.0f, 50.0f, 10.0f, 50.0f, 50.0f, 11.0f, 0, 1, 0);
  175. return true;
  176. }
  177. void RenderScene()
  178. {
  179. glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); // Clears the screen.
  180. glLoadIdentity();  //Reset modelview matrix for new frame.
  181. GetInput();
  182. // Set the camera.
  183. gluLookAt(cam.xPos, cam.yPos, cam.zPos,
  184.  cam.xView, cam.yView, cam.zView,
  185.  cam.xUp, cam.yUp, cam.zUp);
  186. tdata.RenderAll();
  187. SwapBuffers(g_HDC);
  188. }
  189. void GetInput()
  190. {
  191. // Checking for input.  We move the camera based on the button(s) that were pressed.
  192. if(GetKeyState('C') & 0x80) terrain.ConvertMesh();
  193. if(GetKeyState('G') & 0x80)
  194. {
  195. // terrain.CreateTexCoords();
  196. tdata.InitTextures();
  197. }
  198. if(GetKeyState('R') & 0x80)
  199. tedit.DoHill((int)cam.xPos, (int)cam.zPos, 30, 0.01);
  200. if(GetKeyState('F') & 0x80)
  201. tedit.DoHill((int)cam.xPos, (int)cam.zPos, 30, -0.01);
  202. // terrain.CreateTerrainMesh();
  203. if(GetKeyState('I') & 0x80) tedit.GenerateIsland(64, 64, 64, 64, 100, 20, 3);
  204. if(GetKeyState('T') & 0x80) tedit.GenerateTerrain(30, 10, 2);
  205. if(GetKeyState('N') & 0x80) tedit.Normalize(1);
  206. if(GetKeyState('V') & 0x80) tedit.Flatten(1.1);
  207. if(GetKeyState('M') & 0x80) tedit.Multiply(1.2);
  208. if(GetKeyState('B') & 0x80) tedit.Multiply(0.8);
  209. if(GetKeyState('0') & 0x80) tedit.Zero();
  210. if(GetKeyState('8') & 0x80) terrain.SaveMap("grass.bmp");
  211. if(GetKeyState('9') & 0x80) terrain.LoadMap("grass.bmp");
  212. if(GetKeyState('O') & 0x80) { tedit.Increase(1); cam.UpDown(1); }
  213. if(GetKeyState('L') & 0x80) { tedit.Increase(-1); cam.UpDown(-1); }
  214. float y;
  215. y = terrain.GetRealHeight(cam.xPos, cam.zPos);
  216. if (y != 0)
  217. {
  218. float change = y + 0.3 - cam.yPos;
  219. cam.yPos += change;
  220. cam.yView += change;
  221. }
  222. if(GetKeyState('E') & 0x80)
  223. {
  224. // erri("x, z, y", cam.xPos*1000, cam.zPos*1000, y*1000);
  225. erri("x, z, y", cam.xPos, cam.zPos, y);
  226. }
  227. if(GetKeyState('W') & 0x80) cam.MoveCamera(spd);
  228. if(GetKeyState('S') & 0x80) cam.MoveCamera(-spd);
  229. if(GetKeyState('A') & 0x80) cam.StrafeCam(-spd);
  230. if(GetKeyState('D') & 0x80) cam.StrafeCam(spd);
  231. if(GetKeyState('1') & 0x80) spd = 0.001;
  232. if(GetKeyState('2') & 0x80) spd = 0.01;
  233. if(GetKeyState('3') & 0x80) spd = 0.1;
  234. if(GetKeyState('4') & 0x80) spd = 0.5;
  235. if(GetKeyState('5') & 0x80) spd = 1;
  236. if(GetKeyState('6') & 0x80) spd = 3;
  237. if(GetKeyState('7') & 0x80) spd = 10;
  238. if(GetKeyState('J') & 0x80) tdata.SetMode(GL_LINES);
  239. if(GetKeyState('K') & 0x80) tdata.SetMode(GL_TRIANGLES);
  240. // Here we get the position of the mouse, rotate the camera based on its movements,
  241. // then set the mouse position to the middle of the screen.
  242. POINT mousePos;
  243. GetCursorPos(&mousePos);
  244. ShowCursor(0);
  245. cam.RotateByMouse(mousePos.x, mousePos.y, (ScreenWidth >> 1), (ScreenHeight >> 1));
  246. cam.RotateByMouse(mousePos.x, mousePos.y, (ScreenWidth >> 1), (ScreenHeight >> 1));
  247. SetCursorPos((ScreenWidth >> 1), (ScreenHeight >> 1));
  248. }
  249. void Shutdown()
  250. {
  251. tdata.Destroy();
  252. }
  253. // Recap:
  254. // Rendering a mesh using octrees is a lot easier than I am sure many
  255. // people thought.  All we are doing is creating a bounding box around
  256. // the entire terrain.  Then we break that box up into 8 equally sized
  257. // cubes within the box.  We then determine which vertices fall within
  258. // with cubes.  Depending of if we reached the max recursion level or
  259. // if we reached the smallest number of triangles allowed in each cube
  260. // will depend on if we break each cube into another, smaller set of 8
  261. // or if we are done creating the tree.  Technically you can put any mesh
  262. // in a octree, not just terrains.  It might not make sense for character
  263. // models but you can if you want.
  264. // Copyright October 2004
  265. // All Rights Reserved!
  266. // Allen Sherrod
  267. // ProgrammingAce@UltimateGameProgramming.com
  268. // www.UltimateGameProgramming.com