http://dev.10086.cn/cmdn/wiki/index.php?doc-view-4271.html
http://dev.10086.cn/cmdn/wiki/index.php?doc-view-4273.html
免责声明:
本人纯属个人学习使用,并无商业行为。
本人对上述文章进行转载,进行部分的整理。如作者本人或其他版权单位有争议请通知本人,将立刻删帖,谢谢。
源码以打包在附件中:
(一)
上一篇文章我们介绍了常见的各种游戏特效的实现,你现在可以很轻松的实现各种游戏中所需要的特效,但是,你可能已经意识到了,我们的游戏一般都需要进行碰撞检测,比如前面的火柴棍小人,我们需要检测子弹和敌人之间的碰撞;碰撞检测通常是游戏开发的难点,作为引擎必然少不了碰撞检测部分,这里我们还是按照cocos2d的构架,使用Box2d作为物理引擎,下面我们将通过在OPhone平台实现一个小游戏,来对Box2d物理引擎进行学习。
Box2d
Box2D是一个用于游戏的2D刚体仿真库,它可以使物体的运动更加真实,让游戏场景看起来更具交互性。2D物理引擎能增强游戏世界中物体如多边形(砖块,三角形,多边形)的动作的真实感从而提高游戏质量。该引擎通过用户设定的参数如重力,密度,摩擦,弹性等参数计算碰撞,角度,力和动力等。这些计算需要大量的数学,物理等知识,如果有兴趣也可以下载其源码来研究。
Box2d同时也提供了各种语言环境的实现,由于Ophone平台使用Java作为变成语言,所以我们将选择使用Box2d的java版JBox2d,这也将产生一个问题,JBox2D是用processing库来处理图像显示,所以Ophone平台上则不适用,在Ophone平台上的图像渲染主要包括两种:Canvas和Opengl ES,因此我们可以任选其中一种,这里为了配合我们的引擎实现,选择通过Opengl ES来作为渲染部分,这部分就需要我们自己来实现,其实我们也可以不使用其图像渲染部分,因为我们主要是使用Box2d来做物理检测,稍后我们会通过一个实例游戏来介绍。
另外,比较优秀的2D物理引擎还有Chipmunk,对于谁好谁坏,我们这里不去评价,如果要使用Chipmunk作为物理引擎会比Box2d稍微苦难一些,因为Chipmunk目前没有Java版本,所以只能通过JNI方式来使用,这就需要使用NDK来开发原生的C程序,使用C语言来做,效率要高很多,但是开发,调试的难度也将增加,有机会我们将可以介绍如何使用NDK来编写C程序,并同时整合Chipmunk物理引擎。
这里只是我们对Box2d的一个简单介绍,让大家明白其用处,关于更多详细信息,大家可以参考其官方网站http://www.box2d.org/,图12-1则是cocs2d中演示的Box2d物理引擎效果,学完这部分内容,你也可以很轻松将其运行在Ophone平台上。
图12-1中这每个方块都具有重力,摩擦力,碰撞检测规则,他们都处于同一个世界场景中,不必眼红iPhone开发者,下面就给大家看一下,我们在Ophone平台提供的示例物理小游戏。
在学习使用Box2D引擎之前,我们需要了解一下一些常用的概念:
刚体(rigid body)
一块十分坚硬的物质,它上面的任何两点之间的距离都是完全不变的。它们就像钻石那样坚硬。我们用物体(body)来代替刚体。
形状(shape)
一块严格依附于物体(body)的 2D 碰撞几何结构(collision geometry)。形状具有摩擦(friction)和恢复(restitution)的材料性质。
约束(constraint)
一个约束(constraint)就是消除物体自由度的物理连接。在 2D 中,一个物体有 3 个自由度。如果我们把一个物体钉在墙上(像摆锤那样),那我们就把它约束到了墙上。这样,此物体就只能绕着这个钉子旋转,所以这个约束消除了它 2 个自由度。
接触约束(contact constraint)
一个防止刚体穿透,以及用于模拟摩擦(friction)和恢复(restitution)的特殊约束。你永远都不必创建一个接触约束,它们会自动被 Box2D 创建。
关节(joint)
它是一种用于把两个或多个物体固定到一起的约束。Box2D 支持的关节类型有:旋转,棱柱,距离等等。关节可以支持限制(limits)和马达(motors)。
关节限制(joint limit)
一个关节限制(joint limit)限定了一个关节的运动范围。例如人类的胳膊肘只能做某一范围角度的运动。
关节马达(joint motor)
一个关节马达能依照关节的自由度来驱动所连接的物体。例如,你可以使用一个马达来驱动一个肘的旋转。
世界(world)
一个物理世界就是物体,形状和约束相互作用的集合。Box2D 支持创建多个世界,但这通常是不必要的。
这里先给大家介绍就是让大家明白Box2d包括哪些内容,稍后对框架的介绍时就能更加容易理解,当然对于这些具体的功能,我们会在后面跟着示例代码一起学习。
Ophone Box2d
首先分析一下我们在Ophone平台上的Box2dDemo需要实现什么功能,首先我们将整个屏幕构建成一个盒子,然后再盒子中设置各种障碍,当我们触摸屏幕上任意位置时,就释放一个当前选择的物体,然后该物体将受到重力等因素的影响开始运动,直到最后静止下来。运行效果如图12-2所示。
屏幕中间的长条则是我们设置的障碍,而圆形和矩形都是我们在点击屏幕时要释放的物体,前面我们说过,JBox2d中的图形部分在Ophone中不能用,所以我们会专门介绍如何通过Opengl ES来对图形图像进行渲染,另外,该示例中的这些物体都是通过纹理映射来将图片映射到四边形上。为了大家能掌握图形系统相关内容,我们还实现了一个功能,玩家可以自己设置障碍,只需要点击Menu中,选择"编辑模式"就可以进入障碍编辑状态,如图12-3所似。
当障碍编辑完成之后再次选择编辑模式,则恢复到游戏中,此时我们所编辑的这些障碍都会正常的运行。当然该过程我们并没有使用引擎来完成,目的在于让大家更清楚渲染的原理,以及代码能够更多的重用,也就是说,大家可以直接拷贝代码到需要的游戏中去即可。同时大家也能掌握很多Opengl ES的相关知识。
Ophone平台如何使用JBox2d
要使用JBox2d我们首先需要获得其源码或者jar包,这个就不用多说了,知道其官方网站下载即可,这里我们下载了一个完整版本jbox2d-2.0.1-full.jar,让后将其放入我们所建立OphoneBox2d工程的lib文件夹下,JBox2d中大致包含了如图12-4所示的一些包:
图12-4 JBox2d结构图
其中org.jbox2d.collision比较重要,主要负责处理碰撞相关,包括对一些多边形的实现,这里所说的多边形主要是一些数据,比如多边形的位置,大小,重力,形状,质量等属性;org.jbox2d.common包主要用来设置一些全局的属性(Setting.java),调试时所使用的颜色(Color3f.java),以及其他的一些数学相关的内容,因为我们说了Box2d他主要不是来做渲染的,但是有时候我们需要知道所设置的这些物体是否正确,进行调试,就需要绘制这些简单的图形,并显示出来,供我们调试;org.jbox2d.dynamics包主要负责动力学相关的内容,下面是常见的功能包描述。
org.jbox2d.collision包
AABB:AABB坐标
OBB:OBB坐标
ContactID:接触ID
ContactPoint:接触点
ManifoldPoint:繁殖点
Segment:线段
Shape:外形基类
ShapeDef:外形定义基类
CircleDef:圆外形定义
CircleShape:圆外形
FilterData:碰撞过滤器
MassData:质量运算器
PolygonDef:多边开定义
PolygonShape:凸多边形
org.jbox2d.common包
Color3f:调试绘图颜色
Settings:全局设置
Mat22:2*2 矩阵
Sweep:碰撞描述
Vec2:向量(x ,y)
XForm:坐标转换,平移或旋转
标准的版本中还会存在Mat33表示3*3的矩阵和Vec3向量(x,y,z),该java版本中没有出现这些。
org.jbox2d.dynamics包
Body:刚体或叫物体
BodyDef:刚体定义
BoundaryListener:世界边界侦听
ContactFilter:继承这个类用来获取过滤碰撞
ContactListener:继承这个类用来获取碰撞结果
DebugDraw:调试绘图,用于调试
DestructionListener:关节或外形销毁时处理方法
World:物理世界
org.jbox2d.dynamics.contacts
Contact:管理两个外形接触
ContactEdge:接触边用来连接多个物体和接触到一个接触表
ContactResult:记录接触结果
org.jbox2d.dynamics.Joints
DistanceJoint:距离校正器
DistanceJointDef:距离连接定义
GearJoint:齿轮
GearJointDef:齿轮连接定义
Joint:连接基类
JointDef:连接定义基类
JointEdge:用于组合刚体或连接到一起.刚体相当于节点,而连接相当于边
MouseJoint:鼠标连接
MouseJointDef:鼠标连接定义
PrismaticJoint:棱柱连接
PrismaticJointDef:棱柱连接定义
PulleyJoint:滑轮连接
PulleyJointDef:滑轮连接定义
RevoluteJoint:旋转连接
RevoluteJointDef:旋转连接定义
org.jbox2d.testbed:主要是一些用来测试的程序
添加JBox2d到Ophone项目中
要在工程中使用JBox2d库,需要将JBox2d添加到工程中,添加方法如下:
右键单击工程,选择"Properties",进入项目Properties界面。
选择"Java Build Path",选择"Libraries"选项卡。
在点击"Add Jars..."按钮,添加Jar。
选择当前工程中我们之前放入lib文件夹中的jbox2d-2.0.1-full.jar文件,如图12-5所示,单击"确定"按钮即可。
OphoneBox2d框架
现在工程的结构展开应该如图12-6所示。
其中实现该工程的文件如下:
Box2dTest:工程Activity,入口
GameGLSurfaceView:游戏GLSurfaceView
GLRenderer:Opengl es渲染器
DrawObject:使用Opengl ES来绘制常用图形(矩形,圆形)
PhysicsWorld:物理世界场景
OphoneBox2d实现
开始分析代码之前,我们先确定一下需要准备的资源图片,从图12-2所示,我们可以看出,多少需要一个矩形和一个圆形的图片(当然也可直接指定颜色绘制矩形和圆形),这里我们将使用图片来进行纹理映射,该工程所需要的纹理图片如图12-7所示。
Box2dTest实现
该类继承自Activity,将作为本程序的入口,授予我们是通过Opengl ES来渲染的,所以构建需要构建一个GLSurfaceView对象作为Opengl ES的窗口,然后通过setContentView函数来设置显示该窗口视图。然后分别在onPause和onResume函数中调用GLSurfaceView类的GLSurfaceView。当然这也是所有Opengl ES程序的渲染基础框架,所有的Opengl ES程序窗口都由GLSurfaceView来实现。具体实现入代码清单12-1所示。
代码清单12-1:Box2dTest.java
view plaincopy to clipboardprint?
01.import android.app.Activity;
02.import android.content.pm.ActivityInfo;
03.import android.opengl.GLSurfaceView;
04.import android.os.Bundle;
05.import android.view.Menu;
06.import android.view.MenuItem;
07.import android.view.WindowManager;
08.
09.public class Box2dTest extends Activity {
10. private GLSurfaceView mGLView;
11.
12. /** Called when the activity is first created. */
13. @Override
14. public void onCreate(Bundle savedInstanceState) {
15. super.onCreate(savedInstanceState);
16. // 构建GLSurfaceView视图
17. mGLView = new GameGLSurfaceView(this);
18. setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
19. getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
20. // 设置GLSurfaceView视图
21. setContentView(mGLView);
22. }
23.
24. protected void onPause() {
25. super.onPause();
26. mGLView.onPause();
27. }
28.
29. protected void onResume() {
30. super.onResume();
31. mGLView.onResume();
32. }
33.
34. // 创建按钮
35. public boolean onCreateOptionsMenu(Menu menu) {
36. menu.add(Menu.NONE, 1, Menu.NONE, "编辑模式");
37. menu.add(Menu.NONE, 2, Menu.NONE, "选择模型");
38. return true;
39. }
40.
41. // 按钮事件处理
42. public boolean onOptionsItemSelected(MenuItem item) {
43. switch (item.getItemId()) {
44. case 1:
45. // 切换编辑模式
46. ((GameGLSurfaceView) mGLView).toggleEdit();
47. return true;
48. case 2:
49. // 选择模型
50. ((GameGLSurfaceView) mGLView).toggleModel();
51. return true;
52. }
53. return false;
54. }
55.
56.}
import android.app.Activity;
import android.content.pm.ActivityInfo;
import android.opengl.GLSurfaceView;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.WindowManager;
public class Box2dTest extends Activity {
private GLSurfaceView mGLView;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 构建GLSurfaceView视图
mGLView = new GameGLSurfaceView(this);
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
// 设置GLSurfaceView视图
setContentView(mGLView);
}
protected void onPause() {
super.onPause();
mGLView.onPause();
}
protected void onResume() {
super.onResume();
mGLView.onResume();
}
// 创建按钮
public boolean onCreateOptionsMenu(Menu menu) {
menu.add(Menu.NONE, 1, Menu.NONE, "编辑模式");
menu.add(Menu.NONE, 2, Menu.NONE, "选择模型");
return true;
}
// 按钮事件处理
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case 1:
// 切换编辑模式
((GameGLSurfaceView) mGLView).toggleEdit();
return true;
case 2:
// 选择模型
((GameGLSurfaceView) mGLView).toggleModel();
return true;
}
return false;
}
}
前面我们说了,可以通过Menu来切换模式和选择模型,因此我们在onCreateOptionsMenu中添加了两个菜单选项,其中"编辑模式"用来切换编辑状态和游戏状态,"选择模型"则用于选择我们触摸屏幕时所释放的模型,这里主要包括矩形和圆形,因此在菜单事件处理函数onOptionsItemSelected中,我们分别对两个菜单选项进行了处理,具体实现位于GameGLSurfaceView中。
GameGLSurfaceView实现
要实现一个用于显示Opengl ES窗口程序的视图,Ophone为我们提供了GLSurfaceView类,因此我们的GameGLSurfaceView类可以继承自GLSurfaceView很轻松的实现Opengl ES视图,该视图中有一个重要的部分,那就是Renderer,每一个Opengl ES窗口视图都需要一个渲染器来负责渲染。本例中的渲染器GLRenderer直接通过继承自Renderer来实现,GLSurfaceView中可以通过setRenderer来设置一个自定义的渲染器。具体实现入代码清单12-2所示。
代码清单12-2:GLSurfaceView.java
view plaincopy to clipboardprint?
01.package com.oger.demo.box2d.activity;
02.
03.import android.content.Context;
04.import android.opengl.GLSurfaceView;
05.import android.view.MotionEvent;
06.
07.public class GameGLSurfaceView extends GLSurfaceView {
08. GLRenderer mRenderer;
09.
10. public GameGLSurfaceView(Context context) {
11. super(context);
12. mRenderer = new GLRenderer(context);
13. setRenderer(mRenderer);
14. }
15.
16. public void toggleEdit() {
17. mRenderer.toggleEdit();
18. }
19.
20. public void toggleModel() {
21. mRenderer.switchModel();
22. }
23.
24. public boolean onTouchEvent(final MotionEvent event) {
25. mRenderer.setSize(this.getWidth(), this.getHeight());
26. // 线程通信
27. queueEvent(new Runnable() {
28. public void run() {
29. mRenderer.touchEvent(event.getX(), event.getY(),
30. event.getAction());
31. }
32. });
33. // sleep
34. try {
35. Thread.sleep(20);
36. } catch (InterruptedException e) {
37. e.printStackTrace();
38. }
39. return true;
40. }
41.}
package com.oger.demo.box2d.activity;
import android.content.Context;
import android.opengl.GLSurfaceView;
import android.view.MotionEvent;
public class GameGLSurfaceView extends GLSurfaceView {
GLRenderer mRenderer;
public GameGLSurfaceView(Context context) {
super(context);
mRenderer = new GLRenderer(context);
setRenderer(mRenderer);
}
public void toggleEdit() {
mRenderer.toggleEdit();
}
public void toggleModel() {
mRenderer.switchModel();
}
public boolean onTouchEvent(final MotionEvent event) {
mRenderer.setSize(this.getWidth(), this.getHeight());
// 线程通信
queueEvent(new Runnable() {
public void run() {
mRenderer.touchEvent(event.getX(), event.getY(),
event.getAction());
}
});
// sleep
try {
Thread.sleep(20);
} catch (InterruptedException e) {
e.printStackTrace();
}
return true;
}
}
总结
本文主要介绍了开源2D物理引擎Box2d的一些功能特征,以及如何在Ophone平台上使用Box2d的java版JBox2d;同时我们完成了使用该物理引擎实现的小游戏的框架,接下来我们就将通过完整实现该游戏的过程来学习使用JBox2d作为游戏开发中的物理引擎部分,同时大家可以考虑如何使用JNI来使用Box2d开发原生程序,加油哦,下一篇文章我们就将做出一个物理效果很酷的游戏来了。由于分享经验的心情急切,难免会出现一些疏忽或错误,还望不吝赐教!
(二)
接着上一篇我们构建的游戏框架继续完善,加上渲染器(GLRenderer),加上物理系统(PhysicsWorld),使用Opengl ES来作为JBox2d的图像管然系统(DrawObject),就将完成上一篇中给大家演示的示例游戏了,另外,这一节我们还将介绍一些Opengl ES相关的知识,如果你不太熟悉就可以google一些相关资料,同时你还需要熟悉一些基本的物理学概念,例如质量,力,扭矩和冲量。因为它可以使你很好地了解一些基本概念,以便你使用 Box2D,同时如果你好奇 Box2D 内部是如何工作的,你可以看这些文档。
渲染器(GLRenderer)
上一篇我们完成了GameGLSurfaceView,同时也将其渲染器设置为了GLRenderer,它才是我们所有Opengl ES程序的核心,GLRenderer将继承自GLSurfaceView.Renderer,需要实现以下三个接口:
onSurfaceCreated():该方法在渲染开始前调用,OpenGL ES的绘制上下文被重建时也会被调用。当activity暂停时绘制上下文会丢失,当activity继续时,绘制上下文会被重建。另外,创建长期存在的OpenGL资源(如texture)往往也在这里进行。
onSurfaceChanged():当surface的尺寸发生改变时该方法被调用。我们可以在这里设置视口。若你的 camera 是固定的,也可以在这里设置 camera。
onDrawFrame():每帧都通过该方法进行绘制。绘制时通常先调用glClear函数来清空 framebuffer,然后在调用OpenGL ES的起它的接口进行绘制。
具体实现如代码清单13-1所示,省略部分代码,(全部代码已整理)
代码清单13-1:GLRenderer片段
view plaincopy to clipboardprint?
01.package com.oger.demo.box2d.activity;
02.
03.import javax.microedition.khronos.egl.EGLConfig;
04.import javax.microedition.khronos.opengles.GL10;
05.
06.import org.jbox2d.collision.CircleShape;
07.import org.jbox2d.collision.PolygonShape;
08.import org.jbox2d.collision.Shape;
09.import org.jbox2d.collision.ShapeType;
10.import org.jbox2d.common.Vec2;
11.import org.jbox2d.dynamics.Body;
12.
13.import android.content.Context;
14.import android.opengl.GLSurfaceView;
15.import android.opengl.GLU;
16.import android.view.MotionEvent;
17.
18.public class GLRenderer implements GLSurfaceView.Renderer {
19. // 窗口的宽度和高度
20. private int width;
21. private int height;
22.
23. // 选择激活的模型
24. public void switchModel() {
25. activeModel++;
26. if (activeModel > 2) {
27. activeModel = 0;
28. }
29. }
30.
31. // 用于切换编辑模式与游戏模式
32. public void toggleEdit() {
33. if (editMode == false) {
34. editMode = true;
35. } else {
36. editMode = false;
37. }
38. }
39.
40. public void onSurfaceChanged(GL10 gl, int w, int h) {
41. // 设置视口
42. gl.glViewport(0, 0, w, h);
43. }
44.
45. public void onSurfaceCreated(GL10 gl, EGLConfig arg1) {
46. // 设置为正交视口
47. GLU.gluOrtho2D(gl, -12f, 12f, -20f, 20f);
48. // 允许顶点数组和纹理坐标数组
49. gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
50. gl.glEnableClientState(GL10.GL_TEXTURE_COORD_ARRAY);
51. // 设置纹理映射方式
52. gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_S,
53. GL10.GL_REPEAT);
54. gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_T,
55. GL10.GL_REPEAT);
56. }
57.
58. // 设置尺寸
59. public void setSize(int x, int y) {
60. this.width = x;
61. this.height = y;
62. }
63.
64. // 物理世界
65. private PhysicsWorld mWorld;
66. // box对象
67. private DrawObject mBox;
68. // 矩形条对象
69. private DrawObject mLongBox;
70. // 圆形对象
71. private DrawObject mCircle;
72. // 当前选择的模型
73. private int activeModel = 1;
74. // 是否处于编辑状态
75. private boolean editMode = false;
76. // Context,用于装载资源
77. private Context mContext;
78. // 保存鼠标的坐标
79. private float startX, endX, startY, endY;
80.
81. public GLRenderer(Context newContext) {
82. mContext = newContext;
83. // 定一个box
84. mBox = new DrawObject(
85. new float[] { -1, -1, 0, 1, -1, 0, 1, 1, 0, -1, 1, 0 },
86. new float[] { 0f, 1f, 1f, 1f, 1f, 0f, 0f, 0f },
87. new short[] { 0, 1, 2, 3, 0 }, 5);
88. // 定义一个长条
89. mLongBox = new DrawObject(
90. new float[] { -.2f, -2f, 0, .2f, -2f, 0, .2f,2f, 0, -.2f, 2f, 0 },
91. new float[] { 0f, 1f, 1f, 1f, 1f, 0f, 0f,0f },
92. new short[] { 0, 1, 2, 3, 0 }, 5);
93. // 圆形
94. mCircle = new DrawObject(new float[] { 0f, 0f, 0f, 0f, 1f, 0f, -.5f,
95. .866f, 0f, -.866f, .5f, 0f, -1f, 0f, 0f, -.866f, -.5f, 0f,
96. -.5f, -.866f, 0f, 0f, -1f, 0f, .5f, -.866f, 0f, .866f, -.5f,
97. 0f, 1f, 0f, 0f, .866f, .5f, 0f, .5f, .866f, 0f, 0f, 1f, 0f },
98. new float[] { 0.5f, 0.5f, 0.5f, 0.0f, .25f, .067f, .067f, .25f,
99. 0.0f, 0.5f, .067f, .75f, .25f, .933f, 0.5f, 1.0f, .75f,
100. .933f, .933f, .75f, 1.0f, 0.5f, .933f, .25f, .75f,
101. .067f, .5f, .0f },
102. new short[] { 0, 1, 2, 3, 4, 5, 6,
103. 7, 8, 9, 10, 11, 12, 13, 14 }, 14);
104. // 创建一个物理场景
105. mWorld = new PhysicsWorld();
106. mWorld.createWorld();
107. // 向物理场景添加两个实体
108. mWorld.addBox(0f, -25f, 50f, 10f, 0f, false);
109. mWorld.addBall(0f, -15f, 7f, false);
110. mWorld.addBox(-12f, -25f, 0.6f, 40f, 0f, false);
111. mWorld.addBox(12f, -25f, 0.6f, 40f, 0f, false);
112. }
113.
114. // 绘制当前选择的模型
115. public void drawActiveBody(GL10 gl) {
116. float x = 10f;
117. float y = 17f;
118. switch (activeModel) {
119. case 0:
120. mCircle.draw(gl, x, y, 0f);
121. break;
122. case 1:
123. mBox.draw(gl, x, y, 0f);
124. break;
125. case 2:
126. mLongBox.draw(gl, x, y, 0f);
127. break;
128. }
129. }
130.
131. public void onDrawFrame(GL10 gl) {
132. if (editMode) {
133. // 编辑模式则渲染为红色背景
134. gl.glClearColor(0.5f, 0, 0f, 1.0f);
135. } else {
136. // 游戏模式则渲染为蓝色
137. gl.glClearColor(0f, 0, 0.5f, 1.0f);
138. }
139. // 清理颜色缓冲区和深度缓冲区
140. gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT);
141. // 设置材质的混合模式
142. gl.glTexEnvx(GL10.GL_TEXTURE_ENV, GL10.GL_TEXTURE_ENV_MODE,
143. GL10.GL_REPLACE);
144. // 绘制游戏模式下的当前选择的模型
145. if (!editMode) {
146. drawActiveBody(gl);
147. }
148. // 绘制编辑模式时的矩形条
149. if (editMode) {
150. float midX = (startX + endX) / 2f;
151. float midY = (startY + endY) / 2f;
152. float sizeX = (endX - midX);
153. float sizeY = (endY - midY);
154. float rotate = (float) Math.atan((double) (sizeY / sizeX));
155. float size = (float) Math
156. .sqrt((double) ((sizeX * sizeX) + (sizeY * sizeY)));
157. mBox.draw(gl, midX, midY, 0f, rotate * 57.2957795f, size, .2f);
158. }
159. Vec2 vec;
160. // 得到世界场景中的实体列表
161. Body mBody = mWorld.getBodyList();
162. do {
163. // 取得该实体的形状列表
164. Shape mShape = mBody.getShapeList();
165. if (mShape != null) {
166. vec = mBody.getPosition();
167. float rot = mBody.getAngle() * 57f;
168. // 弧度转化为角度
169. if (ShapeType.POLYGON_SHAPE == mShape.getType()) {
170. Vec2[] vertexes = ((PolygonShape) mShape).getVertices();
171. mBox.draw(gl, vec.x, vec.y, 0f, rot, vertexes[2].x,
172. vertexes[2].y);
173. } else if (ShapeType.CIRCLE_SHAPE == mShape.getType()) {
174. float radius = ((CircleShape) mShape).m_radius;
175. mCircle.draw(gl, vec.x, vec.y, 0f, rot, radius);
176. }
177. }
178. // 取得下一个实体
179. mBody = mBody.getNext();
180. } while (mBody != null);
181. // 更新场景
182. mWorld.update();
183. }
184.
185. // 在编辑模式时添加一条线
186. public void addLine() {
187. float midX = (startX + endX) / 2f;
188. float midY = (startY + endY) / 2f;
189. float sizeX = (endX - midX);
190. float sizeY = (endY - midY);
191. float rotate = (float) Math.atan((double) (sizeY / sizeX));
192. float size = (float) Math
193. .sqrt((double) ((sizeX * sizeX) + (sizeY * sizeY)));
194. // 添加一个长的box
195. mWorld.addBox(midX, midY, size, .2f, rotate, false);
196. startX = 0;
197. startY = 0;
198. endX = 0;
199. endY = 0;
200. }
201.
202. public void touchEvent(float x, float y, int eventCode) {
203. // 计算x,y对应的场景坐标
204. float worldX = ((x - (this.width / 2)) * 12f) / (this.width / 2);
205. float worldY = ((y - (this.height / 2)) * -20f) / (this.height / 2);
206. if (!editMode) {
207. // 当鼠标松开时,添加一个指定的实体
208. if (eventCode == MotionEvent.ACTION_UP) {
209. switch (activeModel) {
210. case 0:
211. mWorld.addBall(worldX, worldY, 0.98f, true);
212. break;
213. case 1:
214. mWorld.addBox(worldX, worldY, .98f, .98f, 0f, true);
215. break;
216. case 2:
217. mWorld.addBox(worldX, worldY, .2f, 2f, 0f, true);
218. break;
219. }
220. }
221. } else {
222. // 确定要添加的线的坐标
223. if (eventCode == MotionEvent.ACTION_DOWN) {
224. startX = worldX;
225. startY = worldY;
226. endX = worldX;
227. endY = worldY;
228. } else if (eventCode == MotionEvent.ACTION_MOVE) {
229. endX = worldX;
230. endY = worldY;
231. } else if (eventCode == MotionEvent.ACTION_UP) {
232. endX = worldX;
233. endY = worldY;
234. addLine();
235. }
236. }
237. }
238.}
package com.oger.demo.box2d.activity;
import javax.microedition.khronos.egl.EGLConfig;
import javax.microedition.khronos.opengles.GL10;
import org.jbox2d.collision.CircleShape;
import org.jbox2d.collision.PolygonShape;
import org.jbox2d.collision.Shape;
import org.jbox2d.collision.ShapeType;
import org.jbox2d.common.Vec2;
import org.jbox2d.dynamics.Body;
import android.content.Context;
import android.opengl.GLSurfaceView;
import android.opengl.GLU;
import android.view.MotionEvent;
public class GLRenderer implements GLSurfaceView.Renderer {
// 窗口的宽度和高度
private int width;
private int height;
// 选择激活的模型
public void switchModel() {
activeModel++;
if (activeModel > 2) {
activeModel = 0;
}
}
// 用于切换编辑模式与游戏模式
public void toggleEdit() {
if (editMode == false) {
editMode = true;
} else {
editMode = false;
}
}
public void onSurfaceChanged(GL10 gl, int w, int h) {
// 设置视口
gl.glViewport(0, 0, w, h);
}
public void onSurfaceCreated(GL10 gl, EGLConfig arg1) {
// 设置为正交视口
GLU.gluOrtho2D(gl, -12f, 12f, -20f, 20f);
// 允许顶点数组和纹理坐标数组
gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
gl.glEnableClientState(GL10.GL_TEXTURE_COORD_ARRAY);
// 设置纹理映射方式
gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_S,
GL10.GL_REPEAT);
gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_T,
GL10.GL_REPEAT);
}
// 设置尺寸
public void setSize(int x, int y) {
this.width = x;
this.height = y;
}
// 物理世界
private PhysicsWorld mWorld;
// box对象
private DrawObject mBox;
// 矩形条对象
private DrawObject mLongBox;
// 圆形对象
private DrawObject mCircle;
// 当前选择的模型
private int activeModel = 1;
// 是否处于编辑状态
private boolean editMode = false;
// Context,用于装载资源
private Context mContext;
// 保存鼠标的坐标
private float startX, endX, startY, endY;
public GLRenderer(Context newContext) {
mContext = newContext;
// 定一个box
mBox = new DrawObject(
new float[] { -1, -1, 0, 1, -1, 0, 1, 1, 0, -1, 1, 0 },
new float[] { 0f, 1f, 1f, 1f, 1f, 0f, 0f, 0f },
new short[] { 0, 1, 2, 3, 0 }, 5);
// 定义一个长条
mLongBox = new DrawObject(
new float[] { -.2f, -2f, 0, .2f, -2f, 0, .2f,2f, 0, -.2f, 2f, 0 },
new float[] { 0f, 1f, 1f, 1f, 1f, 0f, 0f,0f },
new short[] { 0, 1, 2, 3, 0 }, 5);
// 圆形
mCircle = new DrawObject(new float[] { 0f, 0f, 0f, 0f, 1f, 0f, -.5f,
.866f, 0f, -.866f, .5f, 0f, -1f, 0f, 0f, -.866f, -.5f, 0f,
-.5f, -.866f, 0f, 0f, -1f, 0f, .5f, -.866f, 0f, .866f, -.5f,
0f, 1f, 0f, 0f, .866f, .5f, 0f, .5f, .866f, 0f, 0f, 1f, 0f },
new float[] { 0.5f, 0.5f, 0.5f, 0.0f, .25f, .067f, .067f, .25f,
0.0f, 0.5f, .067f, .75f, .25f, .933f, 0.5f, 1.0f, .75f,
.933f, .933f, .75f, 1.0f, 0.5f, .933f, .25f, .75f,
.067f, .5f, .0f },
new short[] { 0, 1, 2, 3, 4, 5, 6,
7, 8, 9, 10, 11, 12, 13, 14 }, 14);
// 创建一个物理场景
mWorld = new PhysicsWorld();
mWorld.createWorld();
// 向物理场景添加两个实体
mWorld.addBox(0f, -25f, 50f, 10f, 0f, false);
mWorld.addBall(0f, -15f, 7f, false);
mWorld.addBox(-12f, -25f, 0.6f, 40f, 0f, false);
mWorld.addBox(12f, -25f, 0.6f, 40f, 0f, false);
}
// 绘制当前选择的模型
public void drawActiveBody(GL10 gl) {
float x = 10f;
float y = 17f;
switch (activeModel) {
case 0:
mCircle.draw(gl, x, y, 0f);
break;
case 1:
mBox.draw(gl, x, y, 0f);
break;
case 2:
mLongBox.draw(gl, x, y, 0f);
break;
}
}
public void onDrawFrame(GL10 gl) {
if (editMode) {
// 编辑模式则渲染为红色背景
gl.glClearColor(0.5f, 0, 0f, 1.0f);
} else {
// 游戏模式则渲染为蓝色
gl.glClearColor(0f, 0, 0.5f, 1.0f);
}
// 清理颜色缓冲区和深度缓冲区
gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT);
// 设置材质的混合模式
gl.glTexEnvx(GL10.GL_TEXTURE_ENV, GL10.GL_TEXTURE_ENV_MODE,
GL10.GL_REPLACE);
// 绘制游戏模式下的当前选择的模型
if (!editMode) {
drawActiveBody(gl);
}
// 绘制编辑模式时的矩形条
if (editMode) {
float midX = (startX + endX) / 2f;
float midY = (startY + endY) / 2f;
float sizeX = (endX - midX);
float sizeY = (endY - midY);
float rotate = (float) Math.atan((double) (sizeY / sizeX));
float size = (float) Math
.sqrt((double) ((sizeX * sizeX) + (sizeY * sizeY)));
mBox.draw(gl, midX, midY, 0f, rotate * 57.2957795f, size, .2f);
}
Vec2 vec;
// 得到世界场景中的实体列表
Body mBody = mWorld.getBodyList();
do {
// 取得该实体的形状列表
Shape mShape = mBody.getShapeList();
if (mShape != null) {
vec = mBody.getPosition();
float rot = mBody.getAngle() * 57f;
// 弧度转化为角度
if (ShapeType.POLYGON_SHAPE == mShape.getType()) {
Vec2[] vertexes = ((PolygonShape) mShape).getVertices();
mBox.draw(gl, vec.x, vec.y, 0f, rot, vertexes[2].x,
vertexes[2].y);
} else if (ShapeType.CIRCLE_SHAPE == mShape.getType()) {
float radius = ((CircleShape) mShape).m_radius;
mCircle.draw(gl, vec.x, vec.y, 0f, rot, radius);
}
}
// 取得下一个实体
mBody = mBody.getNext();
} while (mBody != null);
// 更新场景
mWorld.update();
}
// 在编辑模式时添加一条线
public void addLine() {
float midX = (startX + endX) / 2f;
float midY = (startY + endY) / 2f;
float sizeX = (endX - midX);
float sizeY = (endY - midY);
float rotate = (float) Math.atan((double) (sizeY / sizeX));
float size = (float) Math
.sqrt((double) ((sizeX * sizeX) + (sizeY * sizeY)));
// 添加一个长的box
mWorld.addBox(midX, midY, size, .2f, rotate, false);
startX = 0;
startY = 0;
endX = 0;
endY = 0;
}
public void touchEvent(float x, float y, int eventCode) {
// 计算x,y对应的场景坐标
float worldX = ((x - (this.width / 2)) * 12f) / (this.width / 2);
float worldY = ((y - (this.height / 2)) * -20f) / (this.height / 2);
if (!editMode) {
// 当鼠标松开时,添加一个指定的实体
if (eventCode == MotionEvent.ACTION_UP) {
switch (activeModel) {
case 0:
mWorld.addBall(worldX, worldY, 0.98f, true);
break;
case 1:
mWorld.addBox(worldX, worldY, .98f, .98f, 0f, true);
break;
case 2:
mWorld.addBox(worldX, worldY, .2f, 2f, 0f, true);
break;
}
}
} else {
// 确定要添加的线的坐标
if (eventCode == MotionEvent.ACTION_DOWN) {
startX = worldX;
startY = worldY;
endX = worldX;
endY = worldY;
} else if (eventCode == MotionEvent.ACTION_MOVE) {
endX = worldX;
endY = worldY;
} else if (eventCode == MotionEvent.ACTION_UP) {
endX = worldX;
endY = worldY;
addLine();
}
}
}
}
在OpenGL初始化完成之后,我们应该进行一些视图设置。首先是设定视见区域,即告诉OpenGL应把渲染之后的图形绘制在窗体的哪个部位。当视见区域是整个窗体时,OpenGL将把渲染结果绘制到整个窗口。我们可以调用glViewPort函数来决定视见区域;在onSurfaceChanged函数中我们通过glViewport设置了Opengl ES的视口,其中参数X,Y指定了视见区域的左下角在窗口中的位置,一般情况下为(0,0),Width和Height指定了视见区域的宽度和高度。注意OpenGL使用的窗口坐标和Android使用的窗口坐标是不一样的。Opengl使用的窗口坐标的远点位于屏幕左下角。
在onSurfaceCreated函数中,首先通过GLU.gluOrtho2D函数设置二维坐标系统参数,函数有4个参数,可以理解为用该函数设置后,这个二维坐标系的左上角的坐标为(left,top),右下角的坐标为(right,bottom)。如果保持画图的参数不变,将左上角和右下角表示的范围扩大,则图像看起来就缩小了,反之就放大了,这些坐标同样以左下角作为坐标原点。
然后,由于我们绘制这些2D的物体时,需要使用顶点数组和纹理坐标数组,所以通过glEnableClientState函数和参数GL_VERTEX_ARRAY和GL_TEXTURE_COORD_ARRAY分别打开了允许设置顶点数组和纹理坐标数组,稍后再具体绘制时,大家会看到我们如何设置顶点数组和纹理数组的。一般情况我们将图象从纹理图象空间映射到帧缓冲图象空间(映射需要重新构造纹理图像,这样就会造成应用到多边形上的图像失真),这时我们就可用glTexParmeterx()函数来确定如何把纹理象素映射成像素,即纹理映射的方式,通常还有glTexParmeteri函数等,因为opengl提供了几套不同数据类型的函数来完成通一个功能,其中第一个参数GL_TEXTURE_2D表示我们将操作的是2D纹理,因为Opengl支持一维纹理、二维纹理,但是Opengl ES只支持二维纹理,所以第一个参数不会怎么变化,后面的参数通常是需要进行组合的,主要有以下几个参数可以选择设置:
GL_TEXTURE_WRAP_S: S方向上的贴图模式
GL_CLAMP: 将纹理坐标限制在0.0,1.0的范围之内。边缘将会拉伸填充。
GL_TEXTURE_MAG_FILTER: 放大过滤
GL_LINEAR: 线性过滤, 使用距离当前渲染像素中心最近的4个纹素加权平均值
GL_TEXTURE_MIN_FILTER: 缩小过滤
GL_LINEAR_MIPMAP_NEAREST: 使用GL_NEAREST对最接近当前多边形的解析度的两个层级贴图进行采样,然后用这两个值进行线性插值
最后,代码中定义了activeModel和editMode分别表示当前所选择激活的模型,和是否处于编辑状态,然后分别通过函数switchModel和toggleEdit来控制操作者两个状态,逻辑非常简单,大家看看代码就明白了,我们就不多浪费时间了。其中的变量width和height主要是表示窗口的宽度和高度,大家可能已经注意到在GameGLSurfaceView中的onTouchEvent函数中,我们每次都调用GLRenderer的setSize函数来设置了窗口的宽度和高度,在GLRenderer中同样是用于触摸事件的处理用,主要是将触摸坐标转换为场景中的坐标,稍后我们会介绍如何转换。到这里我们就实现了GLRenderer的一部分,下面我们开始学习如何在这里来使用JBox2d了,后面还会介绍在GLRenderer中整个该物理引擎的使用。
使用Opengl绘制图形(DrawObject)
DrawObject主要用于替代JBox2d中的图像渲染部分,但是这里我们只是实现了绘制矩形和圆形,还有更多的没有实现,该游戏中也将主要使用这样两种形状。DrawObject的实现几乎就全是Opengl相关的内容,我们先看具体代码,在来分析,如代码清单13-2所示。
代码清单13-2:DrawObject.java
view plaincopy to clipboardprint?
01.import java.nio.ByteBuffer;
02.import java.nio.ByteOrder;
03.import java.nio.FloatBuffer;
04.import java.nio.ShortBuffer;
05.
06.import javax.microedition.khronos.opengles.GL10;
07.
08.import android.content.Context;
09.import android.graphics.Bitmap;
10.import android.graphics.BitmapFactory;
11.import android.opengl.GLUtils;
12.
13.public class DrawObject {
14. // 顶点缓冲区
15. private FloatBuffer mVertexBuffer;
16. // 索引缓冲区
17. private ShortBuffer mIndexBuffer;
18. // 纹理坐标缓冲区
19. private FloatBuffer mTexBuffer;
20. // 顶点计数
21. private int vertexCount = 0;
22. // 是否拥有贴图
23. private boolean hasTexture = false;
24. // 纹理
25. private int[] mTexture = new int[1];
26.
27. // float[]->FloatBuffer
28. protected static FloatBuffer makeFloatBuffer(float[] arr) {
29. ByteBuffer bb = ByteBuffer.allocateDirect(arr.length * 4);
30. bb.order(ByteOrder.nativeOrder());
31. FloatBuffer fb = bb.asFloatBuffer();
32. fb.put(arr);
33. fb.position(0);
34. return fb;
35. }
36.
37. // short[]->ShortBuffer
38. protected static ShortBuffer makeShortBuffer(short[] arr) {
39. ByteBuffer bb = ByteBuffer.allocateDirect(arr.length * 4);
40. bb.order(ByteOrder.nativeOrder());
41. ShortBuffer ib = bb.asShortBuffer();
42. ib.put(arr);
43. ib.position(0);
44. return ib;
45. }
46.
47. // 构造(顶点数组,纹理数组,索引数组,顶点数)
48. public DrawObject(float[] coords, float[] tcoords, short[] icoords,
49. int vertexes) {
50. this(coords, icoords, vertexes);
51. mTexBuffer = makeFloatBuffer(tcoords);
52. }
53.
54. // 构造(顶点数组,索引数组,顶点数)
55. public DrawObject(float[] coords, short[] icoords, int vertexes) {
56. vertexCount = vertexes;
57. mVertexBuffer = makeFloatBuffer(coords);
58. mIndexBuffer = makeShortBuffer(icoords);
59. }
60.
61. // 装载贴图(gl,context,资源id)
62. public void loadTexture(GL10 gl, Context mContext, int mTex) {
63. hasTexture = true;
64. // 生成纹理
65. gl.glGenTextures(1, mTexture, 0);
66. // 绑定纹理
67. gl.glBindTexture(GL10.GL_TEXTURE_2D, mTexture[0]);
68. // 资源Bitmap
69. Bitmap bitmap;
70. bitmap = BitmapFactory.decodeResource(mContext.getResources(), mTex);
71. // 指定纹理图像
72. GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, bitmap, 0);
73. bitmap.recycle();
74. // 设置纹理参数
75. gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER,
76. GL10.GL_LINEAR);
77. gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER,
78. GL10.GL_LINEAR);
79. }
80.
81. /*----------------------*/
82. // 渲染obj
83. /*----------------------*/
84. public void draw(GL10 gl) {
85. // 判断是否拥有纹理
86. if (hasTexture) {
87. // 打开2d纹理
88. gl.glEnable(GL10.GL_TEXTURE_2D);
89. // 绑定纹理
90. gl.glBindTexture(GL10.GL_TEXTURE_2D, mTexture[0]);
91. // 设置纹理缓冲区
92. gl.glTexCoordPointer(2, GL10.GL_FLOAT, 0, mTexBuffer);
93. } else {
94. // 关闭2d纹理
95. gl.glDisable(GL10.GL_TEXTURE_2D);
96. }
97. // 设置逆时针方向为正面
98. gl.glFrontFace(GL10.GL_CCW);
99. // 设置顶点数组
100. gl.glVertexPointer(3, GL10.GL_FLOAT, 0, mVertexBuffer);
101. // 绘制
102. gl.glDrawElements(GL10.GL_TRIANGLE_FAN, vertexCount,
103. GL10.GL_UNSIGNED_SHORT, mIndexBuffer);
104. // 关闭2d纹理
105. gl.glDisable(GL10.GL_TEXTURE_2D);
106. }
107.
108. public void draw(GL10 gl, float x, float y, float z, float rot, float scale) {
109. this.draw(gl, x, y, z, rot, scale, scale);
110. }
111.
112. // draw(gl,x,y,z,旋转角度,x方向缩放,y方向缩放)
113. public void draw(GL10 gl, float x, float y, float z, float rot,
114. float scaleX, float scaleY) {
115. gl.glPushMatrix();
116. gl.glTranslatef(x, y, z);
117. gl.glRotatef(rot, 0f, 0f, 1f);
118. gl.glScalef(scaleX, scaleY, 1f);
119. this.draw(gl);
120. gl.glPopMatrix();
121. }
122.
123. public void draw(GL10 gl, float x, float y, float z, float rot) {
124. gl.glPushMatrix();
125. gl.glTranslatef(x, y, z);
126. gl.glRotatef(rot, 0f, 0f, 1f);
127. this.draw(gl);
128. gl.glPopMatrix();
129. }
130.
131. public void draw(GL10 gl, float x, float y, float z) {
132. gl.glPushMatrix();
133. gl.glTranslatef(x, y, z);
134. this.draw(gl);
135. gl.glPopMatrix();
136. }
137.}
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.FloatBuffer;
import java.nio.ShortBuffer;
import javax.microedition.khronos.opengles.GL10;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.opengl.GLUtils;
public class DrawObject {
// 顶点缓冲区
private FloatBuffer mVertexBuffer;
// 索引缓冲区
private ShortBuffer mIndexBuffer;
// 纹理坐标缓冲区
private FloatBuffer mTexBuffer;
// 顶点计数
private int vertexCount = 0;
// 是否拥有贴图
private boolean hasTexture = false;
// 纹理
private int[] mTexture = new int[1];
// float[]->FloatBuffer
protected static FloatBuffer makeFloatBuffer(float[] arr) {
ByteBuffer bb = ByteBuffer.allocateDirect(arr.length * 4);
bb.order(ByteOrder.nativeOrder());
FloatBuffer fb = bb.asFloatBuffer();
fb.put(arr);
fb.position(0);
return fb;
}
// short[]->ShortBuffer
protected static ShortBuffer makeShortBuffer(short[] arr) {
ByteBuffer bb = ByteBuffer.allocateDirect(arr.length * 4);
bb.order(ByteOrder.nativeOrder());
ShortBuffer ib = bb.asShortBuffer();
ib.put(arr);
ib.position(0);
return ib;
}
// 构造(顶点数组,纹理数组,索引数组,顶点数)
public DrawObject(float[] coords, float[] tcoords, short[] icoords,
int vertexes) {
this(coords, icoords, vertexes);
mTexBuffer = makeFloatBuffer(tcoords);
}
// 构造(顶点数组,索引数组,顶点数)
public DrawObject(float[] coords, short[] icoords, int vertexes) {
vertexCount = vertexes;
mVertexBuffer = makeFloatBuffer(coords);
mIndexBuffer = makeShortBuffer(icoords);
}
// 装载贴图(gl,context,资源id)
public void loadTexture(GL10 gl, Context mContext, int mTex) {
hasTexture = true;
// 生成纹理
gl.glGenTextures(1, mTexture, 0);
// 绑定纹理
gl.glBindTexture(GL10.GL_TEXTURE_2D, mTexture[0]);
// 资源Bitmap
Bitmap bitmap;
bitmap = BitmapFactory.decodeResource(mContext.getResources(), mTex);
// 指定纹理图像
GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, bitmap, 0);
bitmap.recycle();
// 设置纹理参数
gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER,
GL10.GL_LINEAR);
gl.glTexParameterx(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER,
GL10.GL_LINEAR);
}
/*----------------------*/
// 渲染obj
/*----------------------*/
public void draw(GL10 gl) {
// 判断是否拥有纹理
if (hasTexture) {
// 打开2d纹理
gl.glEnable(GL10.GL_TEXTURE_2D);
// 绑定纹理
gl.glBindTexture(GL10.GL_TEXTURE_2D, mTexture[0]);
// 设置纹理缓冲区
gl.glTexCoordPointer(2, GL10.GL_FLOAT, 0, mTexBuffer);
} else {
// 关闭2d纹理
gl.glDisable(GL10.GL_TEXTURE_2D);
}
// 设置逆时针方向为正面
gl.glFrontFace(GL10.GL_CCW);
// 设置顶点数组
gl.glVertexPointer(3, GL10.GL_FLOAT, 0, mVertexBuffer);
// 绘制
gl.glDrawElements(GL10.GL_TRIANGLE_FAN, vertexCount,
GL10.GL_UNSIGNED_SHORT, mIndexBuffer);
// 关闭2d纹理
gl.glDisable(GL10.GL_TEXTURE_2D);
}
public void draw(GL10 gl, float x, float y, float z, float rot, float scale) {
this.draw(gl, x, y, z, rot, scale, scale);
}
// draw(gl,x,y,z,旋转角度,x方向缩放,y方向缩放)
public void draw(GL10 gl, float x, float y, float z, float rot,
float scaleX, float scaleY) {
gl.glPushMatrix();
gl.glTranslatef(x, y, z);
gl.glRotatef(rot, 0f, 0f, 1f);
gl.glScalef(scaleX, scaleY, 1f);
this.draw(gl);
gl.glPopMatrix();
}
public void draw(GL10 gl, float x, float y, float z, float rot) {
gl.glPushMatrix();
gl.glTranslatef(x, y, z);
gl.glRotatef(rot, 0f, 0f, 1f);
this.draw(gl);
gl.glPopMatrix();
}
public void draw(GL10 gl, float x, float y, float z) {
gl.glPushMatrix();
gl.glTranslatef(x, y, z);
this.draw(gl);
gl.glPopMatrix();
}
}
代码并不多,其实一个DrawObject对象将代表一个物体,通常绘制一个物体,主要包括顶点缓冲区(mVertexBuffer),索引缓冲区(mIndexBuffer),纹理坐标缓冲区(mTexBuffer),同时我们通过vertexCount来记录顶点的个数,hasTexture检测是否有纹理,没有就将使用颜色作为纹理,mTexture数组就表示纹理ID。
其中有两个静态函数makeFloatBuffer和makeShortBuffer用于将数组转换成对应的缓冲区,在Android中使用java来编写Opengl ES程序,在传递顶点等数组时需要使用缓冲区,转换过程很简单,首先构建一个和数组一样的缓冲区,然后检索该缓冲区的字节顺序,最后将数组作为缓冲区即可。
构造函数很简单,将绘制该图形所需要数据传入即可,两个构造函数,其中一个就说明了当没有纹理坐标时,我们就不需要设置纹理坐标缓冲区,直接通过颜色来作为材质即可。
装载纹理需要使用loadTexture函数,首先将hasTexture设置为有纹理存在,glGenTextures函数用于根据纹理参数返回n个纹理名称,用来生成纹理名字的数量、存储纹理名称数组、以及存放在纹理数组中的偏移量。glBindTexture函数实现了将调用glGenTextures函数生成的纹理的名字绑定到对应的目标纹理上,其参数分别是:纹理被绑定的目标(在Opengl ES中它只能取值GL_TEXTURE_2D)和纹理的名称,并且,该纹理的名称在当前的应用中不能被再次使用。然后通过BitmapFactory.decodeResource取得纹理图片资源,然后通过GLUtils.texImage2D函数将纹理图片像素数据绑定到Opengl对象中。GLUtils.texImage2D函数参数的含义如下:
target:指定目标纹理,必须为GL_TEXTURE_2D
level:指定图像级别的编号,0表示基本图像
bitmap:纹理图片数据
border:纹理图像的边框宽度,必须是0或1
如果我们使用glTexImage2D函数,用来指定二维纹理图像,他还有以下几个参数可以使用:
components:纹理中颜色组件的编号,可是是1或2或3或4
width:纹理图像的宽度
height:纹理图像的高度
format:指定像素数据的格式,一共有9个取值:GL_COLOR_INDEX、GL_RED、GL_GREEN、GL_BLUE、GL_ALPHA、GL_RGB、GL_RGBA、GL_BGR_EXT、GL_BGRA_EXT、GL_LUMINANCE、GL_LUMINANCE_ALPHA
type:像素数据的数据类型,取值可以为GL_UNSIGNED_BYTE, GL_BYTE, GL_BITMAP, GL_UNSIGNED_SHORT, GL_SHORT, GL_UNSIGNED_INT, GL_INT, and GL_FLOAT
pixels:内存中像素数据的指针
设置好纹理图像数据之后,我们需要使用recycle来将该图片数据释放掉,因为在opengl es中它将自己保存一份纹理数据。接着需要设置纹理参数,可以使用glTexParameteri函数或者glTexParameterf函数,其参数的含义如下:
arget:目标纹理,必须为GL_TEXTURE_1D或GL_TEXTURE_2D;
pname:用来设置纹理映射过程中像素映射的问题等,取值可以为:GL_TEXTURE_MIN_FILTER、GL_TEXTURE_MAG_FILTER、GL_TEXTURE_WRAP_S、GL_TEXTURE_WRAP_T
param:实际上就是pname的值
另外还可以使用如下代码,实现线形滤波的功能,当纹理映射到图形表面以后,如果因为其它条件的设置导致纹理不能更好地显示的时候,进行过滤,按照指定的方式进行显示,可能会过滤掉显示不正常的纹理像素。
复制到剪贴板 Java代码01.glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR) glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR)
剩下的都是一些绘制函数了,用来根据不同的条件进行不同的渲染。最完整的渲染条件是包括:x,y,z,旋转角度,x方向缩放,y方向缩放。整个过程就是将矩阵压栈(glPushMatrix),然后进行变化,绘制等操作之后,在将矩阵弹出栈(glPopMatrix)。为什么要这样呢?因为我们在变换坐标的时候,使用的是glTranslatef(),glRotaef()等函数来操作,操作的是当前矩阵,这些变化,将会对矩阵进行相乘,使之改变了当前的矩阵,当我们再次使用该矩阵时,就已经被改变而变得很难控制了,所以,我们在进行变换操作之前都需要将将矩阵进行压栈保存起来,操作完成之后,弹出栈的矩阵则和操作前的矩阵一样。下面我们分析一下具体的绘制函数"public void draw(GL10 gl)"。
首先判断是否有纹理,如果有则通过glEnable(GL10.GL_TEXTURE_2D)打开2D纹理,然后通过glBindTexture来绑定纹理,前面我们允许设置了纹理坐标缓冲区,所以这里也需要通过glTexCoordPointer来设置纹理坐标缓冲区;如果没有纹理,则通过glDisable(GL10.GL_TEXTURE_2D)关闭2D纹理映射。然后通过glFrontFace来设置正面,分为逆时针和顺时针,glVertexPointer可以设置顶点缓冲区,使用glDrawElements进行渲染操作,渲染完成之后切忌关闭2D纹理映射。
glDrawElements函数参数定义如下:
mode:指定绘制图元的类型,它应该是下列值之一,GL_POINTS, GL_LINE_STRIP, GL_LINE_LOOP, GL_LINES, GL_TRIANGLE_STRIP, GL_TRIANGLE_FAN, GL_TRIANGLES
count:为绘制图元的数量
type:为索引值的类型,只能是下列值之一:GL_UNSIGNED_BYTE, GL_UNSIGNED_SHORT, or GL_UNSIGNED_INT
indices:索引缓冲区
物理世界(PhysicsWorld)
看这个标题好像很复杂,的确物理世界很复杂,但是我们这
在编写Android程序的时候,我们总是难免会碰到OOM(OUT OF MEMORY)的错误,那么这个错误究竟是怎么来的呢,可以先看一下这篇文章ANDROID BITMAP内存限制OOM,OUT OF MEMORY。
这里,我使用Gallery来举例,在模拟器中,不会出现OOM错误,但是,一旦把程序运行到真机里,图片文件一多,必然会出现OOM,我们通过做一些额外的处理来避免。
1.创建一个图片缓存对象HashMap<Integer,Bitmap> dataCache,integer对应Adapter中的位置position,我们只用缓存处在显示中的图片,对于之外的位置,如果dataCache中有对应的图片,我们需要进行回收内存。在这个例子中,Adapter对象的getView方法首先判断该位置是否有缓存的bitmap,如果没有,则解码图片(bitmapDecoder.getPhotoItem,BitmapDecoder类见后面)并返回bitmap对象,设置dataCache在该位置上的bitmap缓存以便之后使用;若是该位置存在缓存,则直接取出来使用,避免了再一次调用底层的解码图像需要的内存开销。有时为了提高Gallery的更新速度,我们还可以预存储一些位置上的bitmap,比如存储显示区域位置外向上3个向下3个位置的bitmap,这样上或下滚动Gallery时可以加快getView的获取。
public View getView(int position, View convertView, ViewGroup parent) { if(convertView==null){ LayoutInflater inflater = LayoutInflater.from(context); convertView = inflater.inflate(R.layout.photo_item, null); holder = new ViewHolder(); holder.photo = (ImageView) convertView.findViewById(R.id.photo_item_image); holder.photoTitle = (TextView) convertView.findViewById(R.id.photo_item_title); holder.photoDate = (TextView) convertView.findViewById(R.id.photo_item_date); convertView.setTag(holder); }else { holder = (ViewHolder) convertView.getTag(); } cursor.moveToPosition(position); Bitmap current = dateCache.get(position); if(current != null){//如果缓存中已解码该图片,则直接返回缓存中的图片 holder.photo.setImageBitmap(current); }else { current = bitmapDecoder.getPhotoItem(cursor.getString(1), 2) ; holder.photo.setImageBitmap(current); dateCache.put(position, current); } holder.photoTitle.setText(cursor.getString(2)); holder.photoDate.setText(cursor.getString(4)); return convertView; } }
BitmapDecoder.class
package com.wuyi.bestjoy; import java.io.FileNotFoundException; import java.io.FileOutputStream; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Matrix; public class BitmapDecoder { private static final String TAG = "BitmapDecoder"; private Context context; public BitmapDecoder(Context context) { this.context = context; } public Bitmap getPhotoItem(String filepath,int size) { BitmapFactory.Options options = new BitmapFactory.Options(); options.inSampleSize=size; Bitmap bitmap = BitmapFactory.decodeFile(filepath,options); bitmap=Bitmap.createScaledBitmap(bitmap, 180, 251, true);//预先缩放,避免实时缩放,可以提高更新率 return bitmap; } }
2.由于Gallery控件的特点,总有一个item处于当前选择状态,我们利用此时进行dataCache中额外不用的bitmap的清理,来释放内存。
@Override public void onItemSelected(AdapterView<?> parent, View view, int position,long id) { releaseBitmap(); Log.v(TAG, "select id:"+ id); } private void releaseBitmap(){ //在这,我们分别预存储了第一个和最后一个可见位置之外的3个位置的bitmap //即dataCache中始终只缓存了(M=6+Gallery当前可见view的个数)M个bitmap int start = mGallery.getFirstVisiblePosition()-3; int end = mGallery.getLastVisiblePosition()+3; Log.v(TAG, "start:"+ start); Log.v(TAG, "end:"+ end); //释放position<start之外的bitmap资源 Bitmap delBitmap; for(int del=0;del<start;del++){ delBitmap = dateCache.get(del); if(delBitmap != null){ //如果非空则表示有缓存的bitmap,需要清理 Log.v(TAG, "release position:"+ del); //从缓存中移除该del->bitmap的映射 dateCache.remove(del); delBitmap.recycle(); } } freeBitmapFromIndex(end); } /** * 从某一位置开始释放bitmap资源 * @param index */ private void freeBitmapFromIndex(int end) { //释放之外的bitmap资源 Bitmap delBitmap; for(int del =end+1;del<dateCache.size();del++){ delBitmap = dateCache.get(del); if(delBitmap != null){ dateCache.remove(del); delBitmap.recycle(); Log.v(TAG, "release position:"+ del); } } }经过这些额外的操作,有效的避免了OOM的问题。
listview下面,你使用的是哪个方法,促发bitmap的回收事件?
这个cursor是哪来的?不太懂
//弹出窗口大小为300*340,true表示其可以获得焦点 PopupWindow about = new PopupWindow(view, 300, 340, true); //要想实现单击这个PopupWindow窗口外部区域关闭窗口,我们需要设置背景,可以为空,需要注意的是设置背景必须在showAtLocation方法之前 about.setBackgroundDrawable(new ColorDrawable(Color.GRAY)); about.setAnimationStyle(android.R.anim.fade_in); about.showAtLocation(this.findViewById(R.id.root), Gravity.CENTER, 0, 0);