热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[转载]从零开始学习OpenGLES之四–光效

继续我们的iPhoneOpenGLES之旅,我们将讨论光效。目前,我们没有加入任何光效。幸运的是,OpenGL在没有设置光效的情况下仍然可

继续我们的iPhone OpenGL ES之旅,我们将讨论光效。目前,我们没有加入任何光效。幸运的是,OpenGL在没有设置光效的情况下仍然可以看见东西。 它只是提供一种十分单调的整体光让我们看到物体。但是如果不定义光效,物体看上去都很单调,就像你在第二部分程序中看到的那样。

 

 

 

阴影模型(Shade Model)

 

在深入讨论OpenGL ES是怎样处理光线之前,重要的是要了解OpenGL ES实际上定义了两种shade model, GL_FLAT 和 GL_SMOOTH。我们将不会讨论GL_FLAT,因为这只会让你的程序看上去来自九十年代:


GL_FLAT 方式渲染的一个二十面体。15年前的实时渲染技术

 

从发光的角度来看,GL_FLAT将指定三角形上的每个像素都 同等对待。多边形上的每个像素都具有相同的颜色,阴影等。它提供了足够的视觉暗示使其看上去有立体感而且它的计算比每个像素按不同方法计算更为廉价,但是 在这种方式下,物体看上去极为不真实。现在有人使用它可能是为了产生特殊的复古效果,但要使你的3D物体尽量真实,你应该使用 GL_SMOOTH 绘图模式,它使用了一种平滑但较快速的阴影算法,称为Gouraud 算法。 GL_SMOOTH是默认值。

 

启动光效

 

我假定你继续使用第二部分的最终项目,即那个看上去不是很立体的旋转二十面体的项目。如果你手头上还没有那个项目,在这里下载。

 

第一件事就是要启动光效。默认情况下,手工指定光效是被禁止的。现在我们打开这项功能。在GLViewController.msetupView:方法中加入黑体部分:

-(void)setupView:(GLView*)view
{const GLfloat zNear = 0.01, zFar = 1000.0, fieldOfView = 45.0;GLfloat size;
glEnable(GL_DEPTH_TEST);
glMatrixMode(GL_PROJECTION);size = zNear *tanf(DEGREES_TO_RADIANS(fieldOfView) / 2.0);CGRect rect = view.bounds;
glFrustumf(-size, size, -size / (rect.size.width / rect.size.height), size /(rect.size.width / rect.size.height), zNear, zFar);
glViewport(0, 0, rect.size.width, rect.size.height);
glMatrixMode(GL_MODELVIEW);glEnable(GL_LIGHTING);glLoadIdentity();
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
}

 

通常情况下,光效只需在设定时启动一次。不需要在绘图开始前后打开和关闭。可能有些特效的情况需要在程序执行时打开或关闭,但是大部分情况下,你只需在程序启动时打开它。此单行代码就是在OpenGL ES中启动光效。运行时会怎样?

 

启动光效

 

我们启动了光效,但是没有创建任何光源。除清除缓存用的灰色外任何绘制的物体都被渲染成绝对的黑色。没有太多的改进,对吗?让我们在场景中加入光源。

 

启动光效的方式有些奇怪。OpenGL ES允许你创建8个光源。有一个常量对应于这些光源中的一个,常量为GL_LIGHT0 到 GL_LIGHT7。可以任意组合这些光源中的五个,尽管习惯上从 GL_LIGHT0 作为第一个光源,然后是 GL_LIGHT1 等等。下面是“打开”第一个光源GL_LIGHT0的方法:

glEnable(GL_LIGHT0);

 

一旦你启动了光源,你必须设置光源的一些属性。作为初学者,有三个不同的要素用来定义光源。

 

光效三要素

 

在 OpenGL ES中,光由三个元素组成,分别是环境元素(ambient component), 散射元素(diffuse component)和 高光元素(specular component)。我们使用颜色来设定光线元素,这看上去有些奇怪,但是由于它允许你同时指定各光线元素的颜色和相对强度,这个方法工作得很好。明亮的白色光定义为白色 ({1.0, 1.0, 1.0, 1.0}),而暗白色可能定义为灰色 ({0.3, 0.3, 0.3 1.0})。 你还可以通过改变红,绿,蓝元素的百分比来调整色偏。

 

下图说明了各要素产生的效果。

 

高光元素定义了光线直接照射并反射到观察者从而形成了物体上的“热点”或光泽。光点的大小取决于一些因素,但是如果你看到如上图黄球所示一个区域明显的光斑,那通常就是来自于一个或多个光源的高光部分。

 

散射元素定义了比较平均的定向光源,在物体面向光线的一面具有光泽。

 

环境光则没有明显的光源。其光线折射与许多物体,因此无法确定其来源。环境元素平均作用于场景中的所有物体的所有面。

 

环境光

 

你 的光效中有越多的环境元素,那么就越不会产生引入注目的效果。所有光线的环境元素会融合在一起产生效果,意思是场景中的总环境光效是由所有启动光源的环境 光组合在一起所决定的。如果你使用了不止一个光源,那么最好是只指定一个光源的环境元素,而设定其他所有光源的环境因素为黑 ({0.0, 0.0, 0.0, 1.0}),从而很容易地调整场景的环境光效。

 

下面演示了怎样指定一个很暗的白色光源:

const GLfloat light0Ambient[] = {0.05, 0.05, 0.05, 1.0};
glLightfv(GL_LIGHT0, GL_AMBIENT, light0Ambient);

 

使用像这样的很低的环境元素值使场景看上去更引入注目,但同时也意味着物体没有面向光线的面或者有其他物体挡住的物体将在场景中看得不是很清楚。

 

散射光

 

在OpenGL ES中可以设定的第二个光线元素是 散射元素(diffuse component)。在现实世界里,散射光线是诸如穿透光纤或从一堵白墙反射的光线。散射光线是发散的,因而参数较柔和的光,一般不会像直射光一样产生光斑。如果你曾经观察过职业摄影家使用摄影室灯光,你可能会看到他们使用柔光箱 或 者反光伞。两者都会穿透像白布之类的轻型材料并反射与轻型有色材料从而使光线发散以产生令人愉悦的照片。在OpenGL ES中,散射元素作用类似,它使光线均匀地散布到物体之上。然而,不像环境光,由于它是定向光,只有面向光线的物体面才会反射散射光,而场景中的所有多面 体都会被环境光照射。

 

下面的例子演示了设定场景中的第一个散射元素:

const GLfloat light0Diffuse[] = {0.5, 0.5, 0.5, 1.0};
glLightfv(GL_LIGHT0, GL_DIFFUSE, light0Diffuse);

 

高光

 

最后,我们讨论高光。这种类型的光是十分直接的,它们会以热点和光晕的形式反射到观察者的眼中。如果你想产生聚光灯的效果,那么应该设置一个很大的高光元素值及很小的散射和环境元素值(还需要定义其他一些参数,等下会有介绍)。

注意: 在下一篇文章中你将看到,光线的高光值是确定高光尺寸的唯一因素。

 

下面是设定高光元素的例子:

const GLfloat light0Specular[] = {0.7, 0.7, 0.7, 1.0};

 

位置

 

还需要设定光效的另一个重要属性,即光源3D空间中的位置。这不会影响环境元素,但其他两个元素由于其本性,只有在OpenGL在知道了场景中物体与光的相对位置后才能计算。例如:

const GLfloat light0Position[] = {10.0, 10.0, 10.0, 0.0};
glLightfv(GL_LIGHT0, GL_POSITION, light0Position);

 

此位置将第一个光源放置在观察者后方的右上角。

 

这些是用于设定几乎所有光线的属性。如果你没有设定其中一个元素,那么它就采用默认值黑色 {0.0, 0.0, 0.0, 1.0}。如果你没有定义位置,那么它就处于原点,通常这不是你想要的结果。

 

你 可能想知道alpha值对光线的作用。对环境光和高光而言这是个愚蠢的问题。然而在计算散射光确定光线是怎样反射时,需要用到它。我们将在讨论材质时再解 释它是怎样工作的,因为材质和光线值都将出现在方程式中。我们下次再讨论材质,现在将 alpha 设为1.0。改变其值对本文的程序不会产生任何影响,但有可能对以后的程序至少是有关散射元素的部分产生影响。

 

还有一些光线元素你可选择使用。

 

创建点光源(聚光灯)

 

如果你希望创建一个定向点光源 (一种指向特定方向并照亮特定角度范围的光源,本质上,它与灯泡照亮各个方向相反,它只照亮一个锥台的范围),那么你需要设定两个额外参数。设定 GL_SPOT_DIRECTION 允许你指定光照的方向,它类似于上一篇文章中介绍的视野角度的计算。窄角度将产生很小范围的点光源,而宽角度则产生像泛光灯一样的效果。

 

指定光的方向

通过指定定义了光线指向的x,y和z值来使 GL_SPOT_DIRECTION的工作。然而,光线并不指向你在空间中定义的那一点。你提供的三个坐标值是向量(vector), 而非顶点。这是一个很细微但十分重要的区别。一个代表向量的数据结构与一个顶点的数据结构完全一样(都有三个 GLfloats,其中每一个分别是笛卡尔的一个轴)。然而,向量的数据是用来表示方向而不是空间中的一点。
每个人都知道两点可以定义一条线段, 那么空间中的一点怎么可能指定方向?这是因为存在一个隐性的第二点作为起点,即原点。如果你从原点画一条线到向量定义的点,那就是向量代表的方向。向量还 可用于表示速度和距离,一个远离原点的点表示速度越快或距离更远。在大部分的OpenGL应用中,并未使用距离原点的距离。实际上,在大部分使用向量的情 况下,我们需要将向量标准化(normalize) 为长度 1.0。关于向量的标准化,随着我们继续深入将会讨论到。现在你只需知道,如果你希望定义一个定向光,那么你 必须创建一个定义了光的方向的向量。下例演示了定义一个沿z轴而下的光源:

const GLfloat light0Direction = {0.0, 0.0, -1.0};
glLightfv(GL_LIGHT0, GL_SPOT_DIRECTION, light0Direction);

 

现在,如果你希望光线指向一个特定物体,应该怎么办?实际上很简单,将光的位置和物体的位置传入OpenGLCommon.h的函数 Vector3DMakeWithStartAndEndPoints()中,它将返回一个被光照射到的指定点的标准化向量。然后,再将其作为 GL_SPOT_DIRECTION 的值。

 

指定光的角度

 

除非你限制光照的角度,否则指定光的方向并不会产生显著的效果。因为当你指定 GL_SPOT_CUTOFF 值时,它定义了中心线两边的角度,所以如果你指定截止角时,它必须小于180°。如果你的定义为45°,那么实际上你创建了一个总角度为90°的点光源。这意味着你可设定的 GL_SPOT_CUTOFF 的最大值为 180°。下图说明了这个概念:

下例演示了怎样限制角度为 90°(使用 45° 截止角):

glLightf(GL_LIGHT0, GL_SPOT_CUTOFF, 45.0);

 

还有三个光的属性可以设置。它们是配合使用的,这些超出了本文的范围。以后,我可能在有关光衰减 (光线随着远离光源而减弱)的文章中讨论到。通过调整衰减值可以产生很漂亮的效果。

 

综合

 

让我们综合所学内容在setupView: 方法中设定一个光源。使用下列代码替换 setupView: 方法中原有的代码:

-(void)setupView:(GLView*)view
{const GLfloat zNear = 0.01, zFar = 1000.0, fieldOfView = 45.0;GLfloat size;
glEnable(GL_DEPTH_TEST);
glMatrixMode(GL_PROJECTION);size = zNear *tanf(DEGREES_TO_RADIANS(fieldOfView) / 2.0);CGRect rect = view.bounds;
glFrustumf(-size, size, -size / (rect.size.width / rect.size.height), size /(rect.size.width / rect.size.height), zNear, zFar);
glViewport(0, 0, rect.size.width, rect.size.height);
glMatrixMode(GL_MODELVIEW);// Enable lighting
glEnable(GL_LIGHTING);// Turn the first light on
glEnable(GL_LIGHT0);// Define the ambient component of the first lightconst GLfloat light0Ambient[] = {0.1, 0.1, 0.1, 1.0};
glLightfv(GL_LIGHT0, GL_AMBIENT, light0Ambient);// Define the diffuse component of the first lightconst GLfloat light0Diffuse[] = {0.7, 0.7, 0.7, 1.0};
glLightfv(GL_LIGHT0, GL_DIFFUSE, light0Diffuse);// Define the specular component and shininess of the first lightconst GLfloat light0Specular[] = {0.7, 0.7, 0.7, 1.0};const GLfloat light0Shininess = 0.4;
glLightfv(GL_LIGHT0, GL_SPECULAR, light0Specular);// Define the position of the first lightconst GLfloat light0Position[] = {0.0, 10.0, 10.0, 0.0};
glLightfv(GL_LIGHT0, GL_POSITION, light0Position); // Define a direction vector for the light, this one points right down the Z axisconst GLfloat light0Direction[] = {0.0, 0.0, -1.0};
glLightfv(GL_LIGHT0, GL_SPOT_DIRECTION, light0Direction);// Define a cutoff angle. This defines a 90° field of vision, since the cutoff// is number of degrees to each side of an imaginary line drawn from the light's// position along the vector supplied in GL_SPOT_DIRECTION above
glLightf(GL_LIGHT0, GL_SPOT_CUTOFF, 45.0);glLoadIdentity();
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
}

 

很简单吧?应该一切都准备好了吧?好,我们试着运行一下看看。

 

这是什么?太糟糕了吧!我们设定了光源,可是看不到任何东西啊?屏幕上只是显示一个黑和灰的形状,它甚至看上去不像个3D形状,比以前还糟糕。

 

不要紧张,这很正常(注: It’s Normal在这里有两层意思,一是正常,二是法线)

 

Normal数学上的意思是“垂直于”。这是我们目前缺少的。法线。一个背面的法线(或多边形的法线)是一个垂直于指定多边形表面的向量(或直线)。参考下图:

 

OpenGL 渲染一个形状时并不需要知道法线,但在你使用定向光线时需要用到。OpenGL需要表面法线来确定光线是怎样与各多边形交互作用的。

 

OpenGL 要求我们为各使用的顶点提供法线。计算一个三角形的表面法线是很简单的,它是三角形两边的叉积。代码如下:

static inline Vector3DTriangle3DCalculateSurfaceNormal(Triangle3D triangle)
{Vector3D u =Vector3DMakeWithStartAndEndPoints(triangle.v2, triangle.v1);Vector3D v =Vector3DMakeWithStartAndEndPoints(triangle.v3, triangle.v1);Vector3D ret;ret.x = (u.y * v.z) - (u.z * v.y);ret.y = (u.z * v.x) - (u.x * v.z);ret.z = (u.x * v.y) - (u.y * v.x);return ret;
}

 

Vector3DMakeWithStartAndEndPoints()取两顶点值计算其标准化向量。那么既然计算表面法线如此简单,为什么OpenGL ES不为我们完成?有两个原因,第一个和最重要的原因是,开销太大。对每个多边形而言,有许多浮点乘除计算以及调用sqrtf()的开销。

 

第二,因为我们使用 GL_SMOOTH 渲染,所以OpenGL ES需要知道顶点法线(vertex normal) 而不是表面法线(上述计算)。因为顶点法线要求你计算使用了该顶点的所有表面法线的平均向量,所以开销更大。

 

让我们看一个例子。

 

请 注意这不是一个正方体。简单起见,让我们看看一个平面的六个三角形构成的两维形状。它总共由七个顶点构成。顶点A由所有六个三角形共享,所以此顶点的顶点 法线是所有七个三角形(注:六个吧?)的表面法线的平均值。平均值的计算是基于各向量元素的,即x值被平均,y值被平均,然后z值被平均,结果组合在一起 构成了平均向量。

 

所 以,我们怎样计算二十面体的向量?这其实是一个很简单的形状,在计算顶点法线时并不会造成显著的延时。通常,你不会工作于这么少顶点的物体,而将处理复杂 得多而且数量更多的物体。结果是,除非没有替代方法,否则你希望避免使用顶点法线的实时计算。这种情况下,我编写了一个小命令行程序,它循环处理每个顶点 及三角形索引,来计算二十面体的各顶点法线。该程序将结果以C struct的形式输出到控制台,然后我复制到我的OpenGL程序中。

 

注意:大部分3D程序都会为你提供法线的计算,但你需要小心使用 – 大部分3D文件格式存储的是表面法线而不是顶点法线,所以你至少需要计算表面法线的平均值来生成顶点法线。在以后的文章中我将介绍加载和创建3D物体,或参阅有关加载Wavefront OBJ 文件格式的文章。

下面是我写的计算二十面体顶点法线的命令行程序:

#import
#import "OpenGLCommon.h"intmain (int argc, const char * argv[]) {NSAutoreleasePool * pool &#61; [[NSAutoreleasePool alloc] init];NSMutableString *result &#61; [NSMutableString string];static const Vertex3D vertices[]&#61; {{0, -0.525731, 0.850651}, // vertices[0]{0.850651, 0, 0.525731}, // vertices[1]{0.850651, 0, -0.525731}, // vertices[2]{-0.850651, 0, -0.525731}, // vertices[3]{-0.850651, 0, 0.525731}, // vertices[4]{-0.525731, 0.850651, 0}, // vertices[5]{0.525731, 0.850651, 0}, // vertices[6]{0.525731, -0.850651, 0}, // vertices[7]{-0.525731, -0.850651, 0}, // vertices[8]{0, -0.525731, -0.850651}, // vertices[9]{0, 0.525731, -0.850651}, // vertices[10]{0, 0.525731, 0.850651} // vertices[11]};static const GLubyte icosahedronFaces[] &#61; {1, 2, 6,1, 7, 2,3, 4, 5,4, 3, 8,6, 5, 11,5, 6, 10,9, 10, 2,10, 9, 3,7, 8, 9,8, 7, 0,11, 0, 1,0, 11, 4,6, 2, 10,1, 6, 11,3, 5, 10,5, 4, 11,2, 7, 9,7, 1, 0,3, 9, 8,4, 8, 0,};Vector3D *surfaceNormals &#61;calloc(20,sizeof(Vector3D));// Calculate the surface normal for each trianglefor (int i &#61; 0; i <20; i&#43;&#43;){Vertex3D vertex1 &#61; vertices[icosahedronFaces[(i*3)]];Vertex3D vertex2 &#61; vertices[icosahedronFaces[(i*3)&#43;1]];Vertex3D vertex3 &#61; vertices[icosahedronFaces[(i*3)&#43;2]];Triangle3D triangle &#61;Triangle3DMake(vertex1, vertex2, vertex3);Vector3D surfaceNormal &#61;Triangle3DCalculateSurfaceNormal(triangle);
Vector3DNormalize(&surfaceNormal);surfaceNormals[i] &#61; surfaceNormal;}Vertex3D *normals &#61;calloc(12,sizeof(Vertex3D));[result appendString:&#64;"static const Vector3D normals[] &#61; {\n"];for (int i &#61; 0; i <12; i&#43;&#43;){int faceCount &#61; 0;for (int j &#61; 0; j <20; j&#43;&#43;){BOOL contains &#61; NO;for (int k &#61; 0; k <3; k&#43;&#43;){ if (icosahedronFaces[(j * 3) &#43; k] &#61;&#61; i) contains &#61; YES; }if (contains){ faceCount&#43;&#43;; normals[i] &#61;Vector3DAdd(normals[i], surfaceNormals[j]); }}normals[i].x /&#61; (GLfloat)faceCount;normals[i].y /&#61; (GLfloat)faceCount;normals[i].z /&#61; (GLfloat)faceCount;[result appendFormat:&#64;"\t{%f, %f, %f},\n", normals[i].x, normals[i].y, normals[i].z];}[result appendString:&#64;"};\n"];
NSLog(result);[pool drain];return 0;
}

 

可能有点粗糙&#xff0c;但它很好的完成了工作&#xff0c;允许我们预先计算顶点法线&#xff0c;所以在运行时不需要进行计算。程序输出如下&#xff1a;

static const Vector3D normals[] &#61; {{0.000000, -0.417775, 0.675974},{0.675973, 0.000000, 0.417775},{0.675973, -0.000000, -0.417775},{-0.675973, 0.000000, -0.417775},{-0.675973, -0.000000, 0.417775},{-0.417775, 0.675974, 0.000000},{0.417775, 0.675973, -0.000000},{0.417775, -0.675974, 0.000000},{-0.417775, -0.675974, 0.000000},{0.000000, -0.417775, -0.675973},{0.000000, 0.417775, -0.675974},{0.000000, 0.417775, 0.675973},
};

 

指定顶点法线

 

首先我们要启动法线数组&#xff1a;

glEnableClientState(GL_NORMAL_ARRAY);

 

使用下列调用传递法线数组&#xff1a;

glNormalPointer(GL_FLOAT, 0, normals);

 

将所有这些加到 drawSelf: 方法中&#xff1a;

- (void)drawView:(GLView*)view;
{static GLfloat rot &#61; 0.0;// This is the same result as using Vertex3D, just faster to type and// can be made const this waystatic const Vertex3D vertices[]&#61; {{0, -0.525731, 0.850651}, // vertices[0]{0.850651, 0, 0.525731}, // vertices[1]{0.850651, 0, -0.525731}, // vertices[2]{-0.850651, 0, -0.525731}, // vertices[3]{-0.850651, 0, 0.525731}, // vertices[4]{-0.525731, 0.850651, 0}, // vertices[5]{0.525731, 0.850651, 0}, // vertices[6]{0.525731, -0.850651, 0}, // vertices[7]{-0.525731, -0.850651, 0}, // vertices[8]{0, -0.525731, -0.850651}, // vertices[9]{0, 0.525731, -0.850651}, // vertices[10]{0, 0.525731, 0.850651} // vertices[11]};static const Color3D colors[] &#61; {{1.0, 0.0, 0.0, 1.0},{1.0, 0.5, 0.0, 1.0},{1.0, 1.0, 0.0, 1.0},{0.5, 1.0, 0.0, 1.0},{0.0, 1.0, 0.0, 1.0},{0.0, 1.0, 0.5, 1.0},{0.0, 1.0, 1.0, 1.0},{0.0, 0.5, 1.0, 1.0},{0.0, 0.0, 1.0, 1.0},{0.5, 0.0, 1.0, 1.0},{1.0, 0.0, 1.0, 1.0},{1.0, 0.0, 0.5, 1.0}};static const GLubyte icosahedronFaces[] &#61; {1, 2, 6,1, 7, 2,3, 4, 5,4, 3, 8,6, 5, 11,5, 6, 10,9, 10, 2,10, 9, 3,7, 8, 9,8, 7, 0,11, 0, 1,0, 11, 4,6, 2, 10,1, 6, 11,3, 5, 10,5, 4, 11,2, 7, 9,7, 1, 0,3, 9, 8,4, 8, 0,};static const Vector3D normals[] &#61; {{0.000000, -0.417775, 0.675974},{0.675973, 0.000000, 0.417775},{0.675973, -0.000000, -0.417775},{-0.675973, 0.000000, -0.417775},{-0.675973, -0.000000, 0.417775},{-0.417775, 0.675974, 0.000000},{0.417775, 0.675973, -0.000000},{0.417775, -0.675974, 0.000000},{-0.417775, -0.675974, 0.000000},{0.000000, -0.417775, -0.675973},{0.000000, 0.417775, -0.675974},{0.000000, 0.417775, 0.675973},};glLoadIdentity();
glTranslatef(0.0f,0.0f,-3.0f);
glRotatef(rot,1.0f,1.0f,1.0f);
glClearColor(0.7, 0.7, 0.7, 1.0);
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_COLOR_ARRAY);
glEnableClientState(GL_NORMAL_ARRAY);
glVertexPointer(3, GL_FLOAT, 0, vertices);
glColorPointer(4, GL_FLOAT, 0, colors);
glNormalPointer(GL_FLOAT, 0, normals);
glDrawElements(GL_TRIANGLES, 60, GL_UNSIGNED_BYTE, icosahedronFaces);glDisableClientState(GL_VERTEX_ARRAY);
glDisableClientState(GL_COLOR_ARRAY);
glDisableClientState(GL_NORMAL_ARRAY);static NSTimeInterval lastDrawTime;if (lastDrawTime){NSTimeInterval timeSinceLastDraw &#61; [NSDate timeIntervalSinceReferenceDate] - lastDrawTime;rot&#43;&#61;50 * timeSinceLastDraw;}lastDrawTime &#61; [NSDate timeIntervalSinceReferenceDate];
}

基本完工

运行&#xff0c;你将看到一个真实的三维旋转物体。

但是颜色呢&#xff1f;

请听下回分解&#xff1a;OpenGL ES 材质。 当你使用光效和平滑阴影时&#xff0c;OpenGL期待你为多边形提供材质&#xff08;material&#xff09; (或纹理 &#xff08;texture&#xff09;&#xff09;。材质比在颜色数组中提供简单颜色要复杂得多。材质像光一样由许多元素构成&#xff0c;可以产生不同的表面效果。物体的表面效果实际上是由场景 中的光和多边形的材质决定的。

但是&#xff0c;我们不希望显示一个灰暗的二十面体。所以我介绍另一个OpenGL ES的配置参数: GL_COLOR_MATERIAL。启动它&#xff1a;
glEnable(GL_COLOR_MATERIAL);
OpenGL 将使用我们提供的颜色数组来创建简单的材质&#xff0c;结果如下&#xff1a;

如果你不想输入所有代码&#xff0c;可下载源代码。

转:https://www.cnblogs.com/zhoujq/archive/2013/03/15/2961813.html



推荐阅读
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 毕业设计:基于机器学习与深度学习的垃圾邮件(短信)分类算法实现
    本文详细介绍了如何使用机器学习和深度学习技术对垃圾邮件和短信进行分类。内容涵盖从数据集介绍、预处理、特征提取到模型训练与评估的完整流程,并提供了具体的代码示例和实验结果。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 本文探讨了 Objective-C 中的一些重要语法特性,包括 goto 语句、块(block)的使用、访问修饰符以及属性管理等。通过实例代码和详细解释,帮助开发者更好地理解和应用这些特性。 ... [详细]
  • 文件描述符、文件句柄与打开文件之间的关联解析
    本文详细探讨了文件描述符、文件句柄和打开文件之间的关系,通过具体示例解释了它们在操作系统中的作用及其相互影响。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文深入探讨了Linux系统中网卡绑定(bonding)的七种工作模式。网卡绑定技术通过将多个物理网卡组合成一个逻辑网卡,实现网络冗余、带宽聚合和负载均衡,在生产环境中广泛应用。文章详细介绍了每种模式的特点、适用场景及配置方法。 ... [详细]
  • 本文详细介绍了 Apache Jena 库中的 Txn.executeWrite 方法,通过多个实际代码示例展示了其在不同场景下的应用,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 本文详细探讨了VxWorks操作系统中双向链表和环形缓冲区的实现原理及使用方法,通过具体示例代码加深理解。 ... [详细]
  • MySQL索引详解与优化
    本文深入探讨了MySQL中的索引机制,包括索引的基本概念、优势与劣势、分类及其实现原理,并详细介绍了索引的使用场景和优化技巧。通过具体示例,帮助读者更好地理解和应用索引以提升数据库性能。 ... [详细]
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • 在多线程编程环境中,线程之间共享全局变量可能导致数据竞争和不一致性。为了解决这一问题,Linux提供了线程局部存储(TLS),使每个线程可以拥有独立的变量副本,确保线程间的数据隔离与安全。 ... [详细]
author-avatar
雨蝶馨菲_484
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有