aframe-playground-propio

A playground for learning about A-frame

View the Project on GitHub yamoreno2021/aframe-playground-propio

Back to the main page

Adding some physics

Up to now, we have completely defined the behavior of components, usually as static bodies, or with simple movements. But we can add some physics to the scene, so that components behave as physical bodies: falling with gravity, bouncing when interacting with elastic bodies, behaving to push as if bodies had mass, etc.

For all of this, we can use the aframe-physics-system module, using Ammo.js behind the scenes.

Note: As of A-Frame 1.4.0, the Cannon.js backend is not working with aframe-physics-system. Thus, all the demos in this playground are working with the Ammo.js backend. Check detailed instructions about how to configure aframe-physics-system for working with Ammo.js in the Ammo driver documentation. Have in mind also that the Cannon.js backend may be deprecated in the future, according to aframe-physics-system documentation.

Falling boxes

When using aframe-physics-system, you place bodies in the scene, and let physics control their behaviour. For example, if you place a box in the scene, and there is gravity, it will just fall down until something stops it (it that ever happens).

The basic components introduced by this physics module (using the Ammo.js backend) are:

Let’s try them, by defining a simple scena with a static plane, and two dynamic boxes over it.

First, in the head element of the HTML page, we include the physics module, and the ammo.js module, needed for it to work:

<head>
  ...
  <script src="https://mixedreality.mozilla.org/ammo.js/builds/ammo.wasm.js"></script>
  <script src="http://cdn.jsdelivr.net/gh/n5ro/aframe-physics-system@v4.0.1/dist/aframe-physics-system.min.js"></script>
</head>

Then, let’s define the scene (not the physics property in the scene element, so that the scene is subject to physics):

<a-scene physics="driver: ammo; debug: true; debugDrawMode: 1;">
  <a-entity camera position="0 1.6 5" look-controls wasd-controls></a-entity>
  <a-plane ammo-body="type: static" ammo-shape="type: box"
      position="0 0 -4" rotation="-90 0 0" width="8" height="8"
      color="yellow"></a-plane>
  <a-box ammo-body="type: dynamic" ammo-shape="type: box"
      position="0 4 -2" width="3" height="2" depth="1"
      color="red"></a-box>
  <a-box ammo-body="type: dynamic" ammo-shape="type: box"
      position="2 10 0" width="1" height="1" depth="1"
      color="blue"></a-box>
</a-scene>

The debug: true parameter to physics will produce wires around the bodies subject to physics, which makes it easier to understand the behavior of the scene. The rest is just placing the bodies and defining them as static or dynamic.

Watch this scene in your browser, or check its complete source code

The final result is like this:

Boxes falling

Bodies may have impulse

There are several ways of controlling movement in a world with physics. For example, bodies may receive impulse, and depending on the impulse vector, it will behave by gaining velocity, and maybe impacting other bodies.

To illustrate this, we will use some JavaScript to define a new AFrame component, pùsh, which will emulate “pushing” (giving some impulse) to the body. This will happen whenever the body collides with some other in the scene, and when that happens, it will get a “push”, always in the same direction and with the same intensity (that is, with the same impulse vector).

The new component is defined below:

<script>
  AFRAME.registerComponent('push', {
    init: function() {
          var el = this.el;
          el.addEventListener("collidestart", function () {
            const impulse = new Ammo.btVector3(1.4, 1, 0);
            const pos = new Ammo.btVector3(0, 0, 0);
            el.body.applyImpulse(impulse, pos);
            Ammo.destroy(impulse);
            Ammo.destroy(pos);
      });
    }
  });
</script>

We only needed to define a init function for the component, which sets a handler for the collidestart event, which is fired by the physics system when bodies collide. When this happens, the handler (second argument to addEventListener) will use Ammo primitives to set the impulse.

Note that for the collidestart event to be fired, the property emitCollisionEvents of the body has to be set to true:

  ammo-body="type: dynamic; emitCollisionEvents: true;"

To show this new component, we define a scene with a static plane, to “regular” dynamic boxes falling on it, and one dynamic box (the green one), with the push component, which is pushed every time it collides with the plane:

<a-scene physics="driver: ammo; debug: true; debugDrawMode: 1;">
  <a-entity camera position="0 1.6 5" look-controls wasd-controls></a-entity>
  <a-plane ammo-body="type: static" ammo-shape="type: box"
      position="0 0 -4" rotation="-90 0 0" width="16" height="12"
      color="yellow"></a-plane>
  <a-box ammo-body="type: dynamic" ammo-shape="type: box"
      position="0 4 -2" width="3" height="2" depth="1"
      color="red"></a-box>
  <a-box ammo-body="type: dynamic" ammo-shape="type: box"
      position="2 10 0" width="1" height="1" depth="1"
      color="blue"></a-box>
  <a-entity geometry="primitive: box" push
      ammo-body="type: dynamic; emitCollisionEvents: true;" ammo-shape="type: box"
      position="-3 4 -2" width="1" height="1" depth="1"
      material="color: green"></a-entity>
</a-scene>

The movement is complex, and everytime you launch the scene, it will be different (meaning that small changes to how the blue box falls can cause it to bounce more or less times, thus pushing the red box more or less, in some cases even pushing it out of the plane).

Watch this scene in your browser, or check its complete source code

The final result is like this:

Boxes falling

Bodies with uniform velocity

Now, I will add a sphere with a uniform velocity. I want it to run horizontally, not being affected by gravity. For that, it is enough to make its mass equal to 0. The element will be like this:

<a-sphere explode ammo-body="type: kinematic; emitCollisionEvents: true;"
          ammo-shape="type: sphere"
          position="-6 2 -2" velocity="1 0 0" radius="1"
          color="blue"></a-sphere>

As seen, velocity is expressed using the velocity component. For this component to work, we use kinematic as the type of body. Kinematic bodies are not subject to the physical rules of movement, but to the movement we specify (in this case, via the velocity component). Thus, it will not be affected by gravity (but it will still collide with other bodies).

We also use the explode component, which we defined earlier, in the head element of the HTML page:

AFRAME.registerComponent('explode', {
  init: function() {
    var el = this.el;
    el.addEventListener("collidestart", function () {
      el.parentElement.removeChild(el)
    });
  }
});

This component will add a listener for the collidestart event, which will remove the element from the page.

Finally, we add a new box to the right, so it is more likely that the sphere has something to collide with.

Watch this scene in your browser, or check its complete source code

The final result is like this:

Boxes falling