《基于MFC的OpenGL编程》Part 17 Shadows
Shadows
Conceptually drawing a shadow is quite simple. A shadow is produced when an object keeps light from a source from striking some object or surface behind the object, casting the shadow. The area on the shadowed object's surface outlined by the object casting the shadow appears dark. We can produce a shadow programatically by flattening the original object into the plane of the surface in which the object lies. The object is then drawn in black or some other color. This is a very simple method which works when casting shadows on a flat surface.
从这一篇开始将用到一个第三方库PixieLib,本文先对这个库的使用做简单介绍,后续文章中将不再赘述。
Paul DiLascia是兼职软件顾问和资深的 Web/UI 设计师。他是《Windows++: Writing Reusable Windows Code in C++》(Windows++:在 C++ 中编写可重用 Windows 代码)一书(Addison-Wesley,1992)的作者。Paul 在其业余时间里开发出了 PixieLib,可通过他的网站 www.dilascia.com 来获取该 MFC 类库。
将下载下来的源代码编译后会生成Lib目录,我们需要用到的就是Include和Lib这两个目录,当然你也可以看看它自带的Samples。由于使用的是最新PixieLib7.1版,因此接下来几篇文章都在VS2005下进行开发。
1,将Lib和Include目录拷贝到新建的MFC 项目目录下,配置如下:
1)c/c++à附加包含目录,这里加入”."include”。
2)链接器à附加库目录,这里加入”."lib”
3)输入à附加依赖项,这里加入PixieLib71.lib
2,在stdafx.h中包含进PixieLib库文件:
3,在CCY457OpenGLView类中加入下述变量:
//Position and Direction
GLfloat m_PosX;
GLfloat m_PosY;
GLfloat m_PosZ;
GLfloat m_DirX;
GLfloat m_DirY;
GLfloat m_DirZ;
//Rotation
GLfloat m_xRot, m_yRot;//绕x,y轴旋转的角度,随时间不断变化
//Increment for Keyboard Control
GLfloat m_PosIncr; //Positional Increment
GLfloat m_AngIncr; //Angular Increment
//Angle of Camera With X Axis
GLfloat m_AngleX;
GLdouble m_texWrap, m_texFilter, m_texMode;
GLfloat m_ShadowMat[4][4];
//All Texture Names
GLuint m_Texture[4];
并在构造函数中初始化如下:
{
//Rotation
m_xRot = 0.0f;
m_yRot = 0.0f;
//Position Increment
m_PosIncr = 0.25f;
//Angle Increment
m_AngIncr = 5.0f;
//Set Initial Camera Position - looking down negative Z
m_PosX = 0.0f;
m_PosY = 0.5f;
m_PosZ = 2.5f;
//Set Initial Viewing Direction - Pointing Down the -ve Z Axis
m_DirX = m_PosX;
m_DirY = m_PosY;
m_DirZ = m_PosZ-m_PosIncr;
//Angle of Camera with X Axis
m_AngleX = 90.0f;
m_texWrap = GL_CLAMP;
m_texMode = GL_DECAL;
m_texFilter = GL_NEAREST;
}
4,InitializeOpenGL函数修改如下:
{
//Get a DC for the Client Area
m_pDC = new CClientDC(this);
//Failure to Get DC
if(m_pDC == NULL)
{
MessageBox("Error Obtaining DC");
return FALSE;
}
//Failure to set the pixel format
if(!SetupPixelFormat())
{
return FALSE;
}
//Create Rendering Context
m_hRC = ::wglCreateContext (m_pDC->GetSafeHdc ());
//Failure to Create Rendering Context
if(m_hRC == 0)
{
MessageBox("Error Creating RC");
return FALSE;
}
//Make the RC Current
if(::wglMakeCurrent (m_pDC->GetSafeHdc (), m_hRC)==FALSE)
{
MessageBox("Error making RC Current");
return FALSE;
}
// specify black as clear color
::glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
// specify the back of the buffer as clear depth
::glClearDepth(1.0f);
// enable depth testing
::glEnable(GL_DEPTH_TEST);
//Enable Color Tracking
::glEnable(GL_COLOR_MATERIAL);
::glColorMaterial(GL_FRONT_AND_BACK,GL_AMBIENT_AND_DIFFUSE);
::glShadeModel(GL_SMOOTH);
//Setup Lighting Here
SetupLighting();
LoadGLTextures();
//Effects
//Shadow
//Calulate Shadow Matrix
GLfloat lightPos[] = {1.0f,1.5f,-1.0f,0.0f};
GLfloat points[][3] = {
{-2.0f,0.0f,0.0f},
{2.0f,0.0f,0.0f},
{2.0f,0.0f,-2.0f}
};
MakeShadowMatrix(points, lightPos, m_ShadowMat);
return TRUE;
}
5,设置灯光的代码修改如下:
{
//Enable Lighting
glEnable(GL_LIGHTING);
//Set up the Light Model
//Infinite Viewer
glLightModeli(GL_LIGHT_MODEL_LOCAL_VIEWER, GL_FALSE);
//Single Sided Lighting
glLightModeli(GL_LIGHT_MODEL_TWO_SIDE, GL_FALSE);
//Set up the lights
//Light 1
//Directional Light from Front
GLfloat m_SceneAmbient1[] = {0.5f,0.5f,0.5f,1.0f};
GLfloat m_SceneDiffuse1[] = {1.0f,1.0f,1.0f,1.0f};
GLfloat m_SceneSpecular1[] = {1.0f,1.0f,1.0f,1.0f};
GLfloat m_ScenePosition1[] = {1.0f,1.5f,-1.0f,1.0f};
GLfloat m_SceneDirection1[]= {0.0f,0.0f,-1.0f,1.0f};
glLightfv(GL_LIGHT0,GL_AMBIENT,m_SceneAmbient1);
glLightfv(GL_LIGHT0,GL_DIFFUSE,m_SceneDiffuse1);
glLightfv(GL_LIGHT0,GL_SPECULAR,m_SceneSpecular1);
glLightfv(GL_LIGHT0,GL_POSITION,m_ScenePosition1);
glLightf(GL_LIGHT0,GL_SPOT_CUTOFF,75.0f);
glLightfv(GL_LIGHT0,GL_SPOT_DIRECTION,m_SceneDirection1);
glEnable(GL_LIGHT0);
}
6,绘制代码修改如下:
{//绘制函数
//Position Camera
gluLookAt(m_PosX,m_PosY,m_PosZ,m_DirX,m_DirY,m_DirZ,0.0f,1.0f,0.0f);
//Draw the Scene
//Draw the floor
// Draw the ground,
glEnable(GL_TEXTURE_2D);
glBindTexture(GL_TEXTURE_2D, m_Texture[3]);
glBegin(GL_POLYGON);
glColor3ub(0,255,0);
glTexCoord2f(0.0f, 0.0f);
glVertex3f(-2.0f, 0.0f, 0.0f);
glTexCoord2f(1.0f, 0.0f);
glVertex3f(2.0f,0.0f, 0.0f);
glColor3ub(0,100,0);
glTexCoord2f(1.0f, 1.0f);
glVertex3f(2.0f, 0.0f, -2.0f);
glTexCoord2f(0.0f, 1.0f);
glVertex3f(-2.0f,0.0f, -2.0f);
glEnd();
glDisable(GL_TEXTURE_2D);
//Draw the Cube
// Save the matrix state and do the rotations
glPushMatrix();
glTranslatef(-1.0f,0.6f,-1.0f);
// Draw jet at new orientation, put light in correct position
// before rotating the jet
glRotatef(m_xRot,1.0f,0.0f,0.0f);
glRotatef(m_yRot,0.0f,1.0f,0.0f);
DrawCube(FALSE);
// Restore original matrix state
glPopMatrix();
// Get ready to draw the shadow and the ground
// First disable lighting and save the projection state
glDisable(GL_DEPTH_TEST);
glDisable(GL_LIGHTING);
glPushMatrix();
// Multiply by shadow projection matrix
glMultMatrixf((GLfloat *)m_ShadowMat);
glTranslatef(-1.0f,0.6f,-1.0f);
glRotatef(m_xRot,1.0f,0.0f,0.0f);
glRotatef(m_yRot,0.0f,1.0f,0.0f);
// Pass true to indicate drawing shadow
DrawCube(TRUE);
// Restore the projection to normal
glPopMatrix();
// Restore lighting state variables
glEnable(GL_DEPTH_TEST);
glEnable(GL_LIGHTING);
// Draw the light source
glPushMatrix();
glTranslatef(1.5f,1.5f,-1.0f);
glColor3ub(255,255,0);
glutSolidSphere(0.01f,10,10);
glPopMatrix();
}
7,用来计算平面法向量,绘制物体,计算阴影矩阵的辅助函数:

作者:洞庭散人
出处:http://phinecos.cnblogs.com/
posted on 2008-11-07 15:33 Phinecos(洞庭散人) 阅读(3930) 评论(12) 编辑 收藏 举报
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· .NET周刊【3月第1期 2025-03-02】
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· [AI/GPT/综述] AI Agent的设计模式综述