NeHe的OpenGl教程 - 前5课总结

NeHe OpenGL教程实践
本文详细介绍了如何在Windows XP和Visual C++ 2008 Express环境下,通过具体步骤设置和运行NeHe的OpenGL教程示例程序。文章还提供了源代码,并解释了如何解决常见问题。

NeHe 的 OpenGL 教程浅显易懂,每行都有注释,适合入门学习用。不过,要想让教程中的程序运行在我的电脑上,需要做一些小小的设置

我的电脑的软件配置:Windows XP Professional + Visual C++ 2008 Express

1. “项目属性”的 Configuration Properties 选项卡:Character Set
NeHe 教程用的全是 C-sytle null terminated string。所以我们要根据自己的项目的特点做相应设置。我的程序使用的都是UNICODE字符串,所以设置为 Use UNICODE Character Set

2. “Linker” -> “Input”选项卡:Additional Dependencies 需要添加两行

"$(WindowsSdkDir)Lib\glu32.lib"
"$(WindowsSdkDir)Lib\opengl32.lib"

3. 原教程中的代码:#include <glaux.h> 在网上很难找,于是去掉了。反正只要不用里面的代码就OK

然后就可以让下面的代码运行了:

#include <windows.h> #include <math.h> #include <gl\gl.h> #include <gl\glu.h> // global variable HDC g_hDC = NULL; HGLRC g_hRC = NULL; HWND g_hWnd = NULL; HINSTANCE g_hIns = NULL; BOOL g_pKeys[256] = {0}; BOOL g_bActive = TRUE; BOOL g_bFullScreen = TRUE; GLfloat g_fAngle4Tri = 0.0f; GLfloat g_fAngle4Quad = 0.0f; LPCTSTR g_lpszWndClass = TEXT("OpenGL"); const float kRotateSpeed = 0.001f; const float kRadius = 3.0f; const DWORD kStartTime = GetTickCount(); GLvoid ResizeGLWnd(GLsizei nWidth, GLsizei nHeight) { if(nHeight == 0) { nHeight = 1; } glViewport(0, 0, nWidth, nHeight); glMatrixMode(GL_PROJECTION); glLoadIdentity(); gluPerspective(45.0f, (GLfloat)nWidth/(GLfloat)nHeight, 0.1f, 100.0f); glMatrixMode(GL_MODELVIEW); glLoadIdentity(); } GLvoid KillGLWnd() { if(g_bFullScreen) { ChangeDisplaySettings(NULL, 0); ShowCursor(TRUE); } if(g_hRC) { if(!wglMakeCurrent(NULL, 0)) { MessageBox(g_hWnd, TEXT("Desktop cannot be showed properly."), TEXT("SHUTDOWN ERROR"), MB_OK | MB_ICONINFORMATION); } if(!wglDeleteContext(g_hRC)) { MessageBox(g_hWnd, TEXT("OpenGL rendering context cannot be released"), TEXT("SHUTDOWN ERROR"), MB_OK | MB_ICONINFORMATION); } g_hRC = NULL; } if(g_hDC && !ReleaseDC(g_hWnd, g_hDC)) { MessageBox(g_hWnd, TEXT("Windows device context cannot be released"), TEXT("SHUTDOWN ERROR"), MB_OK | MB_ICONINFORMATION); g_hDC = NULL; } if(g_hWnd && !DestroyWindow(g_hWnd)) { MessageBox(g_hWnd, TEXT("Windows cannot be destroyed"), TEXT("SHUTDOWN ERROR"), MB_OK | MB_ICONINFORMATION); g_hWnd = NULL; } if(!UnregisterClass(g_lpszWndClass, g_hIns)) { MessageBox(g_hWnd, TEXT("Windows class unregistering failed"), TEXT("SHUTDOWN ERROR"), MB_OK | MB_ICONINFORMATION); g_hIns = NULL; } } int InitOpenGL() { glShadeModel(GL_SMOOTH); glClearColor(0.0f, 0.0f, 0.0f, 0.0f); glClearDepth(1.0f); glEnable(GL_DEPTH_TEST); glDepthFunc(GL_LEQUAL); glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST); return TRUE; } LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM); BOOL CreateGLWnd(PTSTR lpTitle, int nWidth, int nHeight, int nBits, bool bFullScreen) { GLuint nPixelFormat; WNDCLASS wc; DWORD dwStyle, dwExStyle; RECT WndRect; WndRect.left = (long)0; WndRect.right = (long)nWidth; WndRect.top = (long)0; WndRect.bottom = (long)nHeight; g_bFullScreen = bFullScreen; g_hIns = GetModuleHandle(NULL); wc.style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC; wc.lpfnWndProc = (WNDPROC)WndProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = g_hIns; wc.hIcon = LoadIcon(NULL, IDI_WINLOGO); wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = NULL; wc.lpszMenuName = NULL; wc.lpszClassName = g_lpszWndClass; if(!RegisterClass(&wc)) { MessageBox(NULL, TEXT("Register of window class failed"), TEXT("CREATE ERROR"), MB_OK | MB_ICONEXCLAMATION); return FALSE; } if(g_bFullScreen) { DEVMODE dmScreenSettings; memset(&dmScreenSettings, 0, sizeof(dmScreenSettings)); dmScreenSettings.dmSize = sizeof(dmScreenSettings); dmScreenSettings.dmPelsWidth = nWidth; dmScreenSettings.dmPelsHeight = nHeight; dmScreenSettings.dmBitsPerPel = nBits; dmScreenSettings.dmFields = DM_BITSPERPEL | DM_PELSWIDTH | DM_PELSHEIGHT; if(ChangeDisplaySettings(&dmScreenSettings, CDS_FULLSCREEN)) { if (MessageBox(NULL, TEXT("The Requested Fullscreen Mode Is Not Supported By\nYour Video Card. Use Windowed Mode Instead?"), TEXT("NeHe GL"), MB_YESNO | MB_ICONEXCLAMATION) == IDYES) { g_bFullScreen = FALSE; // Windowed Mode Selected. Fullscreen = FALSE } else { // Pop Up A Message Box Letting User Know The Program Is Closing. MessageBox(NULL, TEXT("Program Will Now Close."),TEXT("ERROR"), MB_OK | MB_ICONSTOP); return FALSE; // Return FALSE } } } if(g_bFullScreen) { dwExStyle = WS_EX_APPWINDOW; dwStyle = WS_POPUP; ShowCursor(FALSE); } else { dwExStyle = WS_EX_APPWINDOW | WS_EX_WINDOWEDGE; dwStyle = WS_OVERLAPPEDWINDOW; } AdjustWindowRectEx(&WndRect, dwStyle, FALSE, dwExStyle); if(!(g_hWnd = CreateWindowEx(dwExStyle, g_lpszWndClass, lpTitle, dwStyle | WS_CLIPSIBLINGS | WS_CLIPCHILDREN, 0, 0, WndRect.right - WndRect.left, WndRect.bottom - WndRect.top, NULL, NULL, g_hIns, NULL))) { KillGLWnd(); MessageBox(NULL, TEXT("Window Creation Error."),TEXT("ERROR"), MB_OK | MB_ICONEXCLAMATION); return FALSE; } static PIXELFORMATDESCRIPTOR pfd = { sizeof(PIXELFORMATDESCRIPTOR), 1, PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER, PFD_TYPE_RGBA, nBits, 0,0,0,0,0,0, 0, 0, 0, 0,0,0,0, 16, 0, 0, PFD_MAIN_PLANE, 0, 0,0,0 }; if(!(g_hDC = GetDC(g_hWnd))) { KillGLWnd(); MessageBox(NULL, TEXT("Can't Create A GL Device Context."), TEXT("ERROR"), MB_OK | MB_ICONEXCLAMATION); return FALSE; } if(!(nPixelFormat = ChoosePixelFormat(g_hDC, &pfd))) { KillGLWnd(); MessageBox(NULL, TEXT("Can't find proper pixel format."), TEXT("ERROR"), MB_OK | MB_ICONEXCLAMATION); return FALSE; } if(!SetPixelFormat(g_hDC, nPixelFormat, &pfd)) { KillGLWnd(); MessageBox(NULL, TEXT("Can't Set The PixelFormat."), TEXT("ERROR"),MB_OK | MB_ICONEXCLAMATION); return FALSE; } if (!(g_hRC = wglCreateContext(g_hDC))) { KillGLWnd(); MessageBox(NULL, TEXT("Can't Create A GL Rendering Context."), TEXT("ERROR"),MB_OK | MB_ICONEXCLAMATION); return FALSE; } if(!wglMakeCurrent(g_hDC, g_hRC)) { KillGLWnd(); MessageBox(NULL, TEXT("Can't Activate The GL Rendering Context."), TEXT("ERROR"),MB_OK | MB_ICONEXCLAMATION); return FALSE; } ShowWindow(g_hWnd, SW_SHOW); SetForegroundWindow(g_hWnd); SetFocus(g_hWnd); ResizeGLWnd(nWidth, nHeight); if(!InitOpenGL()) { KillGLWnd(); MessageBox(NULL, TEXT("Initialization Failed."), TEXT("ERROR"),MB_OK | MB_ICONEXCLAMATION); return FALSE; } } int DrawScene() { const GLfloat kDistanceZ = -15.0f; glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); glLoadIdentity(); // triangle glTranslatef(0.0f, -2.0f, kDistanceZ); glRotatef(g_fAngle4Tri, 0.0f, 1.0f, 0.0f); glBegin(GL_TRIANGLES); glColor3f(1.0f, 0.0f, 0.0f); glVertex3f(0.0f, 1.0f, 0.0f); glColor3f(0.0f, 1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 1.0f); glColor3f(0.0f, 0.0f, 1.0f); glVertex3f(1.0f, -1.0f, 1.0f); glColor3f(1.0f, 0.0f, 0.0f); glVertex3f(0.0f, 1.0f, 0.0f); glColor3f(0.0f, 0.0f, 1.0f); glVertex3f(1.0f, -1.0f, -1.0f); glColor3f(0.0f, 1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, -1.0f); glColor3f(0.0f, 0.0f, 1.0f); glVertex3f(1.0f, -1.0f, 1.0f); glVertex3f(1.0f, -1.0f, -1.0f); glColor3f(0.0f, 1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, -1.0f); glColor3f(0.0f, 1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, -1.0f); glVertex3f(-1.0f, -1.0f, 1.0f); glColor3f(0.0f, 0.0f, 1.0f); glVertex3f(1.0f, -1.0f, 1.0f); glEnd(); glLoadIdentity(); // square DWORD dwDeltaTime = GetTickCount() - kStartTime; GLfloat fNewX = kRadius * cos(dwDeltaTime * kRotateSpeed); GLfloat fNewZ = kRadius * sin(dwDeltaTime * kRotateSpeed); glTranslatef(fNewX, -2.0f, fNewZ + kDistanceZ); glRotatef(g_fAngle4Quad, 0.0f, 1.0f, 0.0f); glBegin(GL_QUADS); glColor3f(0.0f, 1.0f, 0.0f); glVertex3f(1.0f, 1.0f, 0.0f); glVertex3f(-1.0f, 1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 0.0f); glVertex3f(1.0f, -1.0f, 0.0f); glEnd(); g_fAngle4Tri += 2.0f; g_fAngle4Quad += 20.0f; return TRUE; } LRESULT CALLBACK WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch (uMsg) { case WM_ACTIVATE: g_bActive = !HIWORD(wParam) ? TRUE : FALSE; return 0; case WM_SYSCOMMAND: switch (wParam) { case SC_SCREENSAVE: case SC_MONITORPOWER: return 0; } break; case WM_CLOSE: PostQuitMessage(0); return 0; case WM_KEYDOWN: g_pKeys[wParam] = TRUE; return 0; case WM_KEYUP: g_pKeys[wParam] = FALSE; return 0; case WM_SIZE: ResizeGLWnd(LOWORD(lParam), HIWORD(lParam)); return 0; } return DefWindowProc(hWnd, uMsg, wParam, lParam); } int WINAPI WinMain(HINSTANCE hInst, HINSTANCE hPrev, LPSTR lpCmdLine, int nCmdShow) { MSG msg; BOOL bDone = FALSE; // Ask The User Which Screen Mode They Prefer if (MessageBox(NULL, TEXT("Would You Like To Run In Fullscreen Mode?"), TEXT("Start FullScreen?"), MB_YESNO | MB_ICONQUESTION) == IDNO) { g_bFullScreen = FALSE; } if (!CreateGLWnd(TEXT("NeHe's Rotation Tutorial"), 640, 480, 16, g_bFullScreen)) { return 0; } while(!bDone) { if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { if (msg.message == WM_QUIT) { bDone = TRUE; } else { TranslateMessage(&msg); DispatchMessage(&msg); } } else { if ((g_bActive && !DrawScene()) || g_pKeys[VK_ESCAPE]) { bDone = TRUE; } else { SwapBuffers(g_hDC); } if (g_pKeys[VK_F1]) { g_pKeys[VK_F1] = FALSE; KillGLWnd(); g_bFullScreen = !g_bFullScreen; if (!CreateGLWnd(TEXT("NeHe's Rotation Tutorial"), 640, 480, 16, g_bFullScreen)) { return 0; } } } } KillGLWnd(); return (msg.wParam); }

显示效果如下图:

OpenGL 编程 - 两个自转的物体

创建一个OpenGL窗口: 在这个教程里,我将教你在Windows环境中创建OpenGL程序.它将显示一个空的OpenGL窗口,可以在窗口和全屏模式下切换,按ESC退出.它是我们以后应用程序的框架. 理解OpenGL如何工作非常重要,你可以在教程的末尾下载源程序,但我强烈建议你至少读一遍教程,然后再开始编程. 2.你的第一个多边形: 在第一个教程的基础上,我们添加了一个三角形和一个四边形。也许你认为这很简单,但你已经迈出了一大步,要知道任何在OpenGL中绘制的模型都会被分解为这两种简单的图形。 读完了这一,你会学到如何在空间放置模型,并且会知道深度缓存的概念。 3.添加颜色: 作为第二的扩展,我将叫你如何使用颜色。你将理解两种着色模式,在左图中,三角形用的是光滑着色,四边形用的是平面着色。 注意三角形上的颜色是如何混合的。 颜色为OpenGlL 工程增加很多。通过理解平面着色(flat coloring)和平滑着色(smooth coloring),你能显著的改善你的OpenGL Demo的样子。 4.旋转: 在这一里,我将教会你如何旋转三角形和四边形。左图中的三角形沿Y轴旋转,四边形沿着X 轴旋转。 这一章将引入两个变量, rtri 被用来存储三角形的角度, rquad存储四边形的角度。 和容易创建一个多边形组成的场景。让这些物体动起来是整个场景变得生动起来。在后面的程钟我将教给你如何绕屏幕上的一个点旋转物体,使得物体绕屏幕而不是它的轴转动。 5.3D形体: 既然我们已经领会到多边形,方形,色彩和旋转。现在该建立3D物体了。我将使用多边形和矩形c创建3D物体。这次我们将扩展上一章的教程,并且将三角形转换成一个彩色的棱锥,把正方形变为一个实心正方体。棱锥使用混合色,正方体每个面使用一种颜色。在3D空间创建物体可能很费时间,但是所获得的结果(收获)值得这样做。充分发挥你的想象力吧。 6.纹理映射: 你想要它,它现在就在这里了,那就是 ... 纹理映射!!!在这一章我将教会你如何将一幅位图(bitmap)映射到正方体的六个面上去。我们将使用第一章的OpenGL代码来创建工程。创建一个空的窗口比修改上一的代码更容易。 你将会发现第一章的代码在对于快速创建工程来说是及其有价值的。第一章的代码为你设置好了一切,你所需要做的只是集中精力为效果编程。 7.纹理滤波, 光照和键盘控制: 好的,我希望到现在你已经理解了所有的东西,因为这是一个巨大的教程。我想教给你两个新的方法来过滤(filter)你的纹理,简单的光照,键盘控制并且还可能更多 :) .如果你对到这一为止你所学的东西并不充满信心,那就回头复习一下。玩一下其它程的代码,不要操之过急。最好专心把每一学好,而不是蜻蜓点水,只知道如何把东西做出来。 8.混合 有理由等一下,一个来自很酷的Hypercosm的程序员伙伴问(我)他是否可以写一章关于混合的教程。第八通常正是讲混合的,所以太巧了。这一章教程扩展了第七章。混合是一项很酷的技术 .. 我希望你们能好好享受这一章教程。这一章的作者是Tom Stanis他在这制作一章上花费了很多精力,所以让他知道你觉得怎么样。混合可不是一个好讲的话题。 9.在3D空间中移动位图: 这一章覆盖了一些你们要求的主题,你想知道如何移动你在3D屏幕空间上创造的物体。你想要知道如何在屏幕上绘制一幅位图,并且位图的黑色部分不会覆盖它后面的东西。你想要简单的动画,想要更多的混合的应用,这一章将教会你所有这些。You'll notice there's no spinning boxes(yaker:很惭愧这一句我不是很明白)。面的程覆盖了OpenGL的基础,每一章都基于面的内容。面的程涵盖了基础的OpenGL,每一都是在的基础上创建的。这一面几知识的综合,当你学习这时,请确保你已经掌握了面几的知识。 10.加载3D世界,并在其中漫游: 你一直期待的教程来了!这一章友一个叫Lionel Brites的伙伴制作。这一里你讲学到如何导入一个3D世界。代码仍然使用第一章的,但是,程页面只是解释了新的部分,包括导入3D场景,在3D世界中移动。下载VC++代码并且在你阅读教程的同时阅读代码。按[B]键控制混合,[F]键控制滤波,[L]键控制光照(但光并不随场景移动),还有[Page UP]和[Page Down]键。我希望你能喜欢Lionel对网站的贡献。我有空的时候我会让这个教程更容易学习。 11.旗帜效果
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值