My first physics project using Box2DFlashAC3 library.Box2DFlashAC3 is an open source physics library by Colin Northway. Box2DFlashAC3 is an Actionscript port of Erin Catto's c++ physics library Box2D.
Source Code:
helloWorld.as
helloWorld.fla
Box2DFlashAC3:
SourceForge download >>
Step by Step:
1. Prepare the project:
Create a project folder "helloWorld"
Download Box2DFlashAC3 and extract Box2D folder into flash class folder (FLASH_INSTALL_FOLDER\en\Configuration\ActionScript 3.0\Classes\Box2D)
or copy Box2D folder into current project folder (helloWorld/Box2D)
Create a scene file "helloWorld.fla"
In property editor, set the main class name to helloWorld. This tells flash compiler to look for Actionscript file "helloWorld.as" in the same folder.
Create an Actionscript file "helloWorld.as, and start editing it
2. Create a package, and a helloWorld class with constructor function hellowWorld():
package {
// Built-in flash modules
import flash.display.Sprite;
import flash.text.TextField;
import flash.events.Event;
import flash.events.MouseEvent;
// Box2D modules
import Box2D.Dynamics.*;
import Box2D.Collision.*;
import Box2D.Collision.Shapes.*;
import Box2D.Common.Math.*;
public class helloWorld extends Sprite {
public var m_world:b2World; // world object
public var m_iterations:int = 10; // iterations for verlet intergration
public var m_timeStep:Number = 1.0/30.0; // time step
public function helloWorld() {
// Implement in step 3
}
public function Update(e:Event):void {
// Implement in step
}
public function Reset(e:Event):void {
// Implement in step
}
}
}
3.1 Start to implement helloWorld(), start with printing a Hello World Message:
var myText:TextField = new TextField(); myText.text = "Hello World - Click to reset particle position"; myText.x = 10; myText.y = 10; myText.width = 400; myText.height = 20; addChild(myText);
3.2 Construct a world object
// Create bounding box (AABB) var worldAABB:b2AABB = new b2AABB(); worldAABB.lowerBound.Set(-1000.0, -1000.0); worldAABB.upperBound.Set(1000.0, 1000.0); // Define gravity var gravity:b2Vec2 = new b2Vec2(0.0, 10.0); // Allow bodies to sleep when rest var doSleep:Boolean = true; // Create world object m_world = new b2World(worldAABB, gravity, doSleep);
3.3 Define variables to create bodies or body definitions:
var body:b2Body; // for rigid body var bodyDef:b2BodyDef; // for body definition (position, density...) var boxDef:b2PolygonDef; // to define box collision shape var circleDef:b2CircleDef; // to define circle collision shape
3.4 Create ground static rigid body:
// Create ground body definition bodyDef = new b2BodyDef(); bodyDef.position.Set(0, 11); // Create box shape boxDef = new b2PolygonDef(); boxDef.SetAsBox(30, 3); boxDef.friction = 0.3; boxDef.density = 0; // static bodies require zero density // Attach ground sprite to body definition bodyDef.userData = new PhysGround(); bodyDef.userData.width = 30 * 2 * 30; bodyDef.userData.height = 30 * 2 * 3; addChild(bodyDef.userData); // Create ground body body = m_world.CreateBody(bodyDef); body.CreateShape(boxDef); // Set body mass body.SetMassFromShapes();
3.5 Iterate 100 dynamic bodies:
for (var i:int = 1; i < 100; i++) {
// Implement step 8~9
}
3.6 Generic properties for dynamic rigid bodies:
// Create dynamic body definition bodyDef = new b2BodyDef(); bodyDef.position.x = Math.random() * 13; bodyDef.position.y = Math.random() * 5; // Set random x/y for later var rX:Number = Math.random() * 0.4 + 0.3; var rY:Number = Math.random() * 0.4 + 0.3;
3.7 50% chance to create a box:
// If we are creating a box
if (Math.random() < 0.5) {
// Create shape definition
boxDef = new b2PolygonDef();
boxDef.SetAsBox(rX, rY);
boxDef.density = 1.0;
boxDef.friction = 0.5;
boxDef.restitution = 0.2;
// Attach library item to body definition
bodyDef.userData = new PhysBox();
bodyDef.userData.width = rX * 2 * 30;
bodyDef.userData.height = rY * 2 * 30;
// Create box body
body = m_world.CreateBody(bodyDef);
body.CreateShape(boxDef);
}
3.8 Another 50% we create a circle:
// Else we create a circle
else {
// Create circle shape definition
circleDef = new b2CircleDef();
circleDef.radius = rX;
circleDef.density = 1.0;
circleDef.friction = 0.5;
circleDef.restitution = 0.2
// Attach library graphic to body definition
bodyDef.userData = new PhysCircle();
bodyDef.userData.width = rX * 2 * 30;
bodyDef.userData.height = rX * 2 * 30;
// Create circle body
body = m_world.CreateBody(bodyDef);
body.CreateShape(circleDef);
}
3.9 Finishing dynamic bodies creation
// Set body mass body.SetMassFromShapes(); // Add user data to stage addChild(bodyDef.userData);
4. Implement Update()
public function Update(e:Event):void {
// Update current step
// Advent solver to next step
m_world.Step(m_timeStep, m_iterations);
// Go through body list and update sprite positions/rotations
for (var bb:b2Body = m_world.m_bodyList; bb; bb = bb.m_next){
if (bb.m_userData is Sprite){
bb.m_userData.x = bb.GetPosition().x * 30;
bb.m_userData.y = bb.GetPosition().y * 30;
bb.m_userData.rotation = bb.GetAngle() * (180/Math.PI);
}
}
}
5. Implement Reset()
public function Reset(e:Event):void {
// Reset body positions
// Go through body list and update sprite positions/rotations
for (var bb:b2Body = m_world.m_bodyList; bb; bb = bb.m_next){
if (bb.m_userData is Sprite && bb.IsDynamic()){
var newPos = new b2Vec2( Math.random() * 13,
Math.random() * 5 );
bb.SetXForm(newPos, 0);
}
}
}
--
Updated class installation path.
ReplyDelete