专业图形程序接口OpenGL的工程运用.docx
《专业图形程序接口OpenGL的工程运用.docx》由会员分享,可在线阅读,更多相关《专业图形程序接口OpenGL的工程运用.docx(15页珍藏版)》请在冰豆网上搜索。
专业图形程序接口OpenGL的工程运用
功能强大的专业图形程序接口OpenGL
摘要
OpenGL是个专业的图形程序接口,是一个功能强大,调用方便的底层图形库。
OpenGL是个与硬件无关的软件接口,可以在不同的平台之间进行移植。
因此,支持OpenGL的软件具有很好的移植性,可以获得非常广泛的应用。
目录
概述
OpenGL现状
高级功能
OpenGL与DirectX的区别
OpenGL编程入门
概述
OpenGL是个专业的图形程序接口,是一个功能强大,调用方便的底层图形库。
OpenGL的前身是SGI公司为其图形工作站开发的IRISGL。
IRISGL是一个工业标准的3D图形软件接口,功能虽然强大但是移植性不好,于是SGI公司便在IRISGL的基础上开发了OpenGL。
OpenGL的英文全称是“OpenGraphicsLibrary”,顾名思义,OpenGL便是“开放的图形程序接口”。
虽然DirectX在家用市场全面领先,但在专业高端绘图领域,OpenGL是不能被取代的主角。
OpenGL是个与硬件无关的软件接口,可以在不同的平台如Windows95、WindowsNT、Unix、Linux、MacOS、OS/2之间进行移植。
因此,支持OpenGL的软件具有很好的移植性,可以获得非常广泛的应用。
由于OpenGL是图形的底层图形库,没有提供几何实体图元,不能直接用以描述场景。
但是,通过一些转换程序,可以很方便地将AutoCAD、3DS/3DSMAX等3D图形设计软件制作的DFX和3DS模型文件转换成OpenGL的顶点数组。
在OpenGL的基础上还有OpenInventor、Cosmo3D、Optimizer等多种高级图形库,适应不同应用。
其中,OpenInventor应用最为广泛。
该软件是基于OpenGL面向对象的工具包,提供创建交互式3D图形应用程序的对象和方法,提供了预定义的对象和用于交互的事件处理模块,创建和编辑3D场景的高级应用程序单元,有打印对象和用其它图形格式交换数据的能力。
OpenGL的发展一直处于一种较为迟缓的态势,每次版本的提高新增的技术很少,大多只是对其中部分做出修改和完善。
1992年7月,SGI公司发布了OpenGL的1.0版本,随后又与微软公司共同开发了WindowsNT版本的OpenGL,从而使一些原来必须在高档图形工作站上运行的大型3D图形处理软件也可以在微机上运用。
1995年OpenGL的1.1版本面市,该版本比1.0的性能有许多提高,并加入了一些新的功能。
其中包括改进打印机支持,在增强元文件中包含OpenGL的调用,顶点数组的新特性,提高顶点位置、法线、颜色、色彩指数、纹理坐标、多边形边缘标识的传输速度,引入了新的纹理特性等等。
OpenGL1.5又新增了“OpenGLShadingLanguage”,该语言是“OpenGL2.0”的底核,用于着色对象、顶点着色以及片断着色技术的扩展功能。
OpenGL2.0标准的主要制订者并非原来的SGI,而是逐渐在ARB中占据主动地位的3DLabs。
2.0版本首先要做的是与旧版本之间的完整兼容性,同时在顶点与像素及内存管理上与DirectX共同合作以维持均势。
OpenGL2.0将由OpenGL1.3的现有功能加上与之完全兼容的新功能所组成(如图一)。
借此可以对在ARB停滞不前时代各家推出的各种纠缠不清的扩展指令集做一次彻底的精简。
此外,硬件可编程能力的实现也提供了一个更好的方法以整合现有的扩展指令。
目前,随着DirectX的不断发展和完善,OpenGL的优势逐渐丧失,至今虽然已有3Dlabs提倡开发的2.0版本面世,在其中加入了很多类似于DirectX中可编程单元的设计,但厂商的用户的认知程度并不高,未来的OpenGL发展前景迷茫。
OpenGL现状
OpenGL仍然是唯一能够取代微软对3D图形技术的完全控制的API。
它仍然具有一定的生命力,但是SiliconGraphics已经不再以任何让微软不悦的方式推广OpenGL,因而它存在较高的风险。
游戏开发人员是一个有着独立思想的群体,很多重要的开发人员目前仍然在使用OpenGL。
因此,硬件开发商正在设法加强对它的支持。
Direct3D目前还不能支持高端的图形设备和专业应用;OpenGL在这些领域占据着统治地位。
最后,开放源码社区(尤其是Mesa项目)一直致力于为任何类型的计算机(无论它们是否使用微软的操作系统)提供OpenGL支持。
高级功能
OpenGL被设计为只有输出的,所以它只提供渲染功能。
核心API没有窗口系统、音频、打印、键盘/鼠标或其它输入设备的概念。
虽然这一开始看起来像是一种限制,但它允许进行渲染的代码完全独立于他运行的操作系统,允许跨平台开发。
然而,有些整合于原生窗口系统的东西需要允许和宿主系统交互。
这通过下列附加API实现:
*GLX-X11(包括透明的网络)
*WGL-MicrosoftWindows
*AGL-AppleMacOS
另外,GLUT库能够以可移植的方式提供基本的窗口功能。
OpenGL与DirectX的区别
OpenGL只是图形函数库。
DirectX包含图形,声音,输入,网络等模块。
OpenGL稳定,可跨平台使用。
DirectX仅能用于Windows系列平台,包括WindowsMobile/CE系列以及XBOX/XBOX360。
----------------------------------------------------------------------------------------------
1995年至1996年,微软实行了一项新计划,以支持在Windows95上运行游戏,目标是把市场扩展到被任天堂和世嘉控制的游戏领域。
然而,微软不想用已经在NT上提供的OpenGL技术。
微软收购了Rendermorphics,Ltd.并得到他的被称作RealityLab的3DAPI。
经重新整理,微软发布了新的3DAPI——Direct3D。
微软,推行Direct3D,冻结OpenGL!
微软当时拒绝了在Window95上支持OpenGL。
不止如此,微软采取异常手段收回对OpenGL的MCD驱动接口的支持,以致硬件厂商不得不放弃已经进入最后测试的OpenGL驱动。
微软的市场部门开始向游戏开发商、硬件厂商、新闻出版机构推销Direct3D,同时排斥OpenGL。
API之战!
SiliconGraphics和很多OpenGL用户都依赖OpenGL创新且高性能的技术。
但很明显微软打算用Direct3D代替OpenGL,尽管D3D有很多问题而且不能像OpenGL那样被硬件厂商扩展。
SiliconGraphics决定在1996SIGGRAPH会议上作一项演示。
演示证明OpenGL至少和D3D一样快,从而驳倒微软的市场论调。
因为OpenGL是业界公认标准,比D3D功能丰富,而且图像质量要高一些,所以演示在计算机图形和游戏开发社区导致了激烈论战。
游戏开发者要求OpenGL和D3D站在同等地位!
当技术和市场问题暴露,强烈的支持OpenGL行动开始了。
Doom的开发者JohnCarmack声明拒绝D3D,ChrisHecker在游戏开发杂志上发表了两套API的全面分析,移微软应放弃D3D为结论。
游戏开发者先后两次向微软递交请愿书。
第一次由56名首席游戏开发者要求微软发行OpenGLMCD驱动,但未成功,因为会让OpenGL与D3D竞争。
第二次的公开信由254人签名开始,截止时达到1400人。
微软的回答仍是重申旧市场立场。
尽管请愿者清楚的要求两套API同等竞争以促进发展,微软却以增加D3D的投资、更加减少OpenGL的投资为回应。
Fahrenheit——D3D与OpenGL的合并?
SiliconGraphics,Microsoft,HP,Intel达成协议联合开发下一代3DAPI——Fahrenheit。
但不了了之,因为微软的打算是把OpenGL的技术用到D3D里并且以此之名驱除OpenGL的威胁。
(估计DirectX8Graphics即是剩下微软独自开发的Fahrenheit,吸收了OpenGL的很多东西。
)
OpenGL豪气不减当年!
OpenGL依然是唯一能与微软单独控制的D3D对立的API,尽管SiliconGraphics不再以任何微软不能接受的方式推行OpenGL。
游戏开发这是独立的,并且很多关键人物在用OpenGL,因此,硬件厂商正努力提高对其支持。
D3D仍不能支持高端图像和专业应用,而OpenGL主宰着这些土地。
在开放原码社区,Mesa项目正提供独立于微软的OpenGL驱动。
译者注:
表面上好像D3D比OpenGL支持更多的功能,其实由于D3D不支持硬件扩展,如硬件全景阴影,硬件渲染顺序无关半透明材质等新技术根本无法使用,而D3D(特指D3D8)本身提供的功能只有一小部分能在使用HAL且硬件不支持时模拟,你要用大量代码分析硬件能力和采取不同策略.
OpenGL编程入门
OpenGL作图非常方便,故日益流行,但对许多人来说,是在微机上进行的,首先碰到的问题是,如何适应微机环境。
这往往是最关键的一步,虽然也是最初级的。
一般的,我不建议使用glut包.那样难以充分发挥windows的界面上的功能.
下面介绍如何在VC++上进行OpenGL编程。
OpenGL绘图的一般过程可以看作这样的,先用OpenGL语句在OpenGL的绘图环境RenderContext(RC)中画好图,然后再通过一个Swapbuffer的过程把图传给操作系统的绘图环境DeviceContext(DC)中,实实在在地画出到屏幕上.
下面以画一条Bezier曲线为例,详细介绍VC++上OpenGL编程的方法。
文中给出了详细注释,以便给初学者明确的指引。
一步一步地按所述去做,你将顺利地画出第一个OpenGL平台上的图形来。
一、产生程序框架Test.dsw
NewProject|MFCApplicationWizard(EXE)|"Test"|OK
*注*:
加“”者指要手工敲入的字串
二、导入Bezier曲线类的文件
用下面方法产生BezierCurve.hBezierCurve.cpp两个文件:
WorkSpace|ClassView|TestClasses|<右击弹出>NewClass|GenericClass(不用MFC类)|"CBezierCurve"|OK
三、编辑好Bezier曲线类的定义与实现
写好下面两个文件:
BezierCurve.hBezierCurve.cpp
四、设置编译环境:
1.在BezierCurve.h和TestView.h内各加上:
#include
#include
#include
2.在集成环境中
Project|Settings|Link|Object/librarymodule|"opengl32.libglu32.libglaux.lib"|OK
五、设置OpenGL工作环境:
(下面各个操作,均针对TestView.cpp)
1.处理PreCreateWindow():
设置OpenGL绘图窗口的风格
cs.style|=WS_CLIPSIBLINGS|WS_CLIPCHILDREN|CS_OWNDC;
2.处理OnCreate():
创建OpenGL的绘图设备。
OpenGL绘图的机制是:
先用OpenGL的绘图上下文RenderingContext(简称为RC)把图画好,再把所绘结果通过SwapBuffer()函数传给Window的绘图上下文DeviceContext(简记为DC).要注意的是,程序运行过程中,可以有多个DC,但只能有一个RC。
因此当一个DC画完图后,要立即释放RC,以便其它的DC也使用。
在后面的代码中,将有详细注释。
intCTestView:
:
OnCreate(LPCREATESTRUCTlpCreateStruct)
{
if(CView:
:
OnCreate(lpCreateStruct)==-1)
return-1;
myInitOpenGL();
return0;
}
voidCTestView:
:
myInitOpenGL()
{
m_pDC=newCClientDC(this);//创建DC
ASSERT(m_pDC!
=NULL);
if(!
mySetupPixelFormat())//设定绘图的位图格式,函数下面列出
return;
m_hRC=wglCreateContext(m_pDC->m_hDC);//创建RC
wglMakeCurrent(m_pDC->m_hDC,m_hRC);//RC与当前DC相关联
}//CClient*m_pDC;HGLRCm_hRC;是CTestView的成员变量
BOOLCTestView:
:
mySetupPixelFormat()
{//我们暂时不管格式的具体内容是什么,以后熟悉了再改变格式
staticPIXELFORMATDESCRIPTORpfd=
{
sizeof(PIXELFORMATDESCRIPTOR),//sizeofthispfd
1,//versionnumber
PFD_DRAW_TO_WINDOW|//supportwindow
PFD_SUPPORT_OPENGL|//supportOpenGL
PFD_DOUBLEBUFFER,//doublebuffered
PFD_TYPE_RGBA,//RGBAtype
24,//24-bitcolordepth
0,0,0,0,0,0,//colorbitsignored
0,//noalphabuffer
0,//shiftbitignored
0,//noaccumulationbuffer
0,0,0,0,//accumbitsignored
32,//32-bitz-buffer
0,//nostencilbuffer
0,//noauxiliarybuffer
PFD_MAIN_PLANE,//mainlayer
0,//reserved
0,0,0//layermasksignored
};
intpixelformat;
if((pixelformat=ChoosePixelFormat(m_pDC->m_hDC,&pfd))==0)
{
MessageBox("ChoosePixelFormatfailed");
returnFALSE;
}
if(SetPixelFormat(m_pDC->m_hDC,pixelformat,&pfd)==FALSE)
{
MessageBox("SetPixelFormatfailed");
returnFALSE;
}
returnTRUE;
}
3.处理OnDestroy()
voidCTestView:
:
OnDestroy()
{
wglMakeCurrent(m_pDC->m_hDC,NULL);//释放与m_hDC对应的RC
wglDeleteContext(m_hRC);//删除RC
if(m_pDC)
deletem_pDC;//删除当前View拥有的DC
CView:
:
OnDestroy();
}
4.处理OnEraseBkgnd()
BOOLCTestView:
:
OnEraseBkgnd(CDC*pDC)
{
//TODO:
Addyourmessagehandlercodehereand/orcalldefault
//returnCView:
:
OnEraseBkgnd(pDC);
//把这句话注释掉,若不然,Window
//会用白色北景来刷新,导致画面闪烁
returnTRUE;//只要空返回即可。
}
5.处理OnDraw()
voidCTestView:
:
OnDraw(CDC*pDC)
{
wglMakeCurrent(m_pDC->m_hDC,m_hRC);//使RC与当前DC相关联
myDrawScene();//具体的绘图函数,在RC中绘制
SwapBuffers(m_pDC->m_hDC);//把RC中所绘传到当前的DC上,从而
//在屏幕上显示
wglMakeCurrent(m_pDC->m_hDC,NULL);//释放RC,以便其它DC进行绘图
}
voidCTestView:
:
myDrawScene()
{
glClearColor(0.0f,0.0f,0.0f,1.0f);//设置背景颜色为黑色
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glPushMatrix();
glTranslated(0.0f,0.0f,-3.0f);//把物体沿(0,0,-1)方向平移
//以便投影时可见。
因为缺省的视点在(0,0,0),只有移开
//物体才能可见。
//本例是为了演示平面Bezier曲线的,只要作一个旋转
//变换,可更清楚的看到其3D效果。
//下面画一条Bezier曲线
bezier_curve.myPolygon();//画Bezier曲线的控制多边形
bezier_curve.myDraw();//CBezierCurvebezier_curve
//是CTestView的成员变量
//具体的函数见附录
glPopMatrix();
glFlush();//结束RC绘图
return;
}
6.处理OnSize()
voidCTestView:
:
OnSize(UINTnType,intcx,intcy)
{
CView:
:
OnSize(nType,cx,cy);
VERIFY(wglMakeCurrent(m_pDC->m_hDC,m_hRC));//确认RC与当前DC关联
w=cx;
h=cy;
VERIFY(wglMakeCurrent(NULL,NULL));//确认DC释放RC
}
7处理OnLButtonDown()
voidCTestView:
:
OnLButtonDown(UINTnFlags,CPointpoint)
{
CView:
:
OnLButtonDown(nFlags,point);
if(bezier_curve.m_N>MAX-1)
{
MessageBox("顶点个数超过了最大数MAX=50");
return;
}
//以下为坐标变换作准备
GetClientRect(&m_ClientRect);//获取视口区域大小
w=m_ClientRect.right-m_ClientRect.left;//视口宽度w
h=m_ClientRect.bottom-m_ClientRect.top;//视口高度h
//w,h是CTestView的成员变量
centerx=(m_ClientRect.left+m_ClientRect.right)/2;//中心位置,
centery=(m_ClientRect.top+m_ClientRect.bottom)/2;//取之作原点
//centerx,centery是CTestView的成员变量
GLdoubletmpx,tmpy;
tmpx=scrx2glx(point.x);//屏幕上点坐标转化为OpenGL画图的规范坐标
tmpy=scry2gly(point.y);
bezier_curve.m_Vertex[bezier_curve.m_N].x=tmpx;//加一个顶点
bezier_curve.m_Vertex[bezier_curve.m_N].y=tmpy;
bezier_curve.m_N++;//顶点数加一
InvalidateRect(NULL,TRUE);//发送刷新重绘消息
}
doubleCTestView:
:
scrx2glx(intscrx)
{
return(double)(scrx-centerx)/double(h);
}
doubleCTestView:
:
scry2gly(intscry)
{
}
附录:
1.CBezierCurve的声明:
(BezierCurve.h)
classCBezierCurve
{
public:
myPOINT2Dm_Vertex[MAX];//控制顶点,以数组存储
//myPOINT2D是一个存二维点的结构
//成员为Gldoublex,y
intm_N;//控制顶点的个数
public:
CBezierCurve();
virtual~CBezierCurve();
voidbezier_generation(myPOINT2DP[MAX],intlevel);
//算法的具体实现
voidmyDraw();//画曲线函数
voidmyPolygon();//画控制多边形
};
2.CBezierCurve的实现:
(BezierCurve.cpp)
CBezierCurve:
:
CBezierCurve()
{
m_N=4;
m_Vertex[0].x=