outdated: 28.Bezier Patches / Fullscreen Fix
Bézier curve经常应用于计算机图形学邻域。
三次Bézier curve演示:
p为点的坐标。
一次Bézier curve:
二次Bézier curve:
三次Bézier curve:
每次对每个点进行处理。未处理时:
下面为代码。同样修改部分位于双行星号内。
1 /******************************************************************************************************************************************/ 2 /******************************************************************************************************************************************/ 3 #include <windows.h> 4 #include <math.h> 5 #include <stdio.h> 6 #include <gl/glew.h> 7 #include <GL\glut.h> 8 #include <GL/GLUAX.H> 9 #pragma comment(lib, "legacy_stdio_definitions.lib") 10 11 /* 12 * Every OpenGL program is linked to a Rendering Context. 13 * A Rendering Context is what links OpenGL calls to the Device Context. 14 * In order for your program to draw to a Window you need to create a Device Context. 15 * The DC connects the Window to the GDI (Graphics Device Interface). 16 */ 17 18 HGLRC hRC = NULL; // Permanent rendering context 19 HDC hDC = NULL; // Private GDI device context 20 HWND hWnd = NULL; // Holds our window handle 21 HINSTANCE hInstance; // Holds the instance of the application 22 23 /* 24 * It's important to make this global so that each procedure knows if 25 * the program is running in fullscreen mode or not. 26 */ 27 28 bool keys[256]; // Array used for the keyboard routine 29 bool active = TRUE; // Window active flag set to TRUE by default 30 bool fullscreen = TRUE; // Fullscreen flag set to fullscreen mode by default 31 32 typedef struct point_3d { // 3-dimensional 33 double x, y, z; 34 } POINT_3D; 35 36 typedef struct bpatch { 37 POINT_3D anchors[4][4]; // 4X4 grid of anchors ponits 38 GLuint dlBPatch; // Display list for Bezier patch 39 GLuint texture; // Texture for the patch 40 } BEZIER_PATCH; 41 42 DEVMODE DMsaved; // Saves the previous screen settings 43 44 GLfloat rotx = 0.0f; 45 GLfloat rotz = 0.0f; 46 BEZIER_PATCH mybezier; 47 bool showCPoints = true; // Toggles displaying the control point grid 48 49 int divs = 7; // Number of intrapolations (controls poly resolution) 50 51 LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM); // Declaration for WndProc 52 53 // Adds 2 points 54 POINT_3D pointAdd(POINT_3D p, POINT_3D q) 55 { 56 p.x += q.x; p.y += q.y; p.z += q.z; 57 return p; 58 } 59 60 // Multiplies A point and a constant 61 POINT_3D pointTimes(double c, POINT_3D p) 62 { 63 p.x *= c; p.y *= c; p.z *= c; 64 return p; 65 } 66 67 // Function for quick point creation 68 POINT_3D makePoint(double a, double b, double c) 69 { 70 POINT_3D p; 71 p.x = a; p.y = b; p.z = c; 72 return p; 73 } 74 // Calculates 3rd degree polynomial based on array of 4 points 75 // and a single variable (u) which is generally between 0 and 1 76 POINT_3D Bernstein(float u, POINT_3D* p) 77 { 78 POINT_3D a, b, c, d, r; 79 80 a = pointTimes(pow(u, 3), p[0]); 81 b = pointTimes(3 * pow(u, 2) * (1 - u), p[1]); 82 c = pointTimes(3 * u * pow((1 - u), 2), p[2]); 83 d = pointTimes(pow((1 - u), 3), p[3]); 84 85 r = pointAdd(pointAdd(a, b), pointAdd(c, d)); 86 return r; 87 } 88 89 // Generates a display list based on the data in the patch and the number of divisions 90 GLuint genBezier(BEZIER_PATCH patch, int divs) 91 { 92 float px, py, pyold; 93 GLuint drawlist = glGenLists(1); 94 POINT_3D temp[4]; 95 POINT_3D* last = (POINT_3D*)malloc(sizeof(POINT_3D) * (divs + 1)); 96 97 if (patch.dlBPatch != NULL) { // Get rid of any old display lists 98 glDeleteLists(patch.dlBPatch, 1); 99 } 100 temp[0] = patch.anchors[0][3]; // The first derived curve 101 temp[1] = patch.anchors[1][3]; 102 temp[2] = patch.anchors[2][3]; 103 temp[3] = patch.anchors[3][3]; 104 105 for (int v = 0; v <= divs; ++v) { // Create the first line of points 106 px = ((float)v) / ((float)divs); 107 // Use the 4 points from the derived curve to calculate the points along that curve 108 last[v] = Bernstein(px, temp); 109 } 110 glNewList(drawlist, GL_COMPILE); 111 glBindTexture(GL_TEXTURE_2D, patch.texture); 112 113 for (int u = 1; u <= divs; ++u) { 114 py = ((float)u) / ((float)divs); 115 pyold = ((float)u - 1.0f) / ((float)divs); 116 temp[0] = Bernstein(py, patch.anchors[0]); // Calculate new Bezier 117 temp[1] = Bernstein(py, patch.anchors[1]); 118 temp[2] = Bernstein(py, patch.anchors[2]); 119 temp[3] = Bernstein(py, patch.anchors[3]); 120 121 int a = 0; 122 glBegin(GL_TRIANGLE_STRIP); 123 for (int v = 0; v <= divs; ++v) { 124 px = ((float)v) / ((float)divs); 125 glTexCoord2f(pyold + a++, px + a++); 126 glVertex3d(last[v].x, last[v].y, last[v].z); // Old point 127 128 last[v] = Bernstein(px, temp); 129 glTexCoord2f(py + a++, px + a++); 130 glVertex3d(last[v].x, last[v].y, last[v].z); // New point 131 } 132 glEnd(); 133 } 134 glEndList(); 135 136 free(last); 137 return drawlist; 138 } 139 140 void initBezier(void) 141 { 142 mybezier.anchors[0][0] = makePoint(-0.75f, -0.75f, -0.5f); // Set the Bezier vertices 143 mybezier.anchors[0][1] = makePoint(-0.25f, -0.75f, 0.0f); 144 mybezier.anchors[0][2] = makePoint(0.25f, -0.75f, 0.0f); 145 mybezier.anchors[0][3] = makePoint(0.75f, -0.75f, -0.5f); 146 mybezier.anchors[1][0] = makePoint(-0.75f, -0.25f, -0.75f); 147 mybezier.anchors[1][1] = makePoint(-0.25f, -0.25f, 0.5f); 148 mybezier.anchors[1][2] = makePoint(0.25f, -0.25f, 0.5f); 149 mybezier.anchors[1][3] = makePoint(0.75f, -0.25f, -0.75f); 150 mybezier.anchors[2][0] = makePoint(-0.75f, 0.25f, 0.0f); 151 mybezier.anchors[2][1] = makePoint(-0.25f, 0.25f, -0.5f); 152 mybezier.anchors[2][2] = makePoint(0.25f, 0.25f, -0.5f); 153 mybezier.anchors[2][3] = makePoint(0.75f, 0.25f, 0.0f); 154 mybezier.anchors[3][0] = makePoint(-0.75f, 0.75f, -0.5f); 155 mybezier.anchors[3][1] = makePoint(-0.25f, 0.75f, -1.0f); 156 mybezier.anchors[3][2] = makePoint(0.25f, 0.75f, -1.0f); 157 mybezier.anchors[3][3] = makePoint(0.75f, 0.75f, -0.5f); 158 159 mybezier.dlBPatch = NULL; 160 } 161 162 bool LoadGLTexture(GLuint* texPntr, char* name) 163 { 164 bool success = false; 165 AUX_RGBImageRec* TextureImage = NULL; 166 167 glGenTextures(1, texPntr); 168 169 FILE* test = NULL; 170 TextureImage = NULL; 171 172 test = fopen(name, "r"); 173 if (test != NULL) { 174 fclose(test); 175 TextureImage = auxDIBImageLoad(name); 176 } 177 178 if (TextureImage != NULL) { 179 success = true; 180 181 glBindTexture(GL_TEXTURE_2D, *texPntr); 182 glTexImage2D(GL_TEXTURE_2D, 0, 3, TextureImage->sizeX, TextureImage->sizeY, 0, 183 GL_RGB, GL_UNSIGNED_BYTE, TextureImage->data); 184 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); 185 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); 186 } 187 if (TextureImage->data) { 188 free(TextureImage->data); 189 } 190 return success; 191 } 192 /******************************************************************************************************************************************/ 193 /******************************************************************************************************************************************/ 194 GLvoid ReSizeGLScene(GLsizei width, GLsizei height) // Resize and initialize the GL window 195 { 196 if (height == 0) { // Prevent a divide by zero by 197 height = 1; // Making height equal one 198 } 199 200 glViewport(0, 0, width, height); // Reset the current viewport 201 202 /* 203 * The following lines set the screen up for a perspective view. 204 * Meaning things in the distance get smaller. This creates a realistic looking scene. 205 * The perspective is calculated with a 45 degree viewing angle based on 206 * the windows width and height. The 0.1f, 100.0f is the starting point and 207 * ending point for how deep we can draw into the screen. 208 * 209 * The projection matrix is responsible for adding perspective to our scene. 210 * glLoadIdentity() restores the selected matrix to it's original state. 211 * The modelview matrix is where our object information is stored. 212 * Lastly we reset the modelview matrix. 213 */ 214 215 glMatrixMode(GL_PROJECTION); // Select the projection matrix 216 glLoadIdentity(); // Reset the projection matrix 217 218 // Calculate the aspect ratio of the window 219 gluPerspective(45.0f, (GLfloat)width / (GLfloat)height, 0.1f, 100.0f); 220 // glOrtho(0.0f, width, height, 0.0f, -1.0f, 1.0f); // Create orhto 640X480 view (0, 0, at the top) 221 222 glMatrixMode(GL_MODELVIEW); // Seclet the modelview matrix 223 glLoadIdentity(); // Reset the modelview matrix 224 } 225 /******************************************************************************************************************************************/ 226 /******************************************************************************************************************************************/ 227 228 int InitGL(GLvoid) // All setup for OpenGL goes here 229 { 230 glEnable(GL_TEXTURE_2D); 231 glShadeModel(GL_SMOOTH); 232 glClearColor(0.05f, 0.05f, 0.05f, 0.5f); // Background 233 glClearDepth(1.0f); // Depth buffer setup 234 glEnable(GL_DEPTH_TEST); 235 glDepthFunc(GL_LEQUAL); 236 glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST); 237 238 initBezier(); 239 LoadGLTexture(&(mybezier.texture), "image.bmp"); 240 mybezier.dlBPatch = genBezier(mybezier, divs); 241 242 return TRUE; 243 } 244 245 /* 246 * For now all we will do is clear the screen to the color we previously decided on, 247 * clear the depth buffer and reset the scene. We wont draw anything yet. 248 */ 249 bool DrawGLScene(GLvoid) // Here's where we do all the drawing 250 { 251 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT); 252 253 glLoadIdentity(); 254 glTranslatef(0.0f, 0.0f, -4.0f); 255 glRotatef(-75.0f, 1.0f, 0.0f, 0.0f); 256 glRotatef(rotx, 1.0f, 0.0f, 0.0f); 257 glRotatef(rotz, 0.0f, 0.0f, 1.0f); 258 259 glCallList(mybezier.dlBPatch); 260 261 if (showCPoints) { // If drawing the grid is toggled on 262 glDisable(GL_TEXTURE_2D); 263 glColor3f(1.0f, 0.0f, 0.0f); 264 for (int i = 0; i < 4; ++i) { // Draw the horizontal lines 265 glBegin(GL_LINE_STRIP); 266 for (int j = 0; j < 4; ++j) 267 glVertex3d(mybezier.anchors[i][j].x, mybezier.anchors[i][j].y, mybezier.anchors[i][j].z); 268 glEnd(); 269 } 270 for (int i = 0; i < 4; ++i) { // Draw the vertical lines 271 glBegin(GL_LINE_STRIP); 272 for (int j = 0; j < 4; ++j) 273 glVertex3d(mybezier.anchors[j][i].x, mybezier.anchors[j][i].y, mybezier.anchors[j][i].z); 274 glEnd(); 275 } 276 glColor3f(1.0f, 1.0f, 1.0f); 277 glEnable(GL_TEXTURE_2D); 278 } 279 return true; 280 } 281 282 /* 283 * The job of KillGLWindow() is to release the Rendering Context, 284 * the Device Context and finally the Window Handle. 285 */ 286 287 GLvoid KillGLWindow(GLvoid) // Properly kill the window 288 { 289 if (fullscreen) { // Are we in fullscreen mode 290 if (!ChangeDisplaySettings(NULL, CDS_TEST)) { // If the shortcut doesn't work 291 ChangeDisplaySettings(NULL, CDS_RESET); // Do it anyway (to get the values out of the registry) 292 ChangeDisplaySettings(&DMsaved, CDS_RESET); // 293 } 294 else { 295 ChangeDisplaySettings(NULL, 0); // If so switch back to the desktop 296 } 297 //******************************************************************************************************************************************/ 298 //******************************************************************************************************************************************/ 299 /* 300 * We use ChangeDisplaySettings(NULL,0) to return us to our original desktop. 301 * After we've switched back to the desktop we make the cursor visible again. 302 */ 303 ShowCursor(TRUE); // Show mouse pointer 304 } 305 306 if (hRC) { // Do we have a rendering context 307 if (!wglMakeCurrent(NULL, NULL)) { // Are we able to release the DC and RC contexts 308 MessageBox(NULL, "Release of DC and RC failed.", "SHUTDOWN ERROR", MB_OK | MB_ICONINFORMATION); 309 } 310 311 if (!wglDeleteContext(hRC)) { // Are we able to delete the RC 312 MessageBox(NULL, "Release rendering context failed.", "SHUTDOWN ERROR", MB_OK | MB_ICONINFORMATION); 313 hRC = NULL; // Set RC to NULL 314 } 315 316 if (hDC && !ReleaseDC(hWnd, hDC)) { // Are we able to release the DC 317 MessageBox(NULL, "Release device context failed.", "SHUTDOWN ERROR", MB_OK | MB_ICONINFORMATION); 318 hDC = NULL; // Set DC to NULL 319 } 320 if (hWnd && !DestroyWindow(hWnd)) { // Are we able to destroy the window 321 MessageBox(NULL, "Could not release hWnd.", "SHUTDOWN ERROR", MB_OK | MB_ICONINFORMATION); 322 hWnd = NULL; // Set hWnd to NULL 323 } 324 325 if (!UnregisterClass("OpenGL", hInstance)) { // Are we able to unregister class 326 MessageBox(NULL, "Could not register class.", "SHUTDOWN ERROR", MB_OK | MB_ICONINFORMATION); 327 hInstance = NULL; // Set hInstance to NULL 328 } 329 } 330 } 331 /* 332 * The next section of code creates our OpenGL Window. 333 */ 334 335 BOOL CreateGLWindow(char* title, int width, int height, int bits, bool fullscreenflag) 336 { 337 /* 338 * Find a pixel format that matches the one we want 339 */ 340 GLuint PixelFormat; // Holds the result after serching for a match 341 342 /* 343 * Before you create a window, you MUST register a Class for the window 344 */ 345 WNDCLASS wc; // Windows class structure 346 347 /* 348 * dwExStyle and dwStyle will store the Extended and normal Window Style Information. 349 */ 350 DWORD dwExStyle; // Window extend style 351 DWORD dwStyle; // Window style 352 353 354 RECT WindowRect; // Grabs rectangle upper left/lower right values 355 WindowRect.left = (long)0; // Set left value to 0 356 WindowRect.right = (long)width; // Set right value to requested width 357 WindowRect.top = (long)0; // Set top value to 0 358 WindowRect.bottom = (long)height; // Set bottom value to requested height 359 360 fullscreen = fullscreenflag; // Set the global fullscreen flag 361 362 /* 363 * The style CS_HREDRAW and CS_VREDRAW force the Window to redraw whenever it is resized. 364 * CS_OWNDC creates a private DC for the Window. Meaning the DC is not shared across applications. 365 * WndProc is the procedure that watches for messages in our program. 366 * No extra Window data is used so we zero the two fields. Then we set the instance. 367 * Next we set hIcon to NULL meaning we don't want an ICON in the Window, 368 * and for a mouse pointer we use the standard arrow. The background color doesn't matter 369 * (we set that in GL). We don't want a menu in this Window so we set it to NULL, 370 * and the class name can be any name you want. I'll use "OpenGL" for simplicity. 371 */ 372 hInstance = GetModuleHandle(NULL); // Grab an instance for our window 373 wc.style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC; // Redraw on move, and own DC for window 374 wc.lpfnWndProc = (WNDPROC)WndProc; // WndProc handles message 375 wc.cbClsExtra = 0; // No extra window date 376 wc.cbWndExtra = 0; // No extra window date 377 wc.hInstance = hInstance; // set the instance 378 wc.hIcon = LoadIcon(NULL, IDI_WINLOGO); // Load the default icon 379 wc.hCursor = LoadCursor(NULL, IDC_ARROW); // Load the arrow pointer 380 wc.hbrBackground = NULL; // No background requried for GL 381 wc.lpszMenuName = NULL; // We don't want a menu 382 wc.lpszClassName = "OpenGL"; // set the class name 383 //******************************************************************************************************************************************/ 384 //******************************************************************************************************************************************/ 385 EnumDisplaySettings(NULL, ENUM_CURRENT_SETTINGS, &DMsaved); // Save the current display state 386 //******************************************************************************************************************************************/ 387 //******************************************************************************************************************************************/ 388 389 if (!RegisterClass(&wc)) { // Attempt to register the window class 390 MessageBox(NULL, "Failed to register the window class.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 391 return FALSE; // Exit and return false 392 } 393 394 if (fullscreen) { // attempt fullsreen model 395 396 /* 397 T* here are a few very important things you should keep in mind when switching to full screen mode. 398 * Make sure the width and height that you use in fullscreen mode is the same as 399 * the width and height you plan to use for your window, and most importantly, 400 * set fullscreen mode BEFORE you create your window. 401 */ 402 DEVMODE dmScreenSettings; // Device mode 403 memset(&dmScreenSettings, 0, sizeof(dmScreenSettings)); // Make sure memory's cleared 404 dmScreenSettings.dmSize = sizeof(dmScreenSettings); // Size of devmode structure 405 dmScreenSettings.dmPelsWidth = width; // Select window width 406 dmScreenSettings.dmPelsHeight = height; // Select window height 407 dmScreenSettings.dmBitsPerPel = bits; // Select bits per pixel 408 dmScreenSettings.dmFields = DM_BITSPERPEL | DM_PELSWIDTH | DM_PELSHEIGHT; 409 410 /* 411 * In the line below ChangeDisplaySettings tries to switch to a mode that matches 412 * what we stored in dmScreenSettings. I use the parameter CDS_FULLSCREEN when switching modes, 413 * because it's supposed to remove the start bar at the bottom of the screen, 414 * plus it doesn't move or resize the windows on your desktop when you switch to 415 * fullscreen mode and back. 416 */ 417 //Try to set selected mode and get results. Note: CDS_FULLSCREEN gets rid of start bar 418 if (ChangeDisplaySettings(&dmScreenSettings, CDS_FULLSCREEN) != DISP_CHANGE_SUCCESSFUL) { 419 //If the mode fails, offer two options. Quit or run in a window 420 if (MessageBox(NULL, "The requested fullscreen mode is not supported by\n your video card. Use" 421 "windowed mode instead?", "GL", MB_YESNO | MB_ICONEXCLAMATION) == IDYES) 422 { 423 fullscreen = FALSE; // Select windowed mode (fullscreen=FLASE) 424 } 425 else { 426 // Pop up a message box letting user know the programe is closing. 427 MessageBox(NULL, "Program will now close.", "ERROR", MB_OK | MB_ICONSTOP); 428 return FALSE; // Exit and return FALSE 429 } 430 } 431 } 432 433 if (fullscreen) { // Are we still in fullscreen mode 434 435 /* 436 * If we are still in fullscreen mode we'll set the extended style to WS_EX_APPWINDOW, 437 * which force a top level window down to the taskbar once our window is visible. 438 * For the window style we'll create a WS_POPUP window. 439 * This type of window has no border around it, making it perfect for fullscreen mode. 440 441 * Finally, we disable the mouse pointer. If your program is not interactive, 442 * it's usually nice to disable the mouse pointer when in fullscreen mode. It's up to you though. 443 */ 444 dwExStyle = WS_EX_APPWINDOW; // Window extended style 445 dwStyle = WS_POPUP; // Window style 446 ShowCursor(FALSE); // Hide mosue pointer 447 } 448 else { 449 450 /* 451 * If we're using a window instead of fullscreen mode, 452 * we'll add WS_EX_WINDOWEDGE to the extended style. This gives the window a more 3D look. 453 * For style we'll use WS_OVERLAPPEDWINDOW instead of WS_POPUP. 454 * WS_OVERLAPPEDWINDOW creates a window with a title bar, sizing border, 455 * window menu, and minimize / maximize buttons. 456 */ 457 dwExStyle = WS_EX_APPWINDOW | WS_EX_WINDOWEDGE; // Window extended style 458 dwStyle = WS_OVERLAPPEDWINDOW; // Window style 459 } 460 461 /* 462 * By using the AdjustWindowRectEx command none of our OpenGL scene will be covered up by the borders, 463 * instead, the window will be made larger to account for the pixels needed to draw the window border. 464 * In fullscreen mode, this command has no effect. 465 */ 466 AdjustWindowRectEx(&WindowRect, dwStyle, FALSE, dwExStyle); // Adjust window to true resqusted 467 468 /* 469 * WS_CLIPSIBLINGS and WS_CLIPCHILDREN are both REQUIRED for OpenGL to work properly. 470 * These styles prevent other windows from drawing over or into our OpenGL Window. 471 */ 472 if (!(hWnd = CreateWindowEx(dwExStyle, // Extended style for the window 473 "OpenGL", // Class name 474 title, // Window title 475 WS_CLIPSIBLINGS | // Requried window style 476 WS_CLIPCHILDREN | // Requried window style 477 dwStyle, // Select window style 478 0, 0, // Window position 479 WindowRect.right - WindowRect.left, // Calculate adjusted window width 480 WindowRect.bottom - WindowRect.top, // Calculate adjusted window height 481 NULL, // No parent window 482 NULL, // No menu 483 hInstance, // Instance 484 NULL))) // Don't pass anything to WM_CREATE 485 { 486 KillGLWindow(); //Reset the display 487 MessageBox(NULL, "Window creation error.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 488 return FALSE; // Retrurn FALSE; 489 } 490 491 /* 492 * aside from the stencil buffer and the (slow) accumulation buffer 493 */ 494 static PIXELFORMATDESCRIPTOR pfd = // pfd tells windows how we want things to be 495 { 496 sizeof(PIXELFORMATDESCRIPTOR), // Size of this pixel format descriptor 497 1, // Version number 498 PFD_DRAW_TO_WINDOW | // Format must support window 499 PFD_SUPPORT_OPENGL | // Format must support OpenGL 500 PFD_DOUBLEBUFFER, // Must support double buffer 501 PFD_TYPE_RGBA, // Request an RGBA format 502 bits, // Select our color depth 503 0, 0, 0, 0, 0, 0, // Color bits ignored 504 0, // No alpha buffer 505 0, // shift bit ignored 506 0, // No accumulation buffer 507 0, 0, 0, 0, // Accumulation bits ignored 508 16, // 16Bits Z_Buffer (depth buffer) 509 0, // No stencil buffer 510 0, // No auxiliary buffer 511 PFD_MAIN_PLANE, // Main drawing layer 512 0, // Reserved 513 0, 0, 0 // Layer makes ignored 514 }; 515 516 if (!(hDC = GetDC(hWnd))) { // Did we get a device context 517 KillGLWindow(); // Reset the display 518 MessageBox(NULL, "Can't create a GL device context.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 519 return FALSE; // Return FALSE 520 } 521 522 if (!(PixelFormat = ChoosePixelFormat(hDC, &pfd))) { // Did window find a matching pixel format 523 KillGLWindow(); // Reset the display 524 MessageBox(NULL, "Can't find a suitable pixelformat.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 525 return FALSE; // Return FALSE; 526 } 527 528 if (!SetPixelFormat(hDC, PixelFormat, &pfd)) { // Are we able to set the pixel format 529 KillGLWindow(); // Reset the display 530 MessageBox(NULL, "Can't set the pixelformat.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 531 return FALSE; // Return FALSE; 532 } 533 534 if (!(hRC = wglCreateContext(hDC))) { // Are we able to rendering context 535 KillGLWindow(); // Reset the display 536 MessageBox(NULL, "Can't create a GL rendering context.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 537 return FALSE; // Return FASLE; 538 } 539 540 if (!wglMakeCurrent(hDC, hRC)) { // Try to activate the rendering context 541 KillGLWindow(); // Reset the display 542 MessageBox(NULL, "Can't activate the GL rendering context.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 543 return FALSE; // Return FALSE 544 } 545 546 /* 547 * ReSizeGLScene passing the screen width and height to set up our perspective OpenGL screen. 548 */ 549 ShowWindow(hWnd, SW_SHOW); // Show the window 550 SetForegroundWindow(hWnd); // slightly higher priority 551 SetFocus(hWnd); // Sets keyboard focus to the window 552 ReSizeGLScene(width, height); // Set up our perspective GL screen 553 554 /* 555 * we can set up lighting, textures, and anything else that needs to be setup in InitGL(). 556 */ 557 if (!InitGL()) { // Initialize our newly created GL window 558 KillGLWindow(); // Reset the display 559 MessageBox(NULL, "Initialize Failed.", "ERROR", MB_OK | MB_ICONEXCLAMATION); 560 return FALSE; // Return FALSE 561 } 562 return TRUE; 563 } 564 565 LRESULT CALLBACK WndProc(HWND hWnd, // Handle for this window 566 UINT uMsg, // Message for this window 567 WPARAM wParam, // Additional message information 568 LPARAM lParam) // Additional message information 569 { 570 switch (uMsg) { // Check for window message 571 case WM_ACTIVATE: { // Check minimization state 572 if (!HIWORD(wParam)) { 573 active = TRUE; // Program is active 574 } 575 else { 576 active = FALSE; // Program is no longer active 577 } 578 return 0; // Return to the message loop 579 } 580 case WM_SYSCOMMAND: { // Intercept system commands 581 switch (wParam) { // Check system calls 582 case SC_SCREENSAVE: // Screensaver trying to start 583 case SC_MONITORPOWER: // Monitor trying to enter powersave 584 return 0; // Prevent form happening 585 } 586 break; // Exit 587 } 588 case WM_CLOSE: { // Did we receive a close message 589 PostQuitMessage(0); // Send a quit message 590 return 0; 591 } 592 case WM_KEYDOWN: { // Is a key being held down 593 keys[wParam] = TRUE; // if so, mark it as TRUE 594 return 0; // Jump back 595 } 596 case WM_KEYUP: { // Has a key been released 597 keys[wParam] = FALSE; // if so, mark it as FALSE 598 return 0; // Jump back 599 } 600 case WM_SIZE: { // Resize the OpenGL window 601 ReSizeGLScene(LOWORD(lParam), HIWORD(lParam)); // LoWord = width HiWord = height 602 return 0; // Jump back 603 } 604 } 605 return DefWindowProc(hWnd, uMsg, wParam, lParam); // Pass all unhandled message to DefWindwProc 606 } 607 608 int WINAPI WinMain(HINSTANCE hInstance, // Instance 609 HINSTANCE hPrevInstance, // Previous instance 610 LPSTR lpCmdLine, // Command line parameters 611 int nCmdShow) // Window show state 612 { 613 MSG msg; // Window message structure 614 BOOL done = FALSE; // Bool variable to exit loop 615 // Ask the user which screen mode they prefer 616 if (MessageBox(NULL, "Would you like to run in fullscreen mode?", 617 "Start fullscreen?", MB_YESNO | MB_ICONQUESTION) == IDNO) 618 { 619 fullscreen = FALSE; // Window mode 620 } 621 // Create our OpenGL window 622 if (!CreateGLWindow("3D Shapes", 800, 600, 32, fullscreen)) { // (Modified) 623 return 0; // Quit if window was not create 624 } 625 while (!done) { // Loop that runs until donw = TRUE 626 if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { // Is there a message wating 627 if (msg.message == WM_QUIT) { // Havw we received a quit message 628 done = TRUE; // if so done = TRUE 629 } 630 else { // If not, deal with window message 631 TranslateMessage(&msg); // Translate message 632 DispatchMessage(&msg); // Dispatch message 633 } 634 } 635 else { 636 // Draw the scene. Watch for ESC key and quit message from DrawGLScene() 637 if (active) { // Program active 638 if (keys[VK_ESCAPE]) { // Was ESC pressed 639 done = TRUE; // ESC signalled a quit 640 } 641 else { // Not time to quit, update screen 642 DrawGLScene(); // Draw scene 643 SwapBuffers(hDC); // Swap buffers (double buffering) 644 //******************************************************************************************************************************************/ 645 //******************************************************************************************************************************************/ 646 } 647 } 648 if (keys['W']) rotx -= 0.1f; 649 if (keys['S']) rotx += 0.1f; 650 if (keys['A']) rotz -= 0.1f; 651 if (keys['D']) rotz += 0.1f; 652 if (keys[VK_LEFT]) rotz -= 0.1f; 653 if (keys[VK_RIGHT]) rotz += 0.1f; 654 if (keys[VK_UP]) { 655 divs++; 656 mybezier.dlBPatch = genBezier(mybezier, divs); 657 keys[VK_UP] = false; 658 } 659 if (keys[VK_DOWN] && divs > 1) { 660 divs--; 661 mybezier.dlBPatch = genBezier(mybezier, divs); 662 keys[VK_DOWN] = false; 663 } 664 if (keys[VK_SPACE]) { 665 showCPoints = !showCPoints; 666 keys[VK_SPACE] = false; 667 } 668 //******************************************************************************************************************************************/ 669 //******************************************************************************************************************************************/ 670 671 /* 672 * It allows us to press the F1 key to switch from fullscreen mode to 673 * windowed mode or windowed mode to fullscreen mode. 674 */ 675 if (keys[VK_F1]) { // Is F1 being pressed 676 keys[VK_F1] = FALSE; // If so make key FASLE 677 KillGLWindow(); // Kill our current window 678 fullscreen = !fullscreen; // Toggle fullscreen / window mode 679 //Recreate our OpenGL window(modified) 680 if (!CreateGLWindow("3D Shapes", 640, 480, 16, fullscreen)) { 681 return 0; // Quit if window was not create 682 } 683 } 684 } 685 } 686 // Shutdown 687 KillGLWindow(); // Kill the window 688 return (msg.wParam); // Exit the program 689 }
Thanks for Nehe's tutorials, this is his home.