In past posts in this series, we've focused on the concepts behind the artificial intelligence we've been learning about. In this part, we'll wrap all the implementation into an entirely playable hockey game. You'll learn how to add the missing pieces required to turn this into a game, such as score, power-ups, and a bit of game design.
Final Result
Below is the game that will be implemented using all the elements described in this tutorial.
Thinking Game Design
The previous parts of this series focused on explaining how the game AI works. Each part detailed a particular aspect of the game, like how athletes move and how attack and defense are implemented. They were based on concepts like steering behaviors and stack-based finite state machines.
In order to make a fully playable game, however, all those aspects must be wrapped into a core game mechanic. The most obvious choice would be to implement all the official rules of an official hockey match, but that would require a lot of work and time. Let's take a simpler fantasy approach instead.
All hockey rules will be replaced with a single one: if you are carrying the puck and are touched by an opponent, you freeze and shatter into a million pieces! It will make the game simpler to play and fun for both players: the one carrying the puck and the one trying to recover it.
In order to enhance this mechanic, we'll add a few power-ups. They will help the player to score and make the game a bit more dynamic.
Adding the Ability to Score
Let's begin with the scoring system, responsible for determining who wins or loses. A team scores every time the puck enters the opponent's goal.
The easiest way to implement this is by using two overlapped rectangles:
The green rectangle represents the area occupied by the goal structure (the frame and the net). It works like a solid block, so the puck and the athletes will not be able to move through it; they will bounce back.
The red rectangle represents the "score area". If the puck overlaps this rectangle, it means a team just scored.
The red rectangle is smaller than the green one, and placed in front of it, so if the puck touches the goal on any side but the front, it will bounce back and no score will be added:
Organizing Everything After Someone Scores
After a team scores, all athletes must return to their initial position and the puck must be placed at the rink center again. After this process, the match can continue.
Moving Athletes To Their Initial Position
As explained in the first part of this series, all athletes have an AI state called prepareForMatch
that will move them towards the initial position, and cause them to smoothly come to a stop there.
When the puck overlaps one of the "score areas", any currently active AI state of all athlete is removed and prepareForMatch
is pushed into the brain. Wherever the athletes are, they will return to their initial position after a few seconds:
Moving the Camera Towards the Rink Center
Since the camera always follows the puck, if it is directly teleported to the rink center after someone scores, the current view will abruptly change, which would be ugly and confusing.
A better way to do this is to move the puck smoothly towards the rink center; since the camera follows the puck, this will gracefully slide the view from the goal to the rink center.
This can be achieved by changing the puck's velocity vector after it hits any goal area. The new velocity vector must "push" the puck towards the rink center, so it can be calculated as:
var c :Vector3D = getRinkCenter(); var p :Vector3D = puck.position; var v :Vector3D = c - p; v = normalize(v) * 100; puck.velocity = v;
By subtracting the rink center's position from the puck's current position, it is possible to calculate a vector that points directly towards the rink center.
After normalizing this vector, it can be scaled by any value, like 100
, which controls how fast the puck moves towards the rink center.
Below is an image with a representation of the new velocity vector:
This vector V
is used as the puck's velocity vector, so the puck will move towards the rink center as intended.
To prevent any weird behavior while the puck is moving towards the rink center, such as an interaction with an athlete, the puck is deactivated during the process. As a consequence, it stops interacting with athletes and is marked as invisible. The player will not see the puck moving, but the camera will still follow it.
In order to decide whether the puck is already in position, the distance between it and the rink center is calculated during the movement. If it is less than 10
, for instance, the puck is close enough to be directly placed at the rink center and reactivated so that the match can continue.
Adding Power-Ups
The idea behind power-ups is to help the player achieve the game's primary objective, which is to score by carrying the puck to the opponent's goal.
For the sake of scope, our game will have only two power-ups: Ghost Help and Fear The Puck. The former adds three additional athletes to the player's team for some time, while the latter makes the opponents flee the puck for a few seconds.
Power-ups are added to both teams when anyone scores.
Implementing the "Ghost Help" Power-up
Since all athletes added by the Ghost Help power-up are temporary, the Athlete
class must be modified to allow an athlete to be marked as a "ghost". If an athlete is a ghost, it will remove itself from the game after a few seconds.
Below is the Athlete
class, highlighting only the additions made to accommodate the ghost functionality:
public class Athlete { // (...) private var mGhost :Boolean; // tells if the athlete is a ghost (a powerup that adds new athletes to help steal the puck). private var mGhostCounter :Number; // counts the time a ghost will remain active public function Athlete(thePosX :Number, thePosY :Number, theTotalMass :Number) { // (...) mGhost = false; mGhostCounter = 0; // (...) } public function setGhost(theStatus :Boolean, theDuration :Number) :void { mGhost = theStatus; mGhostCounter = theDuration; } public function amIAGhost() :Boolean { return mGhost; } public function update() :void { // (...) // Update powerup counters and stuff updatePowerups(); // (...) } public function updatePowerups() :void { // TODO. } }
The property mGhost
is a boolean that tells if the athlete is a ghost or not, while mGhostCounter
contains the amount of seconds the athlete should wait before removing himself from the game.
Those two properties are used by the updatePowerups()
method:
private function updatePowerups():void { // If the athlete is a ghost, it has a counter that controls // when it must be removed. if (amIAGhost()) { mGhostCounter -= time_elapsed; if (mGhostCounter <= 2) { // Make athlete flicker when it is about to be removed. flicker(0.5); } if (mGhostCounter <= 0) { // Time to leave this world! (again) kill(); } } }
The updatePowerups()
method, called within the athlete's update()
routine, will handle all power-up processing in the athlete. Right now all it does is check whether the current athlete is a ghost or not. If it is, then the mGhostCounter
property is decremented by the amount of time elapsed since the last update.
When the value of mGhostCounter
reaches zero, it means that the temporary athlete has been active for long enough, so it must remove itself from the game. To make the player aware of that, the athlete will start flickering its last two seconds before disappearing.
Finally, it is time to implement the process of adding the temporary athletes when the power-up is activated. That is performed in the powerupGhostHelp()
method, available in the main game logic:
private function powerupGhostHelp() :void { var aAthlete :Athlete; for (var i:int = 0; i < 3; i++) { // Add the new athlete to the list of athletes aAthlete = addAthlete(RINK_WIDTH / 2, RINK_HEIGHT - 100); // Mark the athlete as a ghost which will be removed after 10 seconds. aAthlete.setGhost(true, 10); } }
This method iterates over a loop that corresponds to the amount of temporary athletes being added. Each new athlete is added to the bottom of the rink and marked as a ghost.
As previously described, ghost athletes will remove themselves from the game.
Implementing the "Fear The Puck" Power-Up
The Fear The Puck power-up makes all opponents flee the puck for a few seconds.
Just like the Ghost Help power-up, the Athlete
class must be modified to accommodate that functionality:
public class Athlete { // (...) private var mFearCounter :Number; // counts the time the athlete should evade from puck (when fear powerup is active). public function Athlete(thePosX :Number, thePosY :Number, theTotalMass :Number) { // (...) mFearCounter = 0; // (...) } public function fearPuck(theDuration: Number = 2) :void { mFearCounter = theDuration; } // Returns true if the mFearCounter has a value and the athlete // is not idle or preparing for a match. private function shouldIEvadeFromPuck() :Boolean { return mFearCounter > 0 && mBrain.getCurrentState() != idle && mBrain.getCurrentState() != prepareForMatch; } private function updatePowerups():void { if(mFearCounter > 0) { mFearCounter -= elapsed_time; } // (...) } public function update() :void { // (...) // Update powerup counters and stuff updatePowerups(); // If the athlete is an AI-controlled opponent if (amIAnAiControlledOpponent()) { // Check if "fear of the puck" power-up is active. // If that's true, evade from puck. if(shouldIEvadeFromPuck()) { evadeFromPuck(); } } // (...) } public function evadeFromPuck() :void { // TODO } }
First the updatePowerups()
method is changed to decrement the mFearCounter
property, which contains the amount of time the athlete should avoid the puck. The mFearCounter
property is changed every time the method fearPuck()
is called.
In the Athlete
's update()
method, a test is added to check if the power-up should take place. If the athlete is an opponent controlled by the AI (amIAnAiControlledOpponent()
returns true
) and the athlete should evade the puck (shouldIEvadeFromPuck()
returns true
as well), the evadeFromPuck()
method is invoked.
The evadeFromPuck()
method uses the evade behavior, which makes an entity avoid any object and its trajectory altogether:
private function evadeFromPuck() :void { mBoid.steering = mBoid.steering + mBoid.evade(getPuck().getBoid()); }
All the evadeFromPuck()
method does is to add an evade force to the current athlete's steering force. It makes him evade the puck without ignoring the already added steering forces, such as the one created by the currently active AI state.
In order to be evadable, the puck must behave like a boid, as all athletes do (more information about that in the first part of the series). As a consequence, a boid property, which contains the puck's current position and velocity, must be added to the Puck
class:
class Puck { // (...) private var mBoid :Boid; // (...) public function update() { // (...) mBoid.update(); } public function getBoid() :Boid { return mBoid; } // (...) }
Finally, we update the main game logic to make opponents fear the puck when the power-up is activated:
private function powerupFearPuck() :void { var i :uint, athletes :Array = rightTeam.members, size :uint = athletes.length; for (i = 0; i < size; i++) { if (athletes[i] != null) { // Make athlete fear the puck for 3 seconds. athletes[i].fearPuck(3); } } }
The method iterates over all opponent athletes (the right team, in this case), calling the fearkPuck()
method of each one of them. This will trigger the logic that makes the athletes fear the puck during a few seconds, as previously explained.
Freezing and Shattering
The last addition to the game is the freezing and shattering part. It is performed in the main game logic, where a routine checks whether the athletes of the left team are overlapping with the athletes of the right team.
This overlapping check is automatically performed by the Flixel game engine, which invokes a callback every time an overlap is found:
private function athletesOverlapped(theLeftAthlete :Athlete, theRightAthlete :Athlete) :void { // Does the puck have an owner? if (mPuck.owner != null) { // Yes, it does. if (mPuck.owner == theLeftAthlete) { //Puck's owner is the left athlete theLeftAthlete.shatter(); mPuck.setOwner(theRightAthlete); } else if (mPuck.owner == theRightAthlete) { //Puck's owner is the right athlete theRightAthlete.shatter(); mPuck.setOwner(theLeftAthlete); } } }
This callback receives as parameters the athletes of each team that overlapped. A test checks if the puck's owner is not null, which means it is being carried by someone.
In that case, the puck's owner is compared to the athletes that just overlapped. If one of them is carrying the puck (so he is the puck's owner), he is shattered and the puck's ownership passes to the other athlete.
The shatter()
method in the Athlete
class will mark the athlete as inactive and place it at the bottom of the rink after a few seconds. It will also emit several particles representing ice pieces, but this topic will be covered in another post.
Conclusion
In this tutorial, we implemented a few elements required to turn our hockey prototype into a fully playable game. I intentionally placed the focus on the concepts behind each of those elements, instead of how to actually implement them in game engine X or Y.
The freeze and shatter approach used for the game might sound too fantastical, but it helps keep the project manageable. Sports rules are very specific, and their implementation can be tricky.
By adding a few screens and some HUD elements, you can create your own full hockey game from this demo!
References
- Rink: Hockey Stadium on GraphicRiver
- Sprites: Hockey Players by Taylor J Glidden
- Icons: Game-Icons by Lorc
- Mouse cursor: Cursor by Iwan Gabovitch
- Instruction keys: Keyboard Pack by Nicolae Berbece
- Crosshair: Crosshairs Pack by Bryan
- SFX/Music: shatter by Michel Baradari, puck hit and cheer by gr8sfx, music by DanoSongs.com