Headstart 2018 - Project
Headstart 2018 - Project
Headstart 2018 - Project
Contents
Project introduction .......................................................................... 2
Build a Basic Platform Game ............................................................. 3
Setting up our desktop .................................................................. 3
Files Structure ........................................................................... 4
HTML web page ........................................................................ 4
Aim of the
JavaScript file ............................................................................ 5
booklet
Run the game ............................................................................ 5
In this booklet, you
Development mode (Chrome Dev Tools) ................................. 6
will find a tutorial on
You will be working in a group of three or more, and your job is to develop your own
platform game. You will have to define the look and feel of the game, what are the
Head Start 2018
A lot of research on how things can be done will be done by you, with myself and the
ambassadors helping you when you are stuck. This first booklet is to get you started.
2
Build a Basic Platform Game
3
3. We need a web browser to run the game on a web page. We will be using
Google Chrome and use the Dev Tool provided to check for errors in our code.
Now we are ready to start.
Files Structure
For a game, we need to organize our files into a directory tree. Below is a template
for such a directory tree. The folder assets contains all the element needed for the
game, such as images for the background, sound effects and music. These elements
are themselves sorted into sub-folders. The JavaScript files are stored in the js folder.
game
|---assets
| |---audio
| |---data
| |---image
| |---sprites
|
|---js
To launch our game we need to embed our JavaScript into an HTML file. At template
is shown below.
<!DOCTYPE html>
<html lang="en-UK">
<head>
<meta charset="utf-8">
<script src="../node_modules/phaser-ce/build/phaser.min.js"></script>
<title> Game </title>
</head>
<body>
<!-- include the main game file -->
<script src="js/index.js"></script>
</body>
</html>
The line <script src="../node_modules/phaser-ce/build/phaser.min.js"></script>
indicates where the Phaser code can be found (note it is also in JavaScript). The path
may vary depending on your setup.
4 Your game will be written in the JavaScript file index.js and it should be in the sub-
folder js. The line <script src="js/index.js"></script> indicates where the file is and
that is should be executed.
JavaScript file
Open a new file in the editor, copy the code below and save it as index.js into the
sub-folder js. Write the following code and save the file.
5
Development mode (Chrome Dev Tools)
Use Ctrl + Shift + I to open the DevTools. The browser should look like the image below.
We have access to the source code, a debugger and a console. This will help us to
debug our code when we encounter an issue.
Head Start 2018
• Boot
• Preload
• GameTitle
• Main
• GameOver
The Boot state handles any set up that is
required (like setting the dimensions for the
game) and calls the next state. Preload
handles loading in any assets that are required
and then launches the GameTitle state.
GameTitle displays the title screen for the
application, where a player can typically start
the game, see instructions, or access the
7
Create the main state
To start with, we will be focusing on the main state. The first line of code declare a
variable mainState that will contain all the properties of a state. Finally, the last
line of code adds the state to the game.
var mainState = {
preload: function(){
},
create: function(){
},
update: function(){
}
};
game.state.add('main', mainState);
You may have noticed that three functions have been declared:
Head Start 2018
• The preload method is triggered first and is typically used to load any assets
that are required (like sprites and sound effects).
• The create method is called after preload has finished, it runs once and is
used to do the initial set up for the game.
• The update method starts after create has finished, and then it is
continuously called as part of the “game loop”. This method is where most of
the fun happens, as it can be used to make decisions based on the current
state of the game – is the player touching an enemy, is there less than 10
seconds remaining, are there multiple enemies on screen?
8
Setting the décor
We need to write a function to be able to draw rectangle that will represents
platforms and walls. Later, we will be using images to create the décor. Copy the code
below at the top of the file. Don’t worry if you don’t understand the code it is not
important for the rest of the session.
To build the décor, we need to modify the create function. Walls and platforms are
built using a rectangle (box). We will be using six walls, four for the border of the world
and two for the platforms in the middle. For convenience, we are adding all the walls
into a group called walls.
this.walls = this.game.add.group();
9
We could also change the colour background by adding the following statement at the
top of the create function.
this.game.stage.backgroundColor = '#C2C2C2';
create: function(){
this.game.stage.backgroundColor = '#C2C2C2';
//…
Head Start 2018
Reload the web page and you should see the game
world on the right. Now that we have a player, we want
to be able to interact with it using the keyboard. Add
the following statement at the end of the create
function.
this.cursor = this.game.input.keyboard.createCursorKeys();
this.cursor will collect the keyboard input from the user, however if we reload
the page, nothing happens when we press a key. This is because we haven’t told the
program what it should do when a key is pressed. We want the player to go right when
we press the right arrow key, and left when we press the left arrow key.
10
We need to use a new JavaScript construct: if-else statement.
If(condition){
doTheseCommands;
} else {
doTheOtherCommands;
}
What does the new construct do? condition is a Boolean (i.e. can have only the
values true or false).
If(conditionA){
doCommandsA;
} else If(conditionB){
doCommandsB;
update: function(){
if(this.cursor.left.isDown){
console.log("go left");
} else if (this.cursor.right.isDown){
console.log("go right");
}
}
11
Reload the game and press the left and/or right arrows. The player does not move,
however when we look at the console, we can see the following:
You can see that the messages “go right” and “go left” have been printed on the
console multiple times. So, what happened exactly?
this.cursor.left.isDown is true when the left arrow key is down, therefore
the code executes the code console.log("go left"). When the left arrow key
is not down, this.cursor.left.isDown is false, therefore the code go to the
next else if statement and tests if this.cursor.right.isDown is true. If
the right arrow is pressed down, the code executes console.log("go right").
If none of the two arrow keys are pressed down, nothing happens.
We are now able to read the input from the user and do something, but so far our
Head Start 2018
player does not move. The player sprite needs a way to tell it that it must move one
way or the other, for that we need to give it a velocity. The velocity is described as a
2D vector, with a x and y value. A positive x value means moving to the right, whereas
a negative value means going left. Similarly, a positive y value means going up, a
negative value means going down. So how can we attach a velocity to a sprite? We
can do that by changing the value of a property of its body, in the case of our player
by doing:
this.player.body.velocity.x = 250;
Let’s modify the update function as follow and reload the game.
update: function(){
var speed = 250;
this.player.body.velocity.y = 0;
this.player.body.velocity.x = 0;
if(this.cursor.left.isDown){
this.player.body.velocity.x -= speed;
} else if (this.cursor.right.isDown){
this.player.body.velocity.x += speed;
}
}
12
The error comes from line 36 in the file index.js (note it may be a different line number
for your program). From this it seems that he cannot find the property velocity,
create: function(){
this.game.stage.backgroundColor = '#C2C2C2';
this.game.world.enableBody = true;
…
}
Let’s reload the game and try to move the player. As you can see we can move the
player from left to right. Could you add more code so the player can also go up and
down?
13
Did you notice something?
Our player moves out of this world! You see that testing thoroughly your code is
essential for a successful project. So what is the issue? We forgot to tell the player
sprite it should interact with the world boundary. We can do so by adding some code
to the create function.
create: function(){
this.game.stage.backgroundColor = '#C2C2C2';
//…
this.player = this.game.add.sprite(32, 32, box({width: 32, height: 32, color: '#4F616E'}));
this.player.body.collideWorldBounds = true;
this.cursor = this.game.input.keyboard.createCursorKeys();
},
We are almost there, we can move the player, it does not go out of bounds but
Head Start 2018
annoyingly it goes through the walls. We need to add physical behaviors, such as
dealing with collision between objects. Let’s tell our world it should have physics
constraints and that walls have a body, so we can interact with them. Add to the
create function the statement below:
create: function(){
this.game.stage.backgroundColor = '#C2C2C2';
this.game.physics.startSystem(Phaser.Physics.ARCADE);
this.game.world.enableBody = true;
this.walls = this.game.add.group();
this.walls.enableBody = true;
…
}
We must also tell the program that it should check collision between the player and
the walls by adding a statement to the update function.
update: function(){
this.game.physics.arcade.collide(this.player, this.walls);
var speed = 250;
…
}
14
Interestingly, the behavior of our game is not exactly the one we expected. Our player
does not go through wall (that’s good) but the walls are moving when our player hits
them. Is this normal? Yes, it is from the point of view of the physical world. The walls
behave like a ball on the floor when you bump into it, the ball moves. You must state
explicitly that the walls cannot move (at least for the game we are creating today). For
every single wall declared in the create function add the following statement:
nameOfTheWall.body.immovable = true;
For example:
topWall.body.immovable = true;
Congratulation, we now have a player that can interact with its environment.
Our world now looks like this, with the player at the top left
corner and the goal to reach at the bottom right in green.
15
Have we reach the level’s goal?
Now we need to check constantly if the player overlaps with the goal, so we need to
add a statement in the update function. At the beginning of the function update add
the statement:
var mainState = {
create: function(){
…
},
update: function(){
…
Head Start 2018
},
handlePlayerWin: function(sprite1, sprite2){
sprite1.kill();
this.game.state.start('main');
}
};
Let reload the game and play. As you can see, as soon as our player passes over the
goal, a new game restart.
So, what does the code says? Because handlePlayerWin is the callback function
from overlap, sprite1 will received the object that is the first parameter in the
overlap function (in our case player), and sprite2 receives the second
parameter (in our case goal).
var mainState = {
…
};
var winningState = {
create: function(data){
},
update: function(){
Now let’s modify the handlePlayerWin function to call the new state when the
player overlaps the goal.
create: function(data){
label = this.game.add.text(this.game.world.width/2, this.game.world.height/2,
'Congratulation \n YOU HAVE WON! \n Press SPACE to restart.',
{
font: '22px Arial',
fill: '#b00',
align: 'center'
});
}
Head Start 2018
When we execute the code, and the player win the game we get to the new screen.
label.anchor.setTo(0.5, 0.5);
Next, we need to say that we want to listen to the keyboard input, especially the space
bar. At the end of the create function add the statement:
this.spacebar = this.game.input.keyboard.addKey(Phaser.Keyboard.SPACEBAR);
18
Step 2
Finally, we want to restart the game if and when the space bar is pressed. Like for the
arrow keys, we need to deal with the key pressed in the update function of the state
winningState.
update: function(){
if(this.spacebar.isDown){
this.game.state.start('main');
}
}
Congratulation, we have a completed game. It will not win any award, but that is a
start.
Add Enemies
The least we can say is that our game is far from being interesting or captivating. One
thing that is missing are the villains of the game, i.e. something that could kill us.
var mainState = {
create: function(){
this.game.stage.backgroundColor = '#C2C2C2';
…
this.goal = this.game.add.sprite(this.game.world.width - 80, this.game.world.height - 50,
box({width: 32, height: 32, color: '#50FF50'}));
this.enemy = this.game.add.sprite(this.game.world.width - 200, 130,
box({width: 32, height: 32, color: '#FF5050'}));
},
update: function(){
this.game.physics.arcade.collide(this.player, this.walls);
this.game.physics.arcade.overlap(this.player, this.enemy, this.handlePlayerDeath, null, this);
this.game.physics.arcade.overlap(this.player, this.goal, this.handlePlayerWin, null, this);
…
},
handlePlayerDeath: function(player, enemy){
player.kill();
this.game.state.start('main');
}, 19
…
Rather than restarting the game by calling this.game.state.start('main'),
create a new game state to show a game over screen (in the same way as the
winningState).
Animate an Enemy
A sprite has predefine properties such as its x and y coordinates. They can be accessed
via the name of the sprite object and the name of the property, e.g.
spriteName.propertyName. For example, to get the x coordinate of our enemy we can
use this.enemy.x. In addition to existing properties, we can add our own. For
example we want our enemy to patrol a certain area of the game, e.g the enemy walks
some distance in one direction and then turn around and walks the same distance,
and then turn around and so on. We define a property maxDistance representing
the length of the patrolled area. We must also define a property storing the x position
where the enemy turned around, let’s call it previousX.
create: function(){
…
this.enemy = this.game.add.sprite( this.game.world.width - 200, 130,
box({width: 32, height: 32, color: '#FF5050'}));
Head Start 2018
this.enemy.previousX = this.enemy.x;
this.enemy.maxDistance = 250;
this.enemy.body.velocity.x = -100;
}
The next step is to check the current position of the enemy to see if it should turn
around or not. This is done in the update function and his shown below.
update: function(){
this.game.physics.arcade.overlap(this.player, this.enemy, this.handlePlayerDeath, null, this);
Reload the game and see what happens. Add more enemies to the game, for example
one on each level to have a total of three enemies. Use different values for each
enemy.
20
Final Touches: (stage 6)
GRAVITY!
The idea of a platform game is to have a player being have to jump to and fall from
walls and platform. At this stage our game is not able to do that, we are lacking an
important physics property: GRAVITY!
Gravity is simply implemented by the physics engine using the statement:
create: function(){
this.game.physics.startSystem(Phaser.Physics.ARCADE);
this.game.physics.arcade.gravity.y = 900;
…
}
Reload the game.
The problem here is that every object with a body falls due to gravity (which would be
21
Next step we want the player to jump, we need to modify the code we wrote for the
user input in the update function.
update: function(){
this.game.physics.arcade.collide(this.player, this.walls);
this.game.physics.arcade.collide(this.enemies, this.walls);
this.game.physics.arcade.overlap(this.player, this.enemies, this.handlePlayerDeath, null, this);
this.game.physics.arcade.overlap(this.player, this.goal, this.handlePlayerWin, null, this);
},
Killing Enemies
The final touch to our game is to be able to kill enemies. We have decided that the
player must jump on the enemy to kill it, in other word the player y velocity should
be greater than 0 when it touches the enemy, therefore falling onto them. To do that
we need to modify the callback function handlePlayerDeath.
One important thing when writing code is to make it easier for someone else to read
and understand your code. For a code readability purpose, we may want to rename
22 the function handlePlayerDeath to something like:
handlePlayerEnemiesCollision
A more advanced Platform Game
Setup
For this session, you will need the following files which can be found in the
assets/images and assets/audio folders.
• background.png,
• grass_8x1.png,
• grass_6x1.png,
• grass_4x1.png,
• grass_2x1.png,
• grass_1x1.png,
• hero_stopped.png,
• jump.wav.
In headstart, create a new folder (for example GameSession2) to contain the code for
this session. In the same way as yesterday, create the needed folders, a new
index.html file and a new index.js.
Let start to write our basic template for the JavaScript file:
var mainState = {
preload : function(){
},
create: function(){
},
update: function(){
} 23
};
game.state.add('main', mainState);
game.state.start('main');
In order to make it easier to modify the parameter of our game, let’s create an new
variable containing all the option such as the value for gravity, player velocity, and so
on.
var gameOptions = {
gameWidth: 960,
gameHeight: 600,
bgColor: 0x444444,
playerGravity: 900,
playerSpeed: 200,
playerJump: 600,
}
var game = new Phaser.Game(gameOptions.gameWidth, gameOptions.gameHeight);
var mainState = {
…
};
game.state.add('main', mainState);
game.state.start('main');
It will be much easier to modify the gravity or speed in our game as it will be the place
to change it. When you want to use a value from these option, use
Head Start 2018
Add Graphics
First things we need to do is load the graphics file into memory. This is done at the
preload stage of the game. Change the preload function so it looks like:
preload : function(){
this.game.load.image('background', 'assets/images/background.png');
this.game.load.image('grass:8x1', 'assets/images/grass_8x1.png');
this.game.load.image('grass:6x1', 'assets/images/grass_6x1.png');
this.game.load.image('grass:4x1', 'assets/images/grass_4x1.png');
this.game.load.image('grass:2x1', 'assets/images/grass_2x1.png');
this.game.load.image('grass:1x1', 'assets/images/grass_1x1.png');
this.game.load.image('player', 'assets/images/hero_stopped.png');
},
By using this.game.load.image(valueName, sourcePath), we tell the
program where the file can be found, and when we need to use the image, what name
we should use (valueName)
24 At this stage the image are in memory but not on the game canvas, so we cannot see
them if we run the game.
Now let’s create some platforms and a background. We need to modify the create
function.
create: function(){
this.game.physics.startSystem(Phaser.Physics.ARCADE);
this.cursor = this.game.input.keyboard.createCursorKeys();
this.game.physics.arcade.gravity.y = 900;
this.game.world.enableBody = true;
this.platforms = this.game.add.group();
let sprite = this.platforms.create(50, 200, 'grass:6x1');
game.physics.enable(sprite);
sprite.body.allowGravity = false;
sprite.body.immovable = true;
this.game.add.image(0,0,'background');
}
The statement this.platforms.create(50, 200, 'grass:6x1') says to build a sprite using
an image stored in memory with the valueName ‘grass:6x1’ that we have loaded in
the preload function.
The result should give us a platform with an image in the background. Unfortunately,
create: function(){
this.game.physics.startSystem(Phaser.Physics.ARCADE);
this.cursor = this.game.input.keyboard.createCursorKeys();
this.game.physics.arcade.gravity.y = 900;
this.game.world.enableBody = true;
this.game.add.image(0,0,'background');
this.platforms = this.game.add.group();
let sprite = this.platforms.create(50, 200, 'grass:6x1');
game.physics.enable(sprite);
sprite.body.allowGravity = false;
sprite.body.immovable = true;
}
25
You should now have a game looking like that.
Let’s make something a little bit more fancy, let’s create a lift, i.e. a platform that goes
up and down. It is the same principle as our enemies from yesterday session.
For a lift, we may want to set three properties of our own. The maximum height it can
go to (minY) and the minimum height it should go down to. In addition, we may want
to setup the speed of the lift. It is important to emphasize that these three properties
do not exist in Phaser for a sprite, but we have the ability to add additional properties
ourselves.
Head Start 2018
create: function(){
…
this.lifts = this.game.add.group();
sprite = this. lifts.create(350, 350, 'grass:2x1');
game.physics.enable(sprite);
sprite.body.allowGravity = false;
sprite.body.immovable = true;
sprite.minY = 100;
sprite.maxY = 400;
sprite.speed = 100;
sprite.body.velocity.y = sprite.speed;
},
…
26
We have created the platform, but if we run the game, the platform moves out of the
frame. This is because we have not told the platform to go the other way when it
reaches minY or maxY. We need to do that in the update function.
update: function(){
this.lifts.forEach(function(platform){
if(platform.y > platform.maxY){
platform.body.velocity.y = -platform.speed;
} else if(platform.y < platform.minY){
platform.body.velocity.y = platform.speed;
}
});
}
Add Sound FX
First step is to load the audio file. We do that in the preload function by using the
statement:
this.game.load.audio('sfx:jump', 'assets/audio/jump.wav');
Next we must add the sound to the game. This is done in the create function. Add the
statement:
this.sfx = {
jump: game.add.audio('sfx:jump')
};
Note that we add the sound to a set of pairs name:values. This will be useful to add
all our sound effects to this set called this.sfx. Every time we want to use the sound
jump we can do it by referencing this.sfx.jump.
For example to play the sound jump, use the command this.sfx.jump.play(). Try to find
where you can add that line of code in order to have this sound played when the 27
player jump.
Using a Variable to setup the platforms
At the top of index.js file, add the variable setupData:
var setupData = {
"platforms": [
{"image": "ground", "x": 0, "y": 546},
{"image": "grass:8x1", "x": 0, "y": 420},
{"image": "grass:2x1", "x": 420, "y": 336},
{"image": "grass:1x1", "x": 588, "y": 504},
{"image": "grass:8x1", "x": 672, "y": 378},
{"image": "grass:4x1", "x": 126, "y": 252},
{"image": "grass:6x1", "x": 462, "y": 168},
],
"lifts" : [
{"image": "grass:2x1", "x": 798, "y": 84, "min:y":50, "max:y":450, "speed":50}
]
}
As you can see, setup data is a set of pairs name:value where the values are list of set
of pairs name:value. This is a nested data structure.
Head Start 2018
• Platforms is a list containing the platforms, and the platforms are described
using the name “image” with the image label to use, “x” with the x coordinate
of the platform in the game, and “y” with its y coordinate.
• Lift is a list (in the example above containing only one element) of lifts, and the
lifts have the same properties “image”, “x”, “y” as platforms, but in addition
have “min:y”, “max:y”, and “speed”.
The next step is to create the platforms based on the information stored in setupData.
First remove all our code regarding platforms that we have done previously. At the
bottom of the function create, add the following lines of code.
this.platforms = this.game.add.group();
this.lifts = this.game.add.group();
28
Now we need to write the two functions _ spawnPlatform and _ spawnLift. This could
be done between the create and update functions.
{
"platforms": [
{"image": "ground", "x": 0, "y": 546},
{"image": "grass:8x1", "x": 0, "y": 420},
{"image": "grass:2x1", "x": 420, "y": 336},
{"image": "grass:1x1", "x": 588, "y": 504},
{"image": "grass:8x1", "x": 672, "y": 378},
{"image": "grass:4x1", "x": 126, "y": 252},
{"image": "grass:6x1", "x": 462, "y": 168}
],
"lifts" : [
{"image": "grass:2x1", "x": 798, "y": 84, "min:y":50, "max:y":450, "speed":50}
]
}
29
Save the in the directory assets/data (if you don’t have one, create one) under the
name level00.json. As you may have notice, it looks the same as the variable
setupData. Phaser provides tools to read such a file format and create a variable like
our setupData. Let see how it is done.
First, remove the variable setupData from index.js. Secondly, we need to add the file
into the memory (in the preload function) in the same way we preloaded the images
and sound fx. For such a file (text file) we must use:
this.game.load.text("level:0", 'assets/data/level00.json');
When we want to access the file we use the associated name “level:0”. Now we need
to read the content of the file:
level_text = this.game.cache.getText("level:0");
Finally parse the content to create our setup data
setupData = JSON.parse(level_text);
The rest of the code should not changed and it should work. You will see that we can
Head Start 2018
add/remove platforms from the game without changing the index.js file, but by
changing the level00.json. In fact we could have a collection of such files, each
containing a different level of the game.
30
Setup
Download the zip file from
https://www-users.cs.york.ac.uk/~lblot/outreach/headstart/GameSession3.zip
then unzip the file into your headstart folder.
<!DOCTYPE html>
<html lang="en-UK">
<head>
<meta charset="utf-8">
<script src="../node_modules/phaser-ce/build/phaser.min.js"></script>
<title> Game </title>
</head>
</html>
Note: You may have to change the path to phaser.min.js depending on where it has
been install. <script src="../build/phaser.min.js"></script> should be the right path
for your setup. If it is not working ask us.
In order to make the game works, we need to indicate in the HTML file where all the
scripts can be found. Note that main.js is our main program that will start the game.
It must be declared last to ensure all script have been loaded before it runs.
31
Add all states to the game
main.js
var gameOptions = {
…
}
game.state.add('boot', bootState);
game.state.add('load', loadState);
game.state.add('play', mainState);
game.state.add('gameOver', gameOverState);
2. loadState: Read the parameter of the game from a JSON file and load the
assets (images, audio, data)
3. mainState: this is actually the game, where we can play several level
4. gameOverState: this is the state shown when a player win/lose. It also ask if
the player want to play again and then restart the game.
Finally, once all states are added to the game, we start the game indicating the path
to the JSON file containing the game parameters. In our case level01_assets.json.
level01_assets.json
{
"player": { "type": "image", "source": "assets/images/hero_stopped.png"},
"coin": { "type": "image", "source": "assets/images/coin_icon.png"},
"background": {"type": "image","source": "assets/images/background.png"},
…
"sfx:jump": { "type": "sfx", "source": "assets/audio/jump.wav" },
"sfx:door": { "type": "sfx", "source": "assets/audio/door.wav" },
…
"level:0": { "type": "level", "source": "assets/data/level00.json" },
32 "level:1": { "type": "level", "source": "assets/data/level01.json" }
}
JSON files
We have seen during session 2 that it would be useful to get the information to build
a level from a JSON file. This is what we are doing with our game. The first level is
defined by the file level00.json.
level00.json
{
"platforms": [
{"image": "ground", "x": 0, "y": 546},
{"image": "grass:1x1", "x": 588, "y": 504},
{"image": "grass:8x1", "x": 672, "y": 378}
],
"player": {"x": 21, "y": 500},
"coin": {"x": 800, "y": 500}
}
create : function () {
var assets_text, assets_data;
assets_text = this.game.cache.getText("assets");
assets_data = JSON.parse(assets_text);
this.game.state.start("load", true, false, assets_data);
}
Whatever objects and values store in assets_data, it will be passed to the
loadState object. This variable can be retrieved in loadState via the init function, and
the parameter level_data. Now all objects in asset_data are in level_data in the file
load.js (show below).
var loadState = {
33