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

The Object Game

The GameObject entity will be used by every element in the gae to be rendered, interact with and updated. This is the how-to manage thing into your game.

Physics

Some basic information must be provided to let an object managed by the game; position, speed, and size are the basical ones.

public class GameObject {
    public double x;
    public double y;
    public double dx;
    public double dy;

    public double width;
    public double height;
}

Those attributes are the minimum ones to move any object on a 2D space.

But as developers are humans, they need more usable identifiers to debug and manage those objects. But the machine needs also some internal identifiers.

Let’s now add some useful things:

public class GameObject {

    private static int index = 0;

    public int id = ++index;
    public String name = "noname_" + id;

    public double x;
    public double y;
    public double dx;
    public double dy;

    public double width;
    public double height;
}

The id is a unique identifier for each GameObject instance. The static index is the internal counter used to create the id value. The name is default initialized with a noname_xx value, where xx is the id value.

And finally as we also need to display those object, and this is mainly the main goal, you need to get color for rendering little things. The priority attribute will be used to sort all the object before rendering, managing a depth level between object at rendering time. The image will be used to renderer an… image with a BufferedImage :) If this image attributes is null, a rectangle of size width x height will be rendered (see the chapter 03 at Render class).

public class GameObject {

    private static int index = 0;

    public int id = ++index;
    public String name = "noname_" + id;

    public double x;
    public double y;
    public double dx;
    public double dy;

    public double width;
    public double height;

    public Color color;
    public int priority;
    public BufferedImage image;
}

Ok now that we know what to renderer, we need to add some moves.

public void update(long dt){
        x+=dx*dt;
        y+=dy*dt;
        }

This is a code corresponding formula to compute position depending on its speed:

p1 = p0 + v0*t

Some other methods are added vor some convenient to define easily some properties.

OK, now we will need to update :

The Game

public class Game {
    ...
    Map<String, GameObject> objects = new HashMap<>();
    ...

    public void add(GameObject go) {
        if (!objects.containsKey(go.name)) {
            objects.put(go.name, go);
            objectsList.add(go);
            renderer.add(go)
        }
    }

    ...

    /**
     * Update all the game mechanism
     */
    private void update(long dt) {
        for (GameObject e : objectsList) {
            e.update(dt);
        }
    }
}

A new add(GameObject) method will add the GameObject to the Game objects list and a map. The list and the Map are shortcut to easily manage the object into the Game.

The update method has been modified to update all the objects.

The update(long) method will perform a call to all the GameObject.update(long) method.

Render

The Render class is now to be updated to draw all those objects.

Let’s add some methods.

We need to maintain a list of object to be rendered, and a method to add one.

public class Render {
    ...

    public Render add(GameObject go) {
        if (!objects.contains(go)) {
            objects.add(go);
            objects.sort((a, b) -> {
                return a.priority < b.priority ? -1 : 1;
            });
        }
        return this;
    }
    ...

    private void draw(Graphics2D g, GameObject go) {
        g.setColor(go.color);
        if (go.image != null) {
            g.drawImage(go.image, (int) (go.x), (int) (go.y), null);
        } else {
            g.drawRect((int) (go.x), (int) (go.y), (int) (go.width), (int) (go.height));
        }
    }
}

The add() will add a GameObject to the “rendering list”, and the draw() method will renderer a corresponding image (if not null) or draw a rectangle.

So now, calling the renderer() method will parse in the priority sort order all the GameObject in the objects list, and draw each of those items.

TIPS
The GameObject rendering is test by the Render_draw_GameObject.feature Feature file.