OpenGL ES基础教程

一、设置OpenGL ES视图

设置OpenGL视图并不难,Android上也较简单。我们一般只需要2个步骤。

GLSurfaceView

我们要为GLSurfaceView提供一个专门用于渲染的接口

public void  setRenderer(GLSurfaceView.Renderer renderer)

 

GLSurfaceView.Renderer

GLSurfaceView.Renderer是一个通用渲染接口。我们必须实现下面的三个抽象方法:

// 画面创建

public void onSurfaceCreated(GL10 gl, EGLConfig config)

// 画面绘制

public void onDrawFrame(GL10 gl)

// 画面改变

public void onSurfaceChanged(GL10 gl, int width, int height)

onSurfaceCreated

在这里我们主要进行一些初始化工作,比如对透视进行修正、设置清屏所用颜色等。

onDrawFrame

绘制当前画面

onSurfaceChanged

当设备水平或者垂直变化时调用此方法,设置新的显示比例

 

案例代码:

 
  1. public class OpenGLDemo extends Activity {
  2.     @Override
  3.     public void onCreate(Bundle savedInstanceState) { 
  4.         GLSurfaceView view = new GLSurfaceView(this);
  5.         view.setRenderer(new OpenGLRenderer());
  6.         setContentView(view);
  7.     }
  8. }
复制代码实现renderer需要更多的设置
 
  1.     public void onSurfaceCreated(GL10 gl, EGLConfig config) {
  2.         // 黑色背景
  3.         gl.glClearColor(0.0f, 0.0f, 0.0f, 0.5f);
  4.         // 启用阴影平滑(不是必须的)
  5.         gl.glShadeModel(GL10.GL_SMOOTH);
  6.         // 设置深度缓存
  7.         gl.glClearDepthf(1.0f);
  8.         // 启用深度测试
  9.         gl.glEnable(GL10.GL_DEPTH_TEST);
  10.         // 所作深度测试的类型
  11.         gl.glDepthFunc(GL10.GL_LEQUAL);
  12.         // 对透视进行修正
  13.         gl.glHint(GL10.GL_PERSPECTIVE_CORRECTION_HINT, GL10.GL_NICEST);
  14.     }
  15.  
  16.     public void onDrawFrame(GL10 gl) {
  17.         // 清除屏幕和深度缓存
  18.         gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT);
  19.     }
  20.  
  21.     public void onSurfaceChanged(GL10 gl, int width, int height) {
  22.         // 设置画面的大小
  23.         gl.glViewport(0, 0, width, height);
  24.         // 设置投影矩阵
  25.         gl.glMatrixMode(GL10.GL_PROJECTION);
  26.         // 重置投影矩阵
  27.         gl.glLoadIdentity();
  28.         // 设置画面比例
  29.         GLU.gluPerspective(gl, 45.0f, (float) width / (float) height, 0.1f,100.0f);
  30.         // 选择模型观察矩阵
  31.         gl.glMatrixMode(GL10.GL_MODELVIEW);
  32.         // 重置模型观察矩阵
  33.         gl.glLoadIdentity();
  34.     }
  35. }
复制代码 只要加入这段代码到OpenGLDemo class里就可实现全屏this.requestWindowFeature(Window.FEATURE_NO_TITLE);
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
    WindowManager.LayoutParams.FLAG_FULLSCREEN);
设置完视图后,即可编译运行,可以看到一个“漂亮”的黑屏 = =!
 

 

 OpenGLDemo01.rar (48.38 KB, 下载次数: 325)

 

二、绘制多边形前面的教程都是关于设置GLSurfaceView.的,接下来的教程将教我们渲染出一个多边形。3D模型用较小的元素创建(点,边,面),他们可以被分别操作。
顶点

 

    

 

   在Android中,我们通过float数组定义顶点,并将它放到字节型缓冲区内来获取更好的性能。下例的代码即为上图所示顶点。OpenGL ES的很多功能都必须手动的开启和关闭。

  1. gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
  2. // 设置顶点数据,3代表XYZ坐标系
  3. gl.glVertexPointer(3, GL10.GL_FLOAT, 0, vertexBuffer);
  4. // 关闭顶点设置
  5. gl.glDisableClientState(GL10.GL_VERTEX_ARRAY);
复制代码
 

      




计算多边形面的时候,一定要注意正确的方向.。因为这将决定哪一面为正面哪一面为背面。 所以我们尽量保证整个项目都使用相同的环绕。gl.glFrontFace(GL10.GL_CCW);控制多边形的正面是如何决定的。在默认情况下,mode是GL_CCW。mode的值为:   GL_CCW 表示窗口坐标上投影多边形的顶点顺序为逆时针方向的表面为正面。   GL_CW 表示顶点顺序为顺时针方向的表面为正面。顶点的方向又称为环绕。gl.glEnable(GL10.GL_CULL_FACE);gl.glCullFace(GL10.GL_BACK);剔除多边形的背面,禁用多边形背面上的光照、阴影和颜色计算及操作。gl.glDisable(GL10.GL_CULL_FACE);  
多边形


     


到了绘制面的时候了, 我们使用默认的逆时针环绕。下例代码将绘制上图多边形。
  1.     // 将坐标数组放入字节缓存中
  2.     // (1) 分配缓存,一个short为2个字节,所以要乘以2
  3.     ByteBuffer ibb = ByteBuffer.allocateDirect(indices.length * 2);
  4.     // (2) 设置字节处理规则
  5.     ibb.order(ByteOrder.nativeOrder());
  6.     // (3) 转换为short型字符
  7.     ShortBuffer indexBuffer = ibb.asShortBuffer();
  8.     // (4) 放入坐标数组
  9.     indexBuffer.put(indices);
  10.     // (5) 复位
  11.     indexBuffer.position(0);
复制代码 渲染是时候弄些玩意儿到屏幕上去了,绘制时我们将用到两个函数public abstract void glDrawArrays(int mode, int first, int count)通过我们构造的顶点缓存来绘制顶点public abstract void glDrawElements(int mode, int count, int type, Buffer indices)和glDrawArrays类似,但需要直接传入type(索引值的类型,如GL_UNSIGNED_SHORT, or GL_UNSIGNED_INT),和indices(索引缓存)两者的共同点是,都必须知道他们需要画什么。怎样渲染图元,有不同方式,为了帮助调试,我们应该了解它们。
Mode:GL_POINTS绘制独立的点到屏幕  
   
GL_LINE_STRIP连续的连线,第n个顶点与第n-1个顶点绘制一条直线  
   
GL_LINE_LOOP和上面相同,但首尾相连  
   
GL_LINES各对独立的线段  
   
GL_TRIANGLES各个独立的三角形  
 
GL_TRIANGLE_STRIP
绘制一系列的三角形,先是顶点 v0, v1, v2, 然后是 v2, v1, v3 (注意规律), 然后v2, v3, v4等。该规律确保所有的三角形都以相同的方向绘制。 
   
GL_TRIANGLE_FAN和GL_TRIANGLE_STRIP类似, 但其先绘制 v0, v1, v2, 再是 v0, v2, v3, 然后 v0, v3, v4等。   
我认为GL_TRIANGLES是使用最方便的,所以我们将先使用它。
  1. public class Square {
  2.     // 顶点坐标数组
  3.     private float vertices[] = { -1.0f, 1.0f, 0.0f, // 0, 左上
  4.         -1.0f, -1.0f, 0.0f, // 1, 左下
  5.         1.0f, -1.0f, 0.0f, // 2, 右下
  6.         1.0f, 1.0f, 0.0f, // 3, 右上
  7.     };
  8.     // 连接规则
  9.     private short[] indices = { 0, 1, 2, 0, 2, 3 };
  10.     // 顶点缓存
  11.     private FloatBuffer vertexBuffer;
  12.     // 索引缓存
  13.     private ShortBuffer indexBuffer;
  14.  
  15.  
  16.     public Square() {
  17.         // 一个float为4 bytes, 因此要乘以4
  18.         ByteBuffer vbb = ByteBuffer.allocateDirect(vertices.length * 4);
  19.         vbb.order(ByteOrder.nativeOrder());
  20.         vertexBuffer = vbb.asFloatBuffer();
  21.         vertexBuffer.put(vertices);
  22.         vertexBuffer.position(0);
  23.         // short类型同理
  24.         ByteBuffer ibb = ByteBuffer.allocateDirect(indices.length * 2);
  25.         ibb.order(ByteOrder.nativeOrder());
  26.         indexBuffer = ibb.asShortBuffer();
  27.         indexBuffer.put(indices);
  28.         indexBuffer.position(0);
  29.         }
  30.  
  31.     /**
  32.      * 绘制正方形到屏幕
  33.      * 
  34.      * @param gl
  35.      */
  36.     public void draw(GL10 gl) {
  37.         // 逆时针环绕
  38.         gl.glFrontFace(GL10.GL_CCW);
  39.         // 开启剔除功能
  40.         gl.glEnable(GL10.GL_CULL_FACE);
  41.         // 剔除背面
  42.         gl.glCullFace(GL10.GL_BACK);
  43.         // 开启顶点缓存写入功能
  44.         gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
  45.         // 设置顶点
  46.         // size:每个顶点有几个数指描述。
  47.         // type:数组中每个顶点的坐标类型。
  48.         // stride:数组中每个顶点间的间隔,步长(字节位移)。
  49.         // pointer:存储着每个顶点的坐标值。初始值为0
  50.         gl.glVertexPointer(3, GL10.GL_FLOAT, 0, vertexBuffer);
  51.         gl.glDrawElements(GL10.GL_TRIANGLES, indices.length,
  52.         GL10.GL_UNSIGNED_SHORT, indexBuffer);
  53.         // 关闭各个功能
  54.         gl.glDisableClientState(GL10.GL_VERTEX_ARRAY);
  55.         gl.glDisable(GL10.GL_CULL_FACE);
  56.     }
  57. }
复制代码 我们必须在OpenGLRenderer类中初始化square
  1. square = new Square();<!--EndFragment-->
复制代码 并在主绘制方法中调用square的绘制方法
  1. public void onDrawFrame(GL10 gl) {
  2.     // 清除屏幕和深度缓存
  3.     gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT);
  4.     // 绘制正方形
  5.     square.draw(gl);
  6. }
复制代码 如果你现在运行应用,我们又看到了华丽的黑屏,为什么?因为OpenGL ES渲染默认的当前位置为(0,0,0),窗口的定位也一样。而且OpenGL ES不渲染太靠近窗体定位的东西。解决方法就是移动绘制的位置。
  1. gl.glTranslatef(0, 0, -4);  <!--EndFragment-->
复制代码 再次运行应用你将看到该正方形已经被绘制,但是它好像离我们越来越远一样,最后消失了。OpenGL ES不会在画面之间复位绘制点,所以我们要自己完成。
  1. // 重置当前的模型观察矩阵
  2. gl.glLoadIdentity();<!--EndFragment-->
复制代码 现在,我们运行应用将会看到一个固定位置的正方形。


 
 OpenGLDemo02.rar (64.87 KB, 下载次数: 460) 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
NEHE的OPENGL教程 第42课 多视窗口… NEHE的OPENGL教程 第42课 多视窗口… NeHe的OPENGL中文教程:第41课 体… NeHe的OPENGL中文教程:第40课 绳… NeHe的OPENGL中文教程:第39课 物… NeHe的OPENGL中文教程:第39课 物… NeHe的OPENGL中文教程:第38课 从… NeHe的OPENGL中文教程:第37课 卡… 愚人节十大IT假新闻:Opera浏览器… NeHe的OPENGL中文教程:第36课 放… NeHe的OPENGL中文教程:第35课 AVI… NeHe的OPENGL中文教程:第35课 AVI… NeHe的OPENGL中文教程:第34课 从… NeHe的OPENGL中文教程:第33课 加… NeHe的OPENGL中文教程:第32课 Alp… NeHe的OPENGL中文教程:第32课 Alp… NeHe的OPENGL中文教程:第32课 Alp… NeHe的OPENGL中文教程:第31课 模… NEHE的OPENGL中文教程:第30课 碰… NEHE的OPENGL中文教程:第30课 碰… NeHe的OPENGL中文教程:第29课 Bli… NeHe的OPENGL中文教程:第28课 贝… NeHe的OPENGL中文教程:第27课 影… NeHe的OPENGL中文教程:第26课剪裁… NeHe的OPENGL中文教程:第25课 变… NeHe的OPENGL中文教程:第24课 TAG… NeHe的OPENGL中文教程:第23课 球… NeHe的OPENGL中文教程:第22课 凸… NeHe的OPENGL中文教程:第22课 凸… NeHe的OPENGL中文教程:第21课 反… NeHe的OPENGL中文教程:第21课 反… NeHe的OPENGL中文教程:第20课 蒙… NeHe的OPENGL中文教程:第19课 粒… NeHe的OPENGL中文教程:第18课 二… NeHe的OPENGL中文教程:第17课 2D… NeHe的OPENGL中文教程:第16课 雾 NeHe的OPENGL中文教程:第15课 图… NeHe的OPENGL中文教程:第14课 图… NeHe的OPENGL中文教程:第13课 位… NeHe的OPENGL中文教程:第12课 显… NeHe的OPENGL中文教程:第11课 飘… NeHe的OPENGL中文教程:第十课 漫… NeHe的OPENGL中文教程:第九课 漂… NeHe的OPENGL中文教程:第八课 Alp… NeHe的OPENGL中文教程:第七课 纹… NeHe的OPENGL中文教程:第七课 纹… NeHe的OPENGL中文教程:第六课 纹… NeHe的OPENGL中文教程:第五课 向3… NeHe的OPENGL中文教程:第四课 旋… NeHe的OPENGL中文教程:第三课 着… NeHe的OPENGL中文教程:第二课 多… NeHe的OPENGL中文教程:第一课 新… NeHe的OPENGL中文教程:第一课 新… DirectX与OpenGL方面的经典电子书… VC++ 6.0下OpengGL配置以及glut配… 怎样开始学习OpenGL
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值