“Mike Dangers” HTML5 game engine made with Phaser and ARCADE physics – adding spikes, death and more features
Talking about Mike Dangers game, Game development, HTML5, Javascript and Phaser.
Learn cross platform HTML5 game development
Check my Gumroad page for commented source code, games and books.
var game;
var gameOptions = {
gameWidth: 800,
floorStart: 1 / 8 * 5,
floorGap: 250,
playerGravity: 10000,
playerSpeed: 450,
climbSpeed: 450,
playerJump: 1800,
diamondRatio: 2,
doubleSpikeRatio: 1
}
window.onload = function() {
var windowWidth = window.innerWidth;
var windowHeight = window.innerHeight;
if(windowWidth > windowHeight){
windowHeight = windowWidth * 1.8
}
game = new Phaser.Game(gameOptions.gameWidth, windowHeight * gameOptions.gameWidth / windowWidth);
game.state.add("PreloadGame", preloadGame);
game.state.add("PlayGame", playGame);
game.state.start("PreloadGame");
}
var preloadGame = function(game){}
preloadGame.prototype = {
preload: function(){
game.stage.backgroundColor = 0xaaeaff;
game.scale.scaleMode = Phaser.ScaleManager.SHOW_ALL;
game.scale.pageAlignHorizontally = true;
game.scale.pageAlignVertically = true;
game.stage.disableVisibilityChange = true;
game.load.image("ground", "ground.png");
game.load.image("hero", "hero.png");
game.load.image("ladder", "ladder.png");
game.load.image("diamond", "diamond.png");
game.load.image("spike", "spike.png");
},
create: function(){
game.state.start("PlayGame");
}
}
var playGame = function(game){}
playGame.prototype = {
create: function(){
this.gameOver = false;
game.physics.startSystem(Phaser.Physics.ARCADE);
this.canJump = true;
this.isClimbing = false;
this.defineGroups();
this.drawLevel();
this.defineTweens();
game.input.onTap.add(this.handleTap, this);
},
drawLevel: function(){
this.currentFloor = 0;
this.currentLadder = 0;
this.highestFloorY = game.height * gameOptions.floorStart;
this.floorArray = [];
this.ladderArray = [];
this.diamondArray = [[]];
this.diamondPool = [];
this.spikeArray = [[]];
this.spikePool = [];
while(this.highestFloorY > - 3 * gameOptions.floorGap){
this.addFloor();
this.addLadder();
if(this.currentFloor > 0){
this.addDiamond();
this.addSpikes();
}
this.highestFloorY -= gameOptions.floorGap;
this.currentFloor ++;
}
this.currentFloor = 0;
this.addHero();
},
addSpikes: function(){
this.spikeArray[this.currentFloor] = [];
this.addSpike();
if(game.rnd.integerInRange(0, gameOptions.doubleSpikeRatio) == 0){
this.addSpike();
}
},
addSpike: function(){
var spike = game.add.sprite(game.rnd.integerInRange(50, game.width - 50), this.highestFloorY - 20, "spike");
spike.anchor.set(0.5, 0);
game.physics.enable(spike, Phaser.Physics.ARCADE);
spike.body.immovable = true;
this.spikeGroup.add(spike);
this.spikeArray[this.currentFloor].push(spike);
},
addDiamond: function(){
this.diamondArray[this.currentFloor] = [];
if(game.rnd.integerInRange(0, gameOptions.diamondRatio) != 0){
var diamond = game.add.sprite(game.rnd.integerInRange(50, game.width - 50), this.highestFloorY - gameOptions.floorGap / 2, "diamond");
diamond.anchor.set(0.5, 0);
game.physics.enable(diamond, Phaser.Physics.ARCADE);
diamond.body.immovable = true;
this.diamondGroup.add(diamond);
this.diamondArray[this.currentFloor].push(diamond);
}
},
reviveDiamond: function(){
if(game.rnd.integerInRange(0, gameOptions.diamondRatio) != 0){
if(this.diamondPool.length > 0){
var diamond = this.diamondPool.pop();
diamond.revive();
diamond.y = this.highestFloorY + gameOptions.floorGap - gameOptions.floorGap / 2;
this.diamondArray[this.prevFloor].push(diamond);
}
else{
var diamond = game.add.sprite(game.rnd.integerInRange(50, game.width - 50), this.highestFloorY + gameOptions.floorGap - gameOptions.floorGap / 2, "diamond");
diamond.anchor.set(0.5, 0);
game.physics.enable(diamond, Phaser.Physics.ARCADE);
diamond.body.immovable = true;
this.diamondGroup.add(diamond);
this.diamondArray[this.prevFloor].push(diamond);
}
}
},
reviveSpike: function(){
var spikes = 1;
if(game.rnd.integerInRange(0, gameOptions.doubleSpikeRatio) == 0){
spikes = 2;
}
for(var i = 1; i <= spikes; i++){
if(this.spikePool.length > 0){
var spike = this.spikePool.pop();
spike.y = this.highestFloorY + gameOptions.floorGap - 20;
spike.revive();
this.spikeArray[this.prevFloor].push(spike);
}
else{
var spike = game.add.sprite(game.rnd.integerInRange(50, game.width - 50), this.highestFloorY + gameOptions.floorGap - 20, "spike");
spike.anchor.set(0.5, 0);
game.physics.enable(spike, Phaser.Physics.ARCADE);
spike.body.immovable = true;
this.spikeGroup.add(spike);
this.spikeArray[this.prevFloor].push(spike);
}
}
},
addFloor: function(){
var floor = game.add.sprite(0, this.highestFloorY, "ground");
this.floorGroup.add(floor);
game.physics.enable(floor, Phaser.Physics.ARCADE);
floor.body.immovable = true;
floor.body.checkCollision.down = false;
this.floorArray.push(floor);
},
addLadder: function(){
var ladder = game.add.sprite(game.rnd.integerInRange(50, game.width - 50), this.highestFloorY - gameOptions.floorGap, "ladder");
this.ladderGroup.add(ladder);
ladder.anchor.set(0.5, 0);
game.physics.enable(ladder, Phaser.Physics.ARCADE);
ladder.body.immovable = true;
this.ladderArray.push(ladder);
},
addHero: function(){
this.hero = game.add.sprite(game.width / 2, game.height * gameOptions.floorStart - 40, "hero");
this.gameGroup.add(this.hero)
this.hero.anchor.set(0.5, 0);
game.physics.enable(this.hero, Phaser.Physics.ARCADE);
this.hero.body.collideWorldBounds = true;
this.hero.body.gravity.y = gameOptions.playerGravity;
this.hero.body.velocity.x = gameOptions.playerSpeed;
this.hero.body.onWorldBounds = new Phaser.Signal();
this.hero.body.onWorldBounds.add(function(sprite, up, down, left, right){
if(left){
this.hero.body.velocity.x = gameOptions.playerSpeed;
this.hero.scale.x = 1;
}
if(right){
this.hero.body.velocity.x = -gameOptions.playerSpeed;
this.hero.scale.x = -1;
}
if(down){
game.state.start("PlayGame");
}
}, this)
},
defineTweens: function(){
this.scrollTween = game.add.tween(this.gameGroup).to({
y: gameOptions.floorGap
}, 800, Phaser.Easing.Cubic.Out);
this.scrollTween.onComplete.add(function(){
this.gameGroup.y = 0;
this.gameGroup.forEach(function(item){
if(item.length > 0){
item.forEach(function(subItem) {
subItem.y += gameOptions.floorGap;
}, this);
}
else{
item.y += gameOptions.floorGap;
}
}, this);
}, this);
this.fallingLevelTween = game.add.tween(this.fallingLevelGroup).to({
y: game.height / 2
}, 500, Phaser.Easing.Cubic.Out);
this.fallingLevelTween.onComplete.add(function(){
var numChildren = this.fallingLevelGroup.total;
for(var i = numChildren - 1; i >= 0; i--){
switch(this.fallingLevelGroup.children[i].key){
case "ladder":
console.log(i + "ladder")
this.ladderGroup.add(this.fallingLevelGroup.children[i]);
this.ladderGroup.children[this.ladderGroup.total - 1].y = this.highestFloorY;
break;
case "ground":
console.log(i + "ground")
this.floorGroup.add(this.fallingLevelGroup.children[i]);
this.floorGroup.children[this.floorGroup.total - 1].y = this.highestFloorY + gameOptions.floorGap;
break;
case "diamond":
this.diamondGroup.add(this.fallingLevelGroup.children[i]);
this.killDiamond(Phaser.Math.wrap(this.currentFloor - 1, 0, this.floorArray.length));
break;
case "spike":
this.spikeGroup.add(this.fallingLevelGroup.children[i]);
this.killSpike(Phaser.Math.wrap(this.currentFloor - 1, 0, this.floorArray.length));
}
}
this.fallingLevelGroup.y = 0;
this.reviveDiamond();
this.reviveSpike();
}, this)
},
defineGroups: function(){
this.gameGroup = game.add.group();
this.floorGroup = game.add.group();
this.ladderGroup = game.add.group();
this.diamondGroup = game.add.group();
this.spikeGroup = game.add.group();
this.fallingLevelGroup = game.add.group();
this.gameGroup.add(this.floorGroup);
this.gameGroup.add(this.ladderGroup);
this.gameGroup.add(this.diamondGroup);
this.gameGroup.add(this.spikeGroup);
this.gameGroup.add(this.fallingLevelGroup);
},
handleTap: function(pointer, doubleTap){
if(this.canJump && !this.isClimbing && !this.gameOver){
this.hero.body.velocity.y = -gameOptions.playerJump;
this.canJump = false;
}
},
update: function(){
if(!this.gameOver){
this.checkFloorCollision();
this.checkLadderCollision();
this.checkDiamondCollision();
this.checkSpikeCollision();
this.heroOnLadder();
}
},
checkFloorCollision: function(){
game.physics.arcade.collide(this.hero, this.floorArray[this.currentFloor], function(){
this.canJump = true;
}, null, this);
},
checkLadderCollision: function(){
game.physics.arcade.overlap(this.hero, this.ladderArray, function(player, ladder){
if(!this.isClimbing && Math.abs(player.x - ladder.x) < 10){
this.hero.body.velocity.x = 0;
this.hero.body.velocity.y = - gameOptions.climbSpeed;
this.hero.body.gravity.y = 0;
this.isClimbing = true;
this.scrollTween.start();
}
}, null, this);
},
checkDiamondCollision: function(){
game.physics.arcade.overlap(this.hero, this.diamondArray[this.currentFloor], function(){
this.killDiamond(this.currentFloor);
}, null, this);
},
checkSpikeCollision: function(){
game.physics.arcade.overlap(this.hero, this.spikeArray[this.currentFloor], function(){
this.gameOver = true;
this.hero.body.velocity.x = game.rnd.integerInRange(-20, 20);
this.hero.body.velocity.y = -gameOptions.playerJump
}, null, this);
},
killDiamond: function(floor){
for(var i = 0; i < this.diamondArray[floor].length; i++){
this.diamondArray[floor][i].kill();
this.diamondPool.push(this.diamondArray[floor][i]);
}
this.diamondArray[floor] = [];
},
killSpike: function(floor){
for(var i = 0; i < this.spikeArray[floor].length; i++){
this.spikeArray[floor][i].kill();
this.spikePool.push(this.spikeArray[floor][i]);
}
this.spikeArray[floor] = [];
},
heroOnLadder: function(){
if(this.isClimbing && this.hero.y <= this.floorArray[this.currentFloor].y - gameOptions.floorGap - 40){
this.hero.body.gravity.y = gameOptions.playerGravity;
this.hero.body.velocity.x = gameOptions.playerSpeed * this.hero.scale.x;
this.hero.body.velocity.y = 0;
this.isClimbing = false;
this.fallingLevelGroup.add(this.floorArray[this.currentFloor]);
this.fallingLevelGroup.add(this.ladderArray[this.currentFloor]);
for(var i = 0; i < this.diamondArray[this.currentFloor].length; i++){
this.fallingLevelGroup.add(this.diamondArray[this.currentFloor][i]);
}
for(i = 0; i < this.spikeArray[this.currentFloor].length; i++){
this.fallingLevelGroup.add(this.spikeArray[this.currentFloor][i]);
}
this.fallingLevelGroup
this.fallingLevelTween.start();
this.prevFloor = this.currentFloor;
this.currentFloor = Phaser.Math.wrap(this.currentFloor + 1, 0, this.floorArray.length);
}
}
}
Never miss an update! Subscribe, and I will bother you by email only when a new game or full source code comes out.