代码改变世界

【AwayPhysics学习笔记】:Away3D物理引擎的简介与使用

2015-01-06 15:25 阿诚de窝 阅读(...) 评论(...) 编辑 收藏

首先我们要了解的是AwayPhysics这个物理引擎并不是重头开始写的新物理引擎,而是使用Flascc技术把一个已经很成熟的Bullet物理引擎引入到了Flash中,同时为了让as3可以使用这个C++开发的物理引擎,AwayPhysics库编写了一些必要的AS3类使我们可以方便的使用它。

而为了方便区分和使用AwayPhysics里的所有类都使用了AWP作为类名前缀。

 

创建物理世界

要模拟一个物理世界,首先需要创建一个AWPDynamicsWorld的对象,该对象类似于Away3D中创建3D世界的View3D对象,同时每帧也需要类似View3D对象调用render方法一样调用AWPDynamicsWorld对象的step方法;

创建方法:

1 //获取物理世界
2 _physicsWorld = AWPDynamicsWorld.getInstance();
3 //初始化, 主要包括初始化重力等
4 _physicsWorld.initWithDbvtBroadphase();

step方法一般使用下面的参数:

1 var _timeStep:Number = 1 / 60;//60表示帧率
2 _physicsWorld.step(_timeStep, 1, _timeStep);

关于该方法的更多信息可以参考这里

而正常来说应该是先调用step方法模拟了物理世界之后在调用View3D的render方法来渲染3D世界;

该类有添加和移除各种刚体的方法,可以方便的添加需要使用物理引擎模拟的对象;

 

输出调试图形

如果我们需要查看调试的物件信息(可以查看到添加到物理世界的刚体的坐标系和形状),可以使用AWPDebugDraw类:

1 private var _debugDraw:AWPDebugDraw;
2 
3 //构造函数中创建
4 _debugDraw = new AWPDebugDraw(_view, _physicsWorld);
5 _debugDraw.debugMode |= AWPDebugDraw.DBG_DrawTransform;
6 
7 //帧事件方法中调用绘制
8 _debugDraw.debugDrawWorld();

需要注意的一点是,debugDrawWorld这个方法会严重拉低运行的帧率,当然作为调试工具,我们不需要高效的运行效率,需要调试时别忘了这个类哦。

 

基础使用简介

我们搭建好了物理世界后就可以向物理世界中添加对象了:

  • 首先,AwayPhysics提供了可以用来进行碰撞检测的多种3D形状,存放在awayphysics.collision.shapes包中,我们可以根据需要创建适合的碰撞形状;
  • 然后,我们需要添加一个刚体(AWPRigidBody),将我们的碰撞形状和3D显示对象进行绑定;
  • 最后,我们可以调整刚体的各种属性(如果调整刚体的坐标旋转等信息会实时同步到该刚体绑定的3D对象中),添加到物理世界中即可,当我们的刚体添加到物理世界后他的格子状态就都由物理引擎来模拟了;

示例代码:

 1 //创建地面 Mesh
 2 var material:ColorMaterial = new ColorMaterial(0x252525);
 3 material.lightPicker = lightPicker;
 4 var mesh:Mesh = new Mesh(new PlaneGeometry(50000, 50000), material);
 5 mesh.mouseEnabled = true;
 6 mesh.addEventListener(MouseEvent3D.MOUSE_UP, onMouseUp);
 7 _view.scene.addChild(mesh);
 8 
 9 //创建地面形状和刚体
10 var groundShape:AWPStaticPlaneShape = new AWPStaticPlaneShape(new Vector3D(0, 1, 0));
11 var groundRigidbody:AWPRigidBody = new AWPRigidBody(groundShape, mesh, 0);
12 //添加到物理世界
13 _physicsWorld.addRigidBody(groundRigidbody);

 

碰撞检测

要接收碰撞检测事件需要开启一下标志,这样就可以接收事件了:

1 //开启碰撞侦听, AWPEvent.COLLISION_ADDED 事件被启用
2 _physicsWorld.collisionCallbackOn = true;

碰撞对象:

要使用碰撞检测我们就不用刚体对象(AWPRigidBody)了,要使用碰撞对象(AWPCollisionObject);

同时添加到物理世界时也要使用addCollisionObject方法;

碰撞对象可以添加多个射线,用于射线碰撞检测;

碰撞事件:

碰撞事件都被封装好了,有两个AWPEvent.COLLISION_ADDED和AWPEvent.RAY_CAST:

  • COLLISION_ADDED:当有另一个碰撞物体和自己发生碰撞时会抛出该事件,注意并没有COLLISION_REMOVED等表示碰撞结束的事件,这是由于每次调用step方法模拟时都会重新运算碰撞,所以如果存在碰撞就会一直抛出该事件。
  • RAY_CAST:如果添加过射线,那么射线和其它碰撞对象碰撞时也会受到该事件。

 

游戏应用

我们通过上面的代码可以模拟真实的物理世界,但这是不够的,更多的时候我们需要使用物理引擎来帮助我们实现游戏中的逻辑,而常见的需求是赛车游戏(车辆和地面与其它物体的碰撞)和全3D的RPG游戏(人物和高低不平的地面的碰撞)。

赛车: 

AwayPhysics提供了两个关于赛车的Demo,点击查看源码:Demo1Demo2

由于AwayPhysics已经帮我们封装好了专门用来控制车辆的类,所以创建一个小赛车的游戏是十分简单的,下面说说会使用到的类:

车辆控制类:

  • AWPCompoundShape:这个类可以添加多个基础的形状来达到组合成一个复杂形状的目的,小车使用该类组合了两个长方体;
  • AWPVehicleTuning:这个类用来设置车辆的调节属性;
  • AWPRaycastVehicle:将AWPVehicleTuning和小车刚体进行绑定的类,并且最终会调用_physicsWorld.addVehicle方法添加到物理世界中;
  • AWPRaycastVehicle.addWheel:该方法可以对指定的轮子对象进行绑定;
  • AWPRaycastVehicle.getWheelInfo:该方法可以微调每个轮子的属性;
  • AWPRaycastVehicle.applyEngineForce:该方法可以控制指定的轮子的引擎力,即踩油门,设置为负值可以倒车;
  • AWPRaycastVehicle.setBrake:该方法可以控制指定的轮子的刹车;
  • AWPRaycastVehicle.setSteeringValue:该方法可以控制指定的轮子的转向,即转弯;

地形生成类(直接加载模型作为地形):

  • AWPBvhTriangleMeshShape:我们将加载好的Mesh的geometry属性赋值到该对象后,就创建了一个和Mesh对象一致的刚体形状,再将该对象添加到刚体中即可作为自定义的地形使用。

地形生成类(使用高度图生成地形):

  • AWPTerrain:该对象是一个可视对象,是由一个高度位图创建的Mesh对象。
  • AWPHeightfieldTerrainShape:将我们的AWPTerrain对象赋值到该对象后,就创建了一个和Mesh对象一致的刚体形状,再将该对象添加到刚体中即可作为自定义的地形使用。

RPG:

AwayPhysics提供了一个关于行走的Demo,点击查看源码:Demo

地形生成同上,我们看看不一样的地方:

  • AWPGhostObject:该类用来连接一个形状对象和一个显示对象;
  • AWPKinematicCharacterController:角色控制类,可以设置一个AWPGhostObject对象作为其控制的对象;
  • AWPKinematicCharacterController.setWalkDirection:该方法接收一个3D向量作为移动的距离;
  • AWPKinematicCharacterController.jump:该方法可以用来进行跳跃;

 

补充

除了我们谈到的几点外,大家还可以查看示例程序了解更多的使用技巧:

刚体碰撞过滤测试

重力测试

自定义形状的刚体测试

刚体约束测试