開發(fā)基礎(chǔ) OpenGL極速基礎(chǔ)寶典 發(fā)表時(shí)間:2024-05-17 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣: [摘要]不知為什么,最近給我發(fā)短消息問問題的人是越來越多,我真的有點(diǎn)忙不過來了,現(xiàn)在一點(diǎn)個(gè)人時(shí)間都沒有啦,在公司做公司的項(xiàng)目,在家里寫自己的程序,硬擠出一點(diǎn)點(diǎn)時(shí)間來還要留給CSDN……人活著真累!不過話說回來,做版主不盡職盡責(zé)可不是一件好事情哦 :)上次寫的《Winamp插件詳解》也許對于我們版的很多朋友... 不知為什么,最近給我發(fā)短消息問問題的人是越來越多,我真的有點(diǎn)忙不過來了,現(xiàn)在一點(diǎn)個(gè)人時(shí)間都沒有啦,在公司做公司的項(xiàng)目,在家里寫自己的程序,硬擠出一點(diǎn)點(diǎn)時(shí)間來還要留給CSDN……人活著真累!不過話說回來,做版主不盡職盡責(zé)可不是一件好事情哦 :)上次寫的《Winamp插件詳解》也許對于我們版的很多朋友來說起點(diǎn)有高了,貼出來是叫好不叫座,也就是支持的人多,真正拿回去研究的人少啊,很多人是沖著那200分來的。這次我就來點(diǎn)簡單的吧,相信這也是一個(gè)非常熱門并且很有趣的題目。 按照慣例我還是要先說一些廢話,OpenGL被嚴(yán)格定義為“一種到圖形硬件的軟件接口”。從本質(zhì)上說,它是一個(gè)完全可移植并且速度很快的3D圖形和建模庫。使用OpenGL,你可以創(chuàng)建視覺質(zhì)量接近射線跟蹤程序的精致漂亮的3D圖形。使用OpenGL的最大好處是它比射線跟蹤程序要快好幾個(gè)數(shù)量級。它使用由Silicon Graphcs(SGI)公司精心開發(fā)和優(yōu)化的算法,這家公司在計(jì)算機(jī)圖形和動畫領(lǐng)域是公認(rèn)的業(yè)界領(lǐng)袖。這并不是說每個(gè)人都應(yīng)該用OpenGL為商業(yè)應(yīng)用程序畫餅圖和柱形圖。不過,外觀非常重要,其它方面的功能大致相同的產(chǎn)品,其成功或失敗常常取決于“吸引力”。而用漂亮的3D圖形可以增加許多吸引力!這次我將帶你進(jìn)入真正的計(jì)算機(jī)三維時(shí)代,體驗(yàn)三維編程的魅力。我們將從OpenGL做為入手點(diǎn),開始建立一個(gè)完全獨(dú)立的應(yīng)用程序,能夠顯示一些物體,并且在后面添加一些特效,使我們的顯示畫面更為美觀。在閱讀完本文之后,你應(yīng)該可以寫一些簡單的三維小程序了,如果你是一個(gè)開發(fā)老手,那你也許可以擁有一個(gè)版權(quán)屬于自己的3D小游戲吧?雖然這篇文章的起點(diǎn)很低,但在看下去之前還是需要你評估一下你的實(shí)際編程能力:熟練的使用VC.net開發(fā)環(huán)境和MSDN、寫過完全獨(dú)立的SDK程序、熟悉C語言和C++。請保持愉快的心情閱讀全文。 首先讓VC.net來為我們自動建立一個(gè)可以運(yùn)行的SDK程序(這個(gè)你應(yīng)該會吧?),名字叫做GlTest,然后來了解一下我們需要用到的頭文件和導(dǎo)入庫。一般在VC.net中,OpenGL的頭文件是存放在系統(tǒng)頭文件目錄的子目錄GL中的,所以在指定包含的時(shí)候要指定一下相對路徑。gl.h是基本頭文件,glu.h是應(yīng)用頭文件,大多數(shù)應(yīng)用程序都需要同時(shí)包含這兩個(gè)頭文件。opengl32.lib則是OpenGL的win32實(shí)現(xiàn)的標(biāo)準(zhǔn)導(dǎo)入庫,所以我們在剛剛建立的工程中的StdAfx.cpp的頭文件聲明區(qū)添加下面的編譯器指令: #pragma comment( lib, "opengl32.lib" ) #pragma comment( lib, "glu32.lib" ) #include #include
在這之后,你就可以隨意調(diào)用OpenGL的函數(shù)了。但是不得不稍帶說明的是,VC.net附帶的MSDN里有所有的OpenGL標(biāo)準(zhǔn)函數(shù)的定義說明,但僅是如此,與DirectX的教程比起來相去甚遠(yuǎn)。從這一點(diǎn)也可以看出微軟在大的商業(yè)戰(zhàn)略方針上是一力推崇DirectX,排斥其它的圖形編程接口。如果你是一個(gè)初學(xué)者,并希望從MSDN的OpenGL的說明上得到你所想要的知識,那么我只能告訴你,你錯(cuò)了,應(yīng)該去書店里買一本《OpenGL編程權(quán)威指南》,這本書里才有真正適合你的東西,F(xiàn)在你也許會對我的話不屑一顧,因?yàn)槟悴粫ㄌ嗟木徒疱X在研究這類“無聊且無用的東西”上,僅僅是看這篇文章來消遺,那也無所謂,你現(xiàn)在要做的僅是保持耐心,繼續(xù)看完全文。
我們還需要對VC.net自動生成的工程做一些修改,讓它來適合我們的OpenGL應(yīng)用。第一個(gè)要改的是消息循環(huán),大多數(shù)時(shí)實(shí)渲染的應(yīng)用程序都會把繪圖的代碼放在空閑事件里,空閑事件與其說是一種事件倒不如說它是“沒有事件”,先看一下我們怎么寫消息循環(huán): while ( true ) { if ( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ) ) { if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } continue; } if ( WM_QUIT == msg.message ) { break; } OnIdle(); }
使用PeekMessage而不是GetMessage,這樣當(dāng)消息隊(duì)列中沒有消息時(shí)便不會等待而是返回一個(gè)FALSE值,這樣我們就可能知道當(dāng)前應(yīng)用程序處理空閑狀態(tài)了。另外值得注意的是,如果得到的消息是WM_QUIT,PeekMessage一樣會返回FALSE值,所以我們需要做一些特殊處理。在循環(huán)的最后回調(diào)我們的空閑消息處理函數(shù):OnIdle(); 第二個(gè)要改的是注冊窗體類時(shí),wcex.style應(yīng)該賦為0,因?yàn)檫@是一個(gè)實(shí)時(shí)渲染程序,并不需要系統(tǒng)來自動管理窗體的重繪,這樣可以提高程序的速度。第三個(gè)要改動的地方是把消息處理回調(diào)函數(shù)中對WM_PAINT消息處理的代碼屏蔽掉,使用default的return DefWindowProc(hWnd, message, wParam, lParam);就可以了,不然GDI的繪圖會與OpenGL沖突,并且阻止程序空閑。
接下來就是最重要的部分了,初始化我們的OpenGL。先大致講一下步驟:一,獲取你需要在上面繪圖的設(shè)備環(huán)境(DC);二、為該設(shè)備環(huán)境設(shè)置像素格式;三、創(chuàng)建基于該設(shè)備環(huán)境的OpenGL設(shè)備;四、初始化OpenGL繪制場景及狀態(tài)設(shè)置。下面我們來看前三步的代碼:
g_hDC = GetDC( g_hWnd ); // 獲取DC PIXELFORMATDESCRIPTOR pfd; ZeroMemory( &pfd, sizeof(PIXELFORMATDESCRIPTOR) ); // 無關(guān)項(xiàng)置0 pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR ); pfd.nVersion = 1; // 版本號,必須為1 pfd.dwFlags = PFD_DRAW_TO_WINDOW PFD_SUPPORT_OPENGL PFD_DOUBLEBUFFER; //前兩項(xiàng)必須,PFD_DOUBLEBUFFER指定使用雙緩沖 pfd.iPixelType = PFD_TYPE_RGBA; // 顏色格式為紅、綠、藍(lán)、透明 pfd.cColorBits = 24; // 24位色深 pfd.cDepthBits = 32; // 32位Z緩沖深度
SetPixelFormat( g_hDC, ChoosePixelFormat( g_hDC, &pfd ), &pfd ); // 選擇一個(gè)像素格式,并設(shè)置到DC中
g_glRes = wglCreateContext( g_hDC ); // 創(chuàng)建OpenGL設(shè)備 wglMakeCurrent( g_hDC, g_glRes ); // 啟用OpenGL設(shè)備
看了上面的代碼及注釋后,你應(yīng)該很清楚每一條語句的作用了吧,很簡單不是?不過別忘了,在WM_DESTROY消息觸發(fā)時(shí)釋放OpenGL資源:
ReleaseDC( g_hWnd, g_hDC ); wglDeleteContext( g_glRes );
這樣OpenGL就以經(jīng)初始化完畢了,這也就意味著你可以立即在OnIdle里使用OpenGL語句繪圖了,雖然理論上是如此,但為了達(dá)到我們的效果——一個(gè)旋轉(zhuǎn)的方盒——我們還需要再設(shè)置一下場景:
glEnable( GL_CULL_FACE ); // 將不渲染看不見的隱消面 glCullFace( GL_BACK );
glEnable( GL_DEPTH_TEST ); //無論繪制的先后,讓距離遠(yuǎn)的物體總在距離近的物體后面。 glDepthFunc( GL_LEQUAL );
int LightPos[] = { 50, 50, 10, 1 }; // 最后一個(gè)指定這是一個(gè)無指向的點(diǎn)光源 float LightColor[] = { 0.3f, 0.3f, 0.3f, 1.0f }; // 1.0是最亮,0.3看起來并不那么刺眼 glEnable(GL_LIGHTING); // 打開光照狀態(tài),除非人為改變,該狀態(tài)將一直保留到程序退出 glLightiv( GL_LIGHT0, GL_POSITION, LightPos ); // 設(shè)置燈光位置 glLightfv( GL_LIGHT0, GL_AMBIENT, LightColor ); // 環(huán)境色 glLightfv( GL_LIGHT0, GL_DIFFUSE, LightColor ); // 散射色 glEnable( GL_LIGHT0 ); // 打開第一個(gè)光源,你一共可以開8個(gè) glEnable( GL_COLOR_MATERIAL ); //打開顏色材質(zhì) glColorMaterial( GL_FRONT, GL_AMBIENT_AND_DIFFUSE ); // 我們可以為物體指定顏色 glShadeModel( GL_SMOOTH ); // 啟用光滑的著色 glClearColor( 255.0f / 255.0f, 255.0f / 255.0f, 200.0f / 255.0f, 0.0 ); // 背景色 glColor3ub( 140, 200, 255 ); // 填充色 這里要稍帶一提的是OpenGL是一種狀態(tài)機(jī)模式,比如你用glEnable打開一個(gè)狀態(tài),在以后的繪圖中將一直保留并應(yīng)用這個(gè)狀態(tài),除非你調(diào)用glDisable及同類函數(shù)來改變該狀態(tài)或程序退出。OpenGL絕大多數(shù)函數(shù)都是一種狀態(tài)機(jī),比如你設(shè)置了當(dāng)前的紋理,那么紋理將不會自動改變。
下面要講一些理論的東西了,請不要感到厭煩,因?yàn)槿绻麤]有這些知識,我們的三維教程將很難進(jìn)行下去。為了方便的描述三維場景中物體的旋轉(zhuǎn)、平移、縮放等空間變換操作,我們引入三維變換矩陣的概念。這是一個(gè)4X4的矩陣,當(dāng)然單位矩陣的對角線上的值都是1了?催@貌似平凡的矩陣,里面卻蘊(yùn)藏著無數(shù)的神奇。比如在笛卡爾坐標(biāo)系中有一個(gè)空間點(diǎn),坐標(biāo)是10, 10, 10,現(xiàn)在你想把這一點(diǎn)平移5, -2, 8個(gè)單位,那么你只需要將變換矩陣最后一行的前三列的值為別賦為5、-2和8再將空間點(diǎn)的坐標(biāo)做為一個(gè)4X1的矩陣,最后一列補(bǔ)0再與變換矩陣求積(什么?你不會算矩陣相乘?!我倒。玫降4X1矩陣的前三列值便是變換過的空間點(diǎn)坐標(biāo)的X、Y和Z。同樣的旋轉(zhuǎn)、縮放也是大致的方法,區(qū)別僅在于變換矩陣?yán)锊煌恢玫闹荡聿煌暮x。
現(xiàn)在我們將開始繪圖。先確定一下視角:
// 設(shè)置模形矩陣 void SetModalMatrix( void ) { glMatrixMode( GL_MODELVIEW ); glLoadIdentity( ); // 單位化矩陣 // 這個(gè)函數(shù)是在OnIdle里被調(diào)用的,所以我們用下面的代碼來實(shí)現(xiàn)物體的旋轉(zhuǎn) // 一個(gè)很容易理解的概念是,你繞著物體轉(zhuǎn)和物體自己轉(zhuǎn)在某些簡單場景里的 // 的效果看起來是一樣的,所以我們通過矩陣運(yùn)算讓眼睛點(diǎn)在一定高度做圓周 // 運(yùn)動。知道圓的簡化方程是:(sinα* r)^2 + (cosα* r)^2 = r^2,所以代碼 // 很好理解。 static float fRadius = 0; fRadius += 0.01f; if ( fRadius > M_PI * 2 ) { fRadius = 0; } gluLookAt( cosf( fRadius ) * 30, sinf( fRadius ) * 30, 15.0, 0.0, 0.0, 0.0, // 向原點(diǎn)坐標(biāo)看去 0.0, 0.0, 1.0 ); // 設(shè)置眼睛(攝影機(jī))的方向向量,該向量表示眼表向上 }
// 設(shè)置透視矩陣 void SetProjMatrix( WORD wWidth, WORD wHeight ) { // 此函數(shù)將在WM_SIZE時(shí)被調(diào)用,所以應(yīng)該設(shè)置一下glViewPort glViewport( 0, 0, wWidth, wHeight ); glMatrixMode( GL_PROJECTION ); glLoadIdentity( ); // 這和照象機(jī)很類似,第一個(gè)參數(shù)設(shè)置鏡頭廣角度,第二個(gè)參數(shù)是長寬比,后面是遠(yuǎn)近剪切。 gluPerspective( 45.0, (double)wWidth / (double)wHeight, 1.0, 1000.0 ); }
然后我們在OnDraw里調(diào)用下面的代碼:
// 先將上次渲染的殘留物清為背景色 glClear( GL_COLOR_BUFFER_BIT GL_DEPTH_BUFFER_BIT ); glBegin( GL_QUADS ); // 設(shè)置繪制模式,我們畫一個(gè)平面的四邊形 glVertex2i( 5, 5 ); glVertex2i( 5, -5 ); glVertex2i( -5, -5 ); glVertex2i( -5, 5 ); SwapBuffers( g_hDC ); // 交換前后緩沖,雙緩沖無閃爍
至此,GlTest.cpp中的代碼因該是這個(gè)樣子:
// GlTest.cpp : 定義應(yīng)用程序的入口點(diǎn)。 //
#include "stdafx.h" #include "GlTest.h" #define MAX_LOADSTRING 100
// 全局變量: HINSTANCE hInst; // 當(dāng)前實(shí)例 HWND g_hWnd; HDC g_hDC; HGLRC g_glRes;
TCHAR szTitle[MAX_LOADSTRING]; // 標(biāo)題欄文本 TCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口類名
// 此代碼模塊中包含的函數(shù)的前向聲明: void OnCreate( HWND hWnd ); void OnCreated( void ); void OnDestroy( void ); void OnDraw( void ); void SetProjMatrix( WORD wWidth, WORD wHeight ); void SetModalMatrix( void ); void OnIdle( void );
ATOM MyRegisterClass(HINSTANCE hInstance); BOOL InitInstance(HINSTANCE, int); LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM); LRESULT CALLBACK About(HWND, UINT, WPARAM, LPARAM);
int APIENTRY _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow) { // TODO: 在此放置代碼。 MSG msg; HACCEL hAccelTable;
// 初始化全局字符串 LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING); LoadString(hInstance, IDC_GLTEST, szWindowClass, MAX_LOADSTRING); MyRegisterClass(hInstance);
// 執(zhí)行應(yīng)用程序初始化: if (!InitInstance (hInstance, nCmdShow)) { return FALSE; }
hAccelTable = LoadAccelerators(hInstance, (LPCTSTR)IDC_GLTEST);
// 主消息循環(huán): while ( true ) { if ( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ) ) { if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } continue; } if ( WM_QUIT == msg.message ) { break; } OnIdle(); }
return (int) msg.wParam; }
// // 函數(shù):MyRegisterClass() // // 目的:注冊窗口類。 // // 注釋: // // 僅當(dāng)希望在已添加到 Windows 95 的 // “RegisterClassEx”函數(shù)之前此代碼與 Win32 系統(tǒng)兼容時(shí), // 才需要此函數(shù)及其用法。調(diào)用此函數(shù) // 十分重要,這樣應(yīng)用程序就可以獲得關(guān)聯(lián)的 // “格式正確的”小圖標(biāo)。 // ATOM MyRegisterClass(HINSTANCE hInstance) { WNDCLASSEX wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = 0; wcex.lpfnWndProc = (WNDPROC)WndProc; wcex.cbClsExtra = 0; wcex.cbWndExtra = 0; wcex.hInstance = hInstance; wcex.hIcon = LoadIcon(hInstance, (LPCTSTR)IDI_GLTEST); wcex.hCursor = LoadCursor(NULL, IDC_ARROW); wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1); wcex.lpszMenuName = (LPCTSTR)IDC_GLTEST; wcex.lpszClassName = szWindowClass; wcex.hIconSm = LoadIcon(wcex.hInstance, (LPCTSTR)IDI_SMALL);
return RegisterClassEx(&wcex); }
// // 函數(shù):InitInstance(HANDLE, int) // // 目的:保存實(shí)例句柄并創(chuàng)建主窗口 // // 注釋: // // 在此函數(shù)中,我們在全局變量中保存實(shí)例句柄并 // 創(chuàng)建和顯示主程序窗口。 // BOOL InitInstance(HINSTANCE hInstance, int nCmdShow) { hInst = hInstance; // 將實(shí)例句柄存儲在全局變量中
CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
if ( !g_hWnd ) { return FALSE; } OnCreated();
ShowWindow( g_hWnd, nCmdShow ); UpdateWindow( g_hWnd);
return TRUE; }
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { int wmId, wmEvent; switch (message) { case WM_CREATE: OnCreate( hWnd ); break; case WM_COMMAND: wmId = LOWORD(wParam); wmEvent = HIWORD(wParam); // 分析菜單選擇: switch (wmId) { case IDM_about: DialogBox(hInst, (LPCTSTR)IDD_ABOUTBOX, hWnd, (DLGPROC)About); break; case IDM_EXIT: DestroyWindow(hWnd); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } break; case WM_SIZE: SetProjMatrix( LOWORD( lParam ), HIWORD( lParam ) ); break; case WM_DESTROY: OnDestroy(); PostQuitMessage(0); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } return 0; }
// “關(guān)于”框的消息處理程序。 LRESULT CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam) { switch (message) { case WM_INITDIALOG: return TRUE;
case WM_COMMAND: if (LOWORD(wParam) == IDOK LOWORD(wParam) == IDCANCEL) { EndDialog(hDlg, LOWORD(wParam)); return TRUE; } break; } return FALSE; }
void OnCreate( HWND hWnd ) { g_hWnd = hWnd; }
void OnDestroy( void ) { ReleaseDC( g_hWnd, g_hDC ); wglDeleteContext( g_glRes ); }
void OnCreated( void ) { g_hDC = GetDC( g_hWnd ); PIXELFORMATDESCRIPTOR pfd; ZeroMemory( &pfd, sizeof(PIXELFORMATDESCRIPTOR) ); pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR ); pfd.nVersion = 1; pfd.dwFlags = PFD_DRAW_TO_WINDOW PFD_SUPPORT_OPENGL PFD_DOUBLEBUFFER; pfd.iPixelType = PFD_TYPE_RGBA; pfd.cColorBits = 24; pfd.cDepthBits = 32;
SetPixelFormat( g_hDC, ChoosePixelFormat( g_hDC, &pfd ), &pfd );
g_glRes = wglCreateContext( g_hDC ); wglMakeCurrent( g_hDC, g_glRes );
glEnable( GL_CULL_FACE ); glCullFace( GL_BACK );
glEnable( GL_DEPTH_TEST ); glDepthFunc( GL_LEQUAL );
int LightPos[] = { 50, 50, 10, 1 }; float LightColor[] = { 0.3f, 0.3f, 0.3f, 1.0f };
glEnable(GL_LIGHTING); glLightiv( GL_LIGHT0, GL_POSITION, LightPos ); glLightfv( GL_LIGHT0, GL_AMBIENT, LightColor ); glLightfv( GL_LIGHT0, GL_DIFFUSE, LightColor ); glEnable( GL_LIGHT0 ); glEnable( GL_COLOR_MATERIAL ); glColorMaterial( GL_FRONT, GL_AMBIENT_AND_DIFFUSE );
glShadeModel( GL_SMOOTH );
glClearColor( 255.0f / 255.0f, 255.0f / 255.0f, 200.0f / 255.0f, 0.0 ); glColor3ub( 140, 200, 255 ); }
void OnDraw( void ) { glClear( GL_COLOR_BUFFER_BIT GL_DEPTH_BUFFER_BIT ); glBegin( GL_QUADS ); // 設(shè)置繪制模式,我們畫一個(gè)平面的四邊形 glVertex2i( 5, 5 ); glVertex2i( -5, 5 ); glVertex2i( -5, -5 ); glVertex2i( 5, -5 ); glEnd(); SwapBuffers( g_hDC ); // 交換前后緩沖,雙緩沖無閃爍 }
void SetModalMatrix( void ) { glMatrixMode( GL_MODELVIEW ); glLoadIdentity( ); static float fRadius = 0; fRadius += 0.01f; if ( fRadius > M_PI * 2 ) { fRadius = 0; } gluLookAt( cosf( fRadius ) * 30, sinf( fRadius ) * 30, 15.0, 0.0, 0.0, 0.0, 0.0, 0.0, 1.0 ); }
void SetProjMatrix( WORD wWidth, WORD wHeight ) { glViewport( 0, 0, wWidth, wHeight ); glMatrixMode( GL_PROJECTION ); glLoadIdentity( ); gluPerspective( 45.0, (double)wWidth / (double)wHeight, 1.0, 1000.0 ); }
void OnIdle( void ) { SetModalMatrix(); OnDraw(); }
現(xiàn)在程序可以運(yùn)行了,告訴我你看到了什么?應(yīng)該是一個(gè)旋轉(zhuǎn)的平面四邊形。你什么也沒有看到?請仔細(xì)復(fù)查你上面寫的程序,看是不是每一句都和我一樣。如果你仍然得不到解決,可以發(fā)短消息給我來爭取獲得幫助的機(jī)會。但請注意,我并不會對每一個(gè)愚蠢的問題都做出答復(fù),比如請不要問我為什么你的窗體創(chuàng)建不出來或VC.net在哪里下載。
當(dāng)然,剛才說過我們是要繪制一個(gè)立方盒,我當(dāng)然沒有忘記,只是在這之前想讓更多的讀者熟悉一下OpenGL的繪圖機(jī)制。其它方盒很簡單,我們應(yīng)該有這樣一組數(shù)據(jù),并把它做為全局變量。每個(gè)頂點(diǎn)用X、Y、Z三維short值來表示,一個(gè)八個(gè)頂點(diǎn),你可以跟據(jù)這些數(shù)據(jù)在紙上手畫一個(gè)正方體出來嗎?
short nSrcBox[ 3 * 8 ] = { 5, 5, 0, 5, 5, 10, 5, -5, 0, 5, -5, 10, -5, -5, 0, -5, -5, 10, -5, 5, 0, -5, 5, 10, };
在繪制時(shí)我們可以一個(gè)三角形一個(gè)三角形的畫,也可以像上一個(gè)例子一樣,一個(gè)四邊形一個(gè)四邊形的畫,但為了后面我們要講述的一些東東,先看三角形畫吧。每一個(gè)面由兩個(gè)三角形組成,一共六個(gè)面,十二個(gè)三角形。如果給上面的八個(gè)頂點(diǎn)編號為0-7,那么我們可以按照這樣的順序來畫:
0, 4, 6, 0, 2, 4, 0, 6, 7, 0, 7, 1, 0, 3, 2, 0, 1, 3, 5, 2, 3, 5, 4, 2, 5, 6, 4, 5, 7, 6, 5, 1, 7, 5, 3, 1,
這里需要注意的是右手定則,在OpenGL中按照繪制一個(gè)三角形三維頂點(diǎn)的順序,用右手正正握這個(gè)三角形,大姆指豎起的方向就是這個(gè)面的正方向。其實(shí)你已經(jīng)在不知不覺中了解了什么是索引數(shù)組,就是上面那一堆數(shù)字,我們把他們保存起來,做為全局變量:
BYTE byIndex[36] ={ 0, 4, 6, 0, 2, 4, 0, 6, 7, 0, 7, 1, 0, 3, 2, 0, 1, 3, 5, 2, 3, 5, 4, 2, 5, 6, 4, 5, 7, 6, 5, 1, 7, 5, 3, 1, };
好了,現(xiàn)在是萬事具體只欠東風(fēng)了,怎么畫呢?也許你會想到和上面的程序一樣調(diào)用一個(gè)個(gè)的glVertex來繪制這些頂點(diǎn),是的,你是正確的,但這樣太慢而且太復(fù)雜了,算算你一共要寫多少個(gè)glVertex呢?幸好OpenGL為我們提供了一個(gè)方便繪圖機(jī)制:數(shù)組繪制。在初始化代碼的最下面加上這樣的語句來指定頂點(diǎn)數(shù)組:
glEnableClientState( GL_VERTEX_ARRAY ); // 啟用頂點(diǎn)數(shù)組 glVertexPointer( 3, GL_SHORT, 0, nSrcBox ); // 設(shè)置頂點(diǎn)數(shù)組地址
然后在OnDraw的Clear和Swap之間加上這一句話就足夠了:
// 后兩個(gè)參數(shù)指定索引數(shù)組值的類型和數(shù)組地址 glDrawElements( GL_TRIANGLES, 36, GL_UNSIGNED_BYTE, byIndex );
你應(yīng)該可以看到一個(gè)旋轉(zhuǎn)的立方體了,但它仍然是混沌一片,看不清棱角,這是由于你沒有指定光線反射的方向,下面我們將再引入一個(gè)概念“法向量”。在現(xiàn)實(shí)生活中,人眼看到物體是由于物體對光線的反射,如果物體不反射光,或著不向人的眼睛方向反射,人眼將認(rèn)為這個(gè)物體是沒有光照的。先不說漫反射,每一個(gè)鏡面反射的物體的入射角和射出角都是相等的,因而鏡面反射物體的法向量因該是垂直于平面的。由于光的粒子性,漫反射物體我們可以認(rèn)為是由無限多個(gè)鏡面反射物體所組成的一種復(fù)雜物體。根據(jù)這些理論,OpenGL在繪制三維物體時(shí)將按照給定的頂點(diǎn)法向量值來計(jì)算這個(gè)面的明暗亮度。計(jì)算法向量要用到一個(gè)數(shù)學(xué)基礎(chǔ)知識,可能對于一些人來說有些晦澀難懂,不過沒有關(guān)系,這里有現(xiàn)成的函數(shù),你直接調(diào)用就可以計(jì)算了:
// 歸一化函數(shù),沒什么可說的 template HRESULT ReduceToUnit( Type *pVector ) { double dLength = sqrt( (double)pVector[0] * (double)pVector[0] + (double)pVector[1] * (double)pVector[1] + (double)pVector[2] * (double)pVector[2] );
if ( FLOATEQUAL( dLength, 0.0, 1e-8 ) ) { dLength = 1.0; } pVector[0] /= (Type)dLength; pVector[1] /= (Type)dLength; pVector[2] /= (Type)dLength; return S_OK; }
// 第一個(gè)參數(shù)用9個(gè)double表示一個(gè)三角型,三角型的法向量將由第二個(gè)參數(shù)傳出。 template HRESULT CalcNormal( const T1 *pVertical, T2 *pNormal ) { T1 d1[3], d2[3]; d1[0] = pVertical[0] - pVertical[3]; d1[1] = pVertical[1] - pVertical[4]; d1[2] = pVertical[2] - pVertical[5]; d2[0] = pVertical[3] - pVertical[6]; d2[1] = pVertical[4] - pVertical[7]; d2[2] = pVertical[5] - pVertical[8]; pNormal[0] = (T2)( d1[1] * d2[2] - d1[2] * d2[1] ); pNormal[1] = (T2)( d1[2] * d2[0] - d1[0] * d2[2] ); pNormal[2] = (T2)( d1[0] * d2[1] - d1[1] * d2[0] ); return ReduceToUnit( pNormal ); }
問題又出現(xiàn)了,一個(gè)頂點(diǎn)可以屬于不同的面,但是法向量確只有一個(gè),給個(gè)頂點(diǎn)賦任何一個(gè)面的法向量都將導(dǎo)致顯示不正常,那么我們該怎么辦呢?只好按照索引將這些頂點(diǎn)“拆開”。
for ( int i = 0; i < 36;="" i++=""> { // nTempBox是一個(gè)臨時(shí)數(shù)組變量,用于保存拆開后的數(shù)據(jù)。 MoveMemory( &nTempBox[ i * 3 ], &nSrcBox[ byIndex[i] * 3 ], sizeof(nTempBox[0]) * 3 ); }
下面我們就可以計(jì)算法向量了。
for ( int i = 0; i < 12;="" i++=""> { // 跟據(jù)一個(gè)三角形的三個(gè)頂點(diǎn)計(jì)算出該三角形的法向量 CalcNormal( &nTempBox[ i * 9 ], &fNormal[ i * 9 ] ); // 并把這個(gè)法向量賦給這個(gè)三角形的三個(gè)頂點(diǎn) MoveMemory( &fNormal[ i * 9 + 3 ], &fNormal[ i * 9 ], sizeof(fNormal[0]) * 3 ); MoveMemory( &fNormal[ i * 9 + 6 ], &fNormal[ i * 9 ], sizeof(fNormal[0]) * 3 ); } glEnableClientState( GL_VERTEX_ARRAY ); glVertexPointer( 3, GL_SHORT, 0, nTempBox ); glEnableClientState( GL_NORMAL_ARRAY ); // 啟用法向量 glNormalPointer( GL_FLOAT, 0, fNormal ); // 填寫法向量數(shù)組地址
因?yàn)橛貌恢饕龜?shù)組,下面的繪制代碼將更加簡單:
glDrawArrays( GL_TRIANGLES, 0, 36 );
好了,現(xiàn)在整篇代碼應(yīng)該是這個(gè)樣子了:
// GlTest.cpp : 定義應(yīng)用程序的入口點(diǎn)。 //
#include "stdafx.h" #include "GlTest.h" #define MAX_LOADSTRING 100
// 全局變量: HINSTANCE hInst; // 當(dāng)前實(shí)例 HWND g_hWnd; HDC g_hDC; HGLRC g_glRes;
short nSrcBox[ 3 * 8 ] = { 5, 5, 0, 5, 5, 10, 5, -5, 0, 5, -5, 10, -5, -5, 0, -5, -5, 10, -5, 5, 0, -5, 5, 10, };
BYTE byIndex[36] ={ 0, 4, 6, 0, 2, 4, 0, 6, 7, 0, 7, 1, 0, 3, 2, 0, 1, 3, 5, 2, 3, 5, 4, 2, 5, 6, 4, 5, 7, 6, 5, 1, 7, 5, 3, 1, };
short nTempBox[ 36 * 3 ]; float fNormal[ 36 * 3 ];
TCHAR szTitle[MAX_LOADSTRING]; // 標(biāo)題欄文本 TCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口類名
template HRESULT ReduceToUnit( Type *pVector ) { double dLength = sqrt( (double)pVector[0] * (double)pVector[0] + (double)pVector[1] * (double)pVector[1] + (double)pVector[2] * (double)pVector[2] );
if ( FLOATEQUAL( dLength, 0.0, 1e-8 ) ) { dLength = 1.0; } pVector[0] /= (Type)dLength; pVector[1] /= (Type)dLength; pVector[2] /= (Type)dLength; return S_OK; }
template HRESULT CalcNormal( const T1 *pVertical, T2 *pNormal ) { T1 d1[3], d2[3]; d1[0] = pVertical[0] - pVertical[3]; d1[1] = pVertical[1] - pVertical[4]; d1[2] = pVertical[2] - pVertical[5]; d2[0] = pVertical[3] - pVertical[6]; d2[1] = pVertical[4] - pVertical[7]; d2[2] = pVertical[5] - pVertical[8]; pNormal[0] = (T2)( d1[1] * d2[2] - d1[2] * d2[1] ); pNormal[1] = (T2)( d1[2] * d2[0] - d1[0] * d2[2] ); pNormal[2] = (T2)( d1[0] * d2[1] - d1[1] * d2[0] ); return ReduceToUnit( pNormal ); }
// 此代碼模塊中包含的函數(shù)的前向聲明: void OnCreate( HWND hWnd ); void OnCreated( void ); void OnDestroy( void ); void OnDraw( void ); void SetProjMatrix( WORD wWidth, WORD wHeight ); void SetModalMatrix( void ); void OnIdle( void );
ATOM MyRegisterClass(HINSTANCE hInstance); BOOL InitInstance(HINSTANCE, int); LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM); LRESULT CALLBACK About(HWND, UINT, WPARAM, LPARAM);
int APIENTRY _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow) { // TODO: 在此放置代碼。 MSG msg; HACCEL hAccelTable;
// 初始化全局字符串 LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING); LoadString(hInstance, IDC_GLTEST, szWindowClass, MAX_LOADSTRING); MyRegisterClass(hInstance);
// 執(zhí)行應(yīng)用程序初始化: if (!InitInstance (hInstance, nCmdShow)) { return FALSE; }
hAccelTable = LoadAccelerators(hInstance, (LPCTSTR)IDC_GLTEST);
// 主消息循環(huán): while ( true ) { if ( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ) ) { if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } continue; } if ( WM_QUIT == msg.message ) { break; } OnIdle(); }
return (int) msg.wParam; }
ATOM MyRegisterClass(HINSTANCE hInstance) { WNDCLASSEX wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = 0; wcex.lpfnWndProc = (WNDPROC)WndProc; wcex.cbClsExtra = 0; wcex.cbWndExtra = 0; wcex.hInstance = hInstance; wcex.hIcon = LoadIcon(hInstance, (LPCTSTR)IDI_GLTEST); wcex.hCursor = LoadCursor(NULL, IDC_ARROW); wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1); wcex.lpszMenuName = (LPCTSTR)IDC_GLTEST; wcex.lpszClassName = szWindowClass; wcex.hIconSm = LoadIcon(wcex.hInstance, (LPCTSTR)IDI_SMALL);
return RegisterClassEx(&wcex); }
// // 函數(shù):InitInstance(HANDLE, int) // // 目的:保存實(shí)例句柄并創(chuàng)建主窗口 // // 注釋: // // 在此函數(shù)中,我們在全局變量中保存實(shí)例句柄并 // 創(chuàng)建和顯示主程序窗口。 // BOOL InitInstance(HINSTANCE hInstance, int nCmdShow) { hInst = hInstance; // 將實(shí)例句柄存儲在全局變量中
CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
if ( !g_hWnd ) { return FALSE; } OnCreated();
ShowWindow( g_hWnd, nCmdShow ); UpdateWindow( g_hWnd);
return TRUE; }
// // 函數(shù):WndProc(HWND, unsigned, WORD, LONG) // // 目的:處理主窗口的消息。 // // WM_COMMAND - 處理應(yīng)用程序菜單 // WM_PAINT - 繪制主窗口 // WM_DESTROY - 發(fā)送退出消息并返回 // // LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { int wmId, wmEvent; switch (message) { case WM_CREATE: OnCreate( hWnd ); break; case WM_KEYDOWN: g_bStartSwap = !g_bStartSwap; break; case WM_MOVE: SetWindowText( g_hWnd, "Move" ); break; case WM_COMMAND: wmId = LOWORD(wParam); wmEvent = HIWORD(wParam); // 分析菜單選擇: switch (wmId) { case IDM_about: DialogBox(hInst, (LPCTSTR)IDD_ABOUTBOX, hWnd, (DLGPROC)About); break; case IDM_EXIT: DestroyWindow(hWnd); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } break; case WM_SIZE: SetProjMatrix( LOWORD( lParam ), HIWORD( lParam ) ); break; case WM_DESTROY: OnDestroy(); PostQuitMessage(0); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } return 0; }
// “關(guān)于”框的消息處理程序。 LRESULT CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam) { switch (message) { case WM_INITDIALOG: return TRUE;
case WM_COMMAND: if (LOWORD(wParam) == IDOK LOWORD(wParam) == IDCANCEL) { EndDialog(hDlg, LOWORD(wParam)); return TRUE; } break; } return FALSE; }
void OnCreate( HWND hWnd ) { g_hWnd = hWnd; }
void OnDestroy( void ) { ReleaseDC( g_hWnd, g_hDC ); wglDeleteContext( g_glRes ); }
void OnCreated( void ) { g_hDC = GetDC( g_hWnd );
PIXELFORMATDESCRIPTOR pfd; ZeroMemory( &pfd, sizeof(PIXELFORMATDESCRIPTOR) ); pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR ); pfd.nVersion = 1; pfd.dwFlags = PFD_DRAW_TO_WINDOW PFD_SUPPORT_OPENGL PFD_DOUBLEBUFFER; pfd.iPixelType = PFD_TYPE_RGBA; pfd.cColorBits = 24; pfd.cDepthBits = 32;
SetPixelFormat( g_hDC, ChoosePixelFormat( g_hDC, &pfd ), &pfd );
g_glRes = wglCreateContext( g_hDC ); wglMakeCurrent( g_hDC, g_glRes );
glEnable( GL_CULL_FACE ); glCullFace( GL_BACK );
glEnable( GL_DEPTH_TEST ); glDepthFunc( GL_LEQUAL );
int LightPos[] = { 50, 50, 10, 1 }; float LightColor[] = { 0.3f, 0.3f, 0.3f, 1.0f };
glEnable(GL_LIGHTING); glLightiv( GL_LIGHT0, GL_POSITION, LightPos ); glLightfv( GL_LIGHT0, GL_AMBIENT, LightColor ); glLightfv( GL_LIGHT0, GL_DIFFUSE, LightColor ); glEnable( GL_LIGHT0 ); glEnable( GL_COLOR_MATERIAL ); glColorMaterial( GL_FRONT, GL_AMBIENT_AND_DIFFUSE );
glShadeModel( GL_SMOOTH );
glClearColor( 255.0f / 255.0f, 255.0f / 255.0f, 200.0f / 255.0f, 0.0 ); for ( int i = 0; i < 36;="" i++=""> { MoveMemory( &nTempBox[ i * 3 ], &nSrcBox[ byIndex[i] * 3 ], sizeof(nTempBox[0]) * 3 ); } for ( int i = 0; i < 12;="" i++=""> { CalcNormal( &nTempBox[ i * 9 ], &fNormal[ i * 9 ] ); MoveMemory( &fNormal[ i * 9 + 3 ], &fNormal[ i * 9 ], sizeof(fNormal[0]) * 3 ); MoveMemory( &fNormal[ i * 9 + 6 ], &fNormal[ i * 9 ], sizeof(fNormal[0]) * 3 ); } glEnableClientState( GL_VERTEX_ARRAY ); glVertexPointer( 3, GL_SHORT, 0, nTempBox ); glEnableClientState( GL_NORMAL_ARRAY ); glNormalPointer( GL_FLOAT, 0, fNormal );
glColor3ub( 140, 200, 255 ); }
void OnDraw( void ) { glClear( GL_COLOR_BUFFER_BIT GL_DEPTH_BUFFER_BIT ); glDrawArrays( GL_TRIANGLES, 0, 36 ); SwapBuffers( g_hDC ); }
void SetModalMatrix( void ) { glMatrixMode( GL_MODELVIEW ); glLoadIdentity( ); static float fRadius = 0; fRadius += 0.01f; if ( fRadius > M_PI * 2 ) { fRadius = 0; } gluLookAt( cosf( fRadius ) * 30, sinf( fRadius ) * 30, 15.0, 0.0, 0.0, 0.0, 0.0, 0.0, 1.0 ); }
void SetProjMatrix( WORD wWidth, WORD wHeight ) { glViewport( 0, 0, wWidth, wHeight ); glMatrixMode( GL_PROJECTION ); glLoadIdentity( ); gluPerspective( 45.0, (double)wWidth / (double)wHeight, 1.0, 1000.0 ); }
void OnIdle( void ) { SetModalMatrix(); OnDraw(); }
現(xiàn)在已經(jīng)達(dá)到我們想要的效果了,心情很愉快對嗎?
| |