




已阅读5页,还剩5页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
计算机图形学课程设计 课程设计 球体Phong光照模型一、实验目的 (1)掌握双线性法矢插值模型; (2)掌握ZBuffer算法的思想; (3)掌握有效边表填充算法; 二、实验要求1、建立三维坐标系Oxyz,原点位于屏幕客户区中心,x轴水平向右为正,y轴垂直向上为正,z轴垂直于屏幕指向观察者。2、绘制体心和坐标系中心重合的球体表面,使用Z-Buffer消隐算法进行消隐。3、使用单点光源对球体进行照射生成Phong光照模型,光源位置位于球体右上方。4、背景色设置为RGB(128,0,0)。5、使用键盘方向键旋转球体。6、使用鼠标左击缩小球体、右击增大球体。三、实验步骤 建立球体的网格模型,使用地理划分法将球体北极和南极划分为三角形面片,其余部分划分为四边形面片,先对球体网格模型进行背面剔除,然后使用深度缓冲算法进行消隐。计算面片各顶点的平均法矢量,然后采用双线性法失插值计算面片内各点的法矢量。最终根据每点的法矢量对光源的朝向,通过简单光照模型计算所获得的光强。面片使用有效边表算法填1、Phong双线性法矢插值模型Gouraud双线性光强插值模型解决了相邻多边形之间的颜色突变问题,产生的真实感图形颜色过渡均匀,图形显得非常光滑,这是它的优点,但是,由于采用光强插值,其镜面反射光效果不太理想,而且相邻多边形边界处的马赫带效应并不能完全消除。Phong模型提出的双线性法矢插值模型可以有效的解决上述问题,产生正确的高光区域。Phong模型在进行光强插值的时候,需要先对面片的每一个顶点计算平均法矢量,然后通过双线性法矢插值计算面片内每个点的法矢量,最后根据简单光照模型计算面片上各点的颜色值。基本算法如下。(1)计算面片顶点的平均法矢量。由于球心位于三维坐标系原点,所以球面上任意面片的顶点平均法矢量就是该点的位置矢量。(2)计算面片内部各点的法矢量。在图中,三角形面片的顶点坐标为,法矢量为;法矢量是;任一扫面线于三角形边 ;为进行双线性值插值计算三角形内点F的法矢量。边边上任意一点A点的法矢量可以通过拉格朗日线性插值法得到: 边边上B点的法矢量通过拉格朗日线性插值法得到: 扫描线AB上F点的法矢量通过拉格朗日线性插值法得到: (3)对面片内的每一点根据简单光照模型计算光强,获得该点颜色。y A F B 2、修改CAET类 在CAET类内不仅包含边的起点坐标和终点坐标,同时增加起点和终点的法矢量。3、修改CZBuffer类在CZBuffer类先对面片每个点的法矢量进行双线性插值获得面片内每一点的法矢量,然后再调用简单光照模型计算面片内每点的光强。4、光照环境初始化在CTestView类的构造函数内设置光源个数为1,位于右上方,材质颜色为红色。5、绘制球面函数 使用Phong双线性法矢量插值模型时,需要计算每个面片上的矢法量。定义了Normal3数组存储三角形面片的顶点法矢量,定义了Normal4数组存储四边形面片的顶点法矢量。4、 程序核心代码及思路1、程序设计中的主要思路及所使用的主要类 在程序设计中,使用Phong双线性法矢量插值模型时,计算每个面片的顶点坐标,同时计算每个面片的顶点法矢量。在CZBuffer类内定义了双线性法矢量插值函数Interpolation()计算面片内的每个点的法矢量。根据面片内每一点的法矢量调用CLighting类的成员函数Lighing()计算该点的光强。并在MyView构造函数中调用了透视变化初始化函数 InitParameter(); 构造顶点表函数ReadPoint(); 构造面表函数ReadFace(); 在OnDraw()函数中使用双缓冲函数 ,并使用绘制球面函数DrawObject() 函数,画出球体模型。设计使用的类如下: (1)颜色类:CRGB 成员函数:Normalize() 将颜色分量red 、green、blue 规范化到 0, 1闭区间内。 (2)定义矢量类:CVector 成员函数: double Mold() 求矢量的模CVector Unit() 单位矢量 功能: 在类中重载 +、-、*、 等运算符,并利用Dot ()计算矢量点积 。 (3)定义边节点类:CAET 和 定义桶节点类:CBucket 设置当前扫描线与有效边的交点的横坐标 x 定义扫描线 ScanLine 来求 图形与有效边表的交点 (4)设计光源类: CLight void SetDiffuse(CRGB); 设置光源的漫反射光 void SetSpecular(CRGB); 设置光源的镜面反射光 void SetPosition(double,double,double); 设置光源的直角坐标系 void SetGlobal(double,double,double); 设置光源的球坐标 void SetCoef(double,double,double); 设置光强的衰减系数 void SetOnOff(bool); 设置光源开关状态 void GlobalToXYZ(); 球坐标转换为直角坐标 (5)设计材质类: CMaterial void SetAmbient(CRGB); 设置材质对环境光的反射率 void SetDiffuse(CRGB); 设置材质对漫反射光的反射率 void SetSpecular(CRGB); 设置材质对镜面反射光的反射率 void SetEmit(CRGB); 设置材质自身辐射的颜色 void SetExp(double); 设置材质的高光指数 (6)、设置光照类: CLighting 功能: 在类中定义了光照函数Lighting (),计算物体表面网格顶点所获得的光照函数 。在该函数中分五步来实现网格顶点的光亮度,第一,累加漫反射光的颜色;第二,累加镜面反射光的颜色;第三,进行光强衰减;第四,加入环境光;第五,返回所计算顶点的光强颜色。 (7)、定义CZBuffer 类: void CreateBucket(); 在函数中使用CBucket 类创建桶节点 void CreateEdge(); 在函数中使用CAET 类创建边表 void Phong(CDC *pDC,CPi3 ViewPoint,CLighting *pLight,CMaterial *pMaterial); Phong 填充函数 void InitDeepBuffer(int,int,double); 初始化深度缓存 CVector Interpolation(double,double,double,CVector,CVector); 法矢量线性插值 (2)程序中使用的重要函数及部分代码:(1)、构造球体顶点表函数 ReadPoint () 在函数中定义了片面夹角为 gafa=gbeta= 10; 纬度区域为 N1=180/gafa=18,经度区域 N2=360/gbeta=36 ; 利用数组 P(N1-1)*N2+2设置的球体共有616个顶点,经纬网格的夹角为 10。利用:P0.x=0,P0.y=r,P0.z=0; 计算北极点坐标利用如下代码计算球体上的点坐标: for(int i=0;iN1-1;i+)gafa1=(i+1)*gafa*3.14/180;for(int j=0;jN2;j+)gbeta1=j*gbeta*3.14/180;Pi*N2+j+1.x=r*sin(gafa1)*sin(gbeta1);Pi*N2+j+1.y=r*cos(gafa1);Pi*N2+j+1.z=r*sin(gafa1)*cos(gbeta1);利用:P(N1-1)*N2+1.x=0,P(N1-1)*N2+1.y=-r,P(N1-1)*N2+1.z=0;计算南极点坐标 (2)、构造面片表函数 ReadFace() 面片用二维数组表示,第一维按维度自北极向南极增加的方向定义,第二维在同一纬度带上z 轴正向开始,按逆时针方向定义。球体共有N1*N2 个面,北极和南极各有N2 个面,其余部分有 (N1-2)*N2个面片 。具体实现代码和算法见源程序。构造北极三角形面片代码:for(int j=0;jN2;j+)int tempj=j+1;if(tempj=N2) tempj=0;int NorthIndex3;NorthIndex0=0;NorthIndex1=j+1;NorthIndex2=tempj+1;F0j.SetEN(3);for(int k=0;kF0j.En;k+)F0j.pk=NorthIndexk;F0j.SetNormal(PNorthIndex0,PNorthIndex1,PNorthIndex2);构造球体四边形面片代码: for(int i=1;iN1-1;i+)for(int j=0;jN2;j+)int tempi=i+1;int tempj=j+1;if(tempj=N2) tempj=0;int BodyIndex4;BodyIndex0=(i-1)*N2+j+1;BodyIndex1=(tempi-1)*N2+j+1;BodyIndex2=(tempi-1)*N2+tempj+1;BodyIndex3=(i-1)*N2+tempj+1;Fij.SetEN(4);for(int k=0;kFij.En;k+)Fij.pk=BodyIndexk;Fij.SetNormal(PBodyIndex0,PBodyIndex1,PBodyIndex2); (3)、绘制球体函数 DrawObject (CDC *pDC) 使用Z-Buffer 算法对球面进行深度消隐,然后使用有效边表算法进行填充,为减少渲染的面片数,先使用凸多面体消隐算法对球体不可见面片进行剔除。然后使用Z-Buffer 算法对可见面进行消隐,最后使用有效边表算法进行填充。在函数中,使用Phong双线性法矢量插值模型,计算每个面片上的矢法量。定义了Normal3数组存储三角形面片的顶点法矢量,定义了Normal4数组存储四边形面片的顶点法矢量。 (4)、使用键盘方向键OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)消息响应窗口函数,代码如下: if(!Play)switch(nChar)case VK_UP: 使用 Phi 每次自减5 ,按向上键使球体向里旋转 Phi=Phi-5; break;case VK_DOWN: 使用 Phi 每次自加5 ,按向下键使球体向外旋转Phi=Phi+5; break; case VK_LEFT: 使用 Theta 每次自加5 ,按向左键使球体向左旋转Theta=Theta+5; break;case VK_RIGHT: 使用 Theta 每次自减5 ,按向左键使球体向右旋转 Theta=Theta-5;break;default: break; InitParameter();DoubleBuf(); (5)、使用鼠标左右点击窗口响应函数,对球体进行放大和缩小 void CMyView:OnLButtonDblClk(UINT nFlags, CPoint point) R=R+30; 在鼠标左击函数中视点半径每次增加30,使球体缩小DoubleBuf();CView:OnLButtonDblClk(nFlags, point);void CMyView:OnRButtonDblClk(UINT nFlags, CPoint point) R=R-30; 在鼠标右击函数中视点半径每次减少30,使球体增大DoubleBuf();CView:OnRButtonDblClk(nFlags, point); 五、程序运行结果 (1)当球体半径r=150,面片夹角=10,纬度区间N1=18,经度区间N2=36时,球体共有(N1-1)*N2+2=616个顶点,其运行结果如图:(2)使用键盘方向向左旋转球体,Theta=Thet
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 目标设定中的常见误区考核试卷
- 学校文化氛围对学生职业适应能力的培养考核试卷
- 园林工具制造中的复合材料应用研究考核试卷
- 风味稳定性考核试卷
- 光纤通信网络传输速率标准考核试卷
- 急救知识试题库+参考答案
- 机器人科技探索未来主题班会课件
- 低碳生活活动总结15篇
- 每周双讲活动方案
- 民俗开学仪式活动方案
- 抄表业务课件
- GB/T 45700-2025物业管理术语
- 2025至2030年中国酮洛芬行业市场发展调研及投资方向分析报告
- 学校固定资产管理培训
- (高清版)DG∕TJ 08-55-2019 城市居住地区和居住区公共服务设施设置标准
- 警校面试考试试题及答案
- 高速公路救援试题及答案
- 2025年军事理论与国防教育课程考核试卷及答案
- 国家开放大学药学考试试题及答案
- SJG 130 – 2023《混凝土模块化建筑技术规程》
- 2025厌氧好氧缺氧(AOA)活性污泥法设计标准
评论
0/150
提交评论