fromClassToGame

A journey from a simple class to a game with the Java language https://mcgivrer.github.io/fromClassToGame/ , and open Hub https://www.openhub.net/p/fromClassToGame.

View the Project on GitHub mcgivrer/fromClassToGame

A Basic Physic Engine

The BPE[^1] is very basic one using only position and speed to move object. The mechanic formula we will use in the computation of object moves will take only in account the velocity and th e current positio nof an oibject to compute its own next position.

p_1 = p_0 + v_0 * t

where:

But to be able to correctly simulate the move behavior, we will add 2 new parameters: friction and bouncyness.

But those factor will be applied only if there is some collision or point of contact between 2 objects.

p_1 = p_0 + (v_0 + g) * b * f * t

Where:

and to get a real break effect on the object according to the friction, we will recompute the velocity with this attenuation factor:

v_1 = v_0 * b * f

here we see the following variables:

And now the next p1 position will be computed as bellow :

p_1 = p_0 + (v_1 + g) * t

the g gravity factor is a force.

In a next chapter we will enhanced this physic and mechnanic computation engine with a new bundle of forces that can be added at any time to a GameObject.

Some C0D3 ?

The implementation of such engine can be dne throught mulitple solution. We will use one that is very simple.

First we are going to define a World where our GameObject will freely evolve.

And to our existing GameIObject we are going to add the new attributes we discovered through our math formula, but not directly, we will add a new dimention to our design: a Material.

This class will provide all information about physc and mechanic characteristics our engine will need to compute next position.

first thing first, the World object.

World

the World class will provide some basic information about the world our object will moved on: gravity is our first one.

public class World{
    private double gravity;

}

Material

The Material Object will contains, as seen before, the all attributes needed by our engine to compute next object position.

public class Material{
    private String name;
    private double bouncyness;
    private double friction;

}

We need a name for this material, and as the formula gives us the characteristics, wee need to add bouncyness and friction attribtues.

Ok, we have our needed info. let’s dive into computation:

our GameObject will be updated iwth this new material attribute:

public class GameObject {

    //...
    public double mass = 0.0;
    public double contact = 0.0;
    public Material material;
    //...
}

And the World object will copute the GameObject postionaccording to wll those parameters:

public class PhysicEngine {
    public Game game;
    public World world;
    public List<GameObject> objects = new ArrayList<>();

    public PhysicEngine(Game g) {
        this.game = g;
    }

    public void update(GameObject go, double dt) {
        go.dx = go.dx * (go.contact * go.material.
                friction * go.material.bouncyness);
        go.x += go.x * (go.dx + this.gravity) * dt;
    }

    public void add(GameObject go) {
        this.objects.add(go);
    }
}

Enhancing the simulation

To get a better simulation, we will need to distinguish 2 kind of friction:

So our Material will provide such new attributes in place of the old friction :

public class Material{
    private String name;
    private double bouncyness;
    private double staticFriction;
    private double dynFriction;
}

and the World update method formula will be updated with that:

PhysicEngine update

So now, the PhysicEngine#update(double) must be updated was following:

public class PhysicEngine {
    //...

    public void update(double dt) {
        if (!go.relativeToCamera) {
            go.dx = go.dx * go.material.staticFriction;
            go.dy = (go.dy + go.gravity + (world.gravity * 0.11)) * go.material.staticFriction * 1 / go.mass;

            go.x += go.dx * dt;
            go.y += go.dy * dt;
        }
    }
    //...
}

The Default material

Materials are so difficult to create, we need some references. According to some web reference, we can find some and use them as is.

But for our own usage, we are going to use the following table:

Description Density Restitution
Rock 0.6 0.1
Wood 0.3 0.2
Metal 1.2 0.05
BouncyBall 0.3 0.8
SuperBall 0.3 0.95
Pillow 0.1 0.2
Static 0.0 0.0

So to generate the Material, we will use an Enumeration:

public enum DefaultMaterial {
    ROCK(new Material("rock", 0.6, 1, 1, 1)),
    WOOD(new Material("wood", 0.1, 0.69, 0.69, 0.3)),
    METAL(new Material("metal", 0.05, 1, 1, 1.2)),
    RUBBER(new Material("rubber", 0.8, 1, 1, 0.3)),
    GLASS(new Material("glass", 0.4, 1, 1, 1)),
    ICE(new Material("ice", 0.1, 0.1, 1, 1)),
    AIR(new Material("air", 1, 1, 1, 1));

    private Material material;

    DefaultMaterial(Material m) {
        this.material = m;
    }

    public Material getMaterial() {
        return this.material;
    }
}

So to get one of the default Material:

class MyScene {
  //...
  public void create(Game g) {
    // creating a GameObject
    GameObject player = new GameObject("player", 160, 100)
      .setType(GameObject.GOType.IMAGE)
      .setColor(Color.RED)
      .setImage(
        ResourceManager
          .getImage("images/tiles.png:player"))
      // setting the material
      .setMaterial(DefaultMaterial.WOOD.getMaterial());
    add(player);
  }
  //...
}

So the last line is getting one of the preset material. Here is another usage:

  Material wood = DefaultMaterial.WOOD.getMaterial();

And that’s all !