<canvas width="360" height="480" id="canvas" tabindex="0"></canvas>
* {
padding: 0;
margin: 0;
}
body {
background: black;
}
#canvas {
background: black;
border: solid 5px white;
position: absolute;
top: calc(50% - 245px);
left: calc(50% - 185px);
outline: none;
}
"use strict";
/**************************/
/***** CLASS : HELPER *****/
/**************************/
class Helper {
randNumGen(from, to) {
return Math.floor((Math.random() * (to + 1)) + from);
}
}
/**************************/
/***** CLASS : IMAGES *****/
/**************************/
class Images {
base64ToImg(value) {
var img = new Image();
img.src = value;
return img;
}
spaceship() {
return this.base64ToImg("");
}
bacteria() {
return this.base64ToImg("");
}
title() {
return this.base64ToImg("");
}
controls() {
return this.base64ToImg("");
}
begin() {
return this.base64ToImg("");
}
gameover() {
return this.base64ToImg("");
}
score() {
return this.base64ToImg("");
}
life0() {
return this.base64ToImg("");
}
life1() {
return this.base64ToImg("");
}
life2() {
return this.base64ToImg("");
}
life3() {
return this.base64ToImg("");
}
numbers() {
return this.base64ToImg("");
}
}
/**************************/
/***** CLASS : SCRIPT *****/
/**************************/
class Script {
constructor() {
this.canvas = document.querySelector("#canvas");
this.context = canvas.getContext("2d");
this.helper = new Helper();
this.images = new Images();
this.keyArray = [
32, // 'Space bar'
37, // 'Left' key
39 // 'Right' key
];
this.keyState = {};
this.data = {
start: false,
end: false,
canvas: {
width: 360,
height: 480
},
images: {
title: {
width: 175,
height: 55,
position: {
x: 92,
y: 100
},
image: this.images.title()
},
controls: {
width: 225,
height: 87,
position: {
x: 69,
y: 250
},
image: this.images.controls()
},
begin: {
width: 273,
height: 15,
position: {
x: 43,
y: 400
},
image: this.images.begin()
},
gameover: {
width: 273,
height: 33,
position: {
x: 43,
y: 100
},
image: this.images.gameover()
},
life: {
width: 69,
height: 15,
position: {
x: 25,
y: 460
},
images: [
this.images.life0(),
this.images.life1(),
this.images.life2(),
this.images.life3()
]
},
score: {
width: 69,
height: 15,
position: {
x: 135,
y: 460
},
image: this.images.score()
},
numbers: {
width: 9,
height: 15,
position: {
x: 207,
y: 460
},
image: this.images.numbers(),
clip: [
{x: 0, y: 0},
{x: 9, y: 0},
{x: 18, y: 0},
{x: 27, y: 0},
{x: 36, y: 0},
{x: 45, y: 0},
{x: 54, y: 0},
{x: 63, y: 0},
{x: 72, y: 0},
{x: 81, y: 0}
]
}
},
spaceship: {
image: this.images.spaceship(),
width: 65,
height: 65,
position: {
x: 0,
y: 0
},
fire: {
width: 5,
height: 15,
spawn: [],
moveSpeed: 4
},
moveSpeed: 2,
life: 3,
score: 0
},
bacteria: {
image: this.images.bacteria(),
width: 31,
height: 31,
spawn: [],
moveSpeed: {
min: 2,
max: 8
},
summonTime: {
min: 2,
max: 6,
multiple: 10
},
summonCount: {
min: 1,
max: 4
},
timer: 0
}
};
this.gameOverShowTime = 100;
this.gameOverCount = 0;
}
/*******************/
/***** HELPERS *****/
/*******************/
random(min, max) {
return Math.random() * (max - min) + min;
}
/***********************/
/***** EVENTS AREA *****/
/***********************/
setEvents() {
var script = this;
window.addEventListener("keydown", function(evt) {
if (script.keyArray.indexOf(evt.keyCode) >= 0) {
script.keyState[evt.keyCode] = true;
}
if (evt.keyCode === 32 && !script.data.start) {
script.keyState[32] = false;
script.data.start = true;
}
if (evt.keyCode === 32 && script.data.end) {
script.keyState[32] = false;
if (script.gameOverCount < script.gameOverShowTime) {
return;
}
script.data.start = false;
script.data.end = false;
script.data.spaceship.life = 3;
script.data.spaceship.score = 0;
script.data.spaceship.fire.spawn = [];
script.data.bacteria.spawn = [];
script.gameOverCount = 0;
script.setImages();
}
}, true);
window.addEventListener("keyup", function(evt) {
if (script.keyArray.indexOf(evt.keyCode) >= 0) {
script.keyState[evt.keyCode] = false;
}
}, true);
}
/***************************/
/***** EVENTS AREA END *****/
/***************************/
/**************************/
/***** COLLISION AREA *****/
/**************************/
fireToBacteriaCollision(i) {
for (var j = 0; j < this.data.bacteria.spawn.length; j++) {
if (
typeof this.data.spaceship.fire.spawn[i] === "undefined" ||
typeof this.data.bacteria.spawn[j] === "undefined") {
continue;
}
if (
this.data.spaceship.fire.spawn[i].y < (this.data.bacteria.spawn[j].y + this.data.bacteria.height)
&& (this.data.spaceship.fire.spawn[i].y + this.data.spaceship.fire.height) > this.data.bacteria.spawn[j].y
&& this.data.spaceship.fire.spawn[i].x < (this.data.bacteria.spawn[j].x + this.data.bacteria.width)
&& (this.data.spaceship.fire.spawn[i].x + this.data.spaceship.fire.width) > this.data.bacteria.spawn[j].x
) {
this.data.spaceship.fire.spawn.splice(i, 1);
this.data.bacteria.spawn.splice(j, 1);
script.data.spaceship.score++;
}
}
}
bacteriaToSpaceShipCollision(i) {
if (typeof this.data.bacteria.spawn[i] === "undefined") {
return;
}
if (
(this.data.bacteria.spawn[i].y + this.data.bacteria.height) > this.data.spaceship.position.y
&& this.data.bacteria.spawn[i].y < (this.data.spaceship.position.y + this.data.spaceship.height)
&& (this.data.bacteria.spawn[i].x + this.data.bacteria.width) > this.data.spaceship.position.x
&& this.data.bacteria.spawn[i].x < (this.data.spaceship.position.x + this.data.spaceship.width)
) {
this.data.bacteria.spawn.splice(i, 1);
script.data.spaceship.life--;
}
}
/******************************/
/***** COLLISION AREA END *****/
/******************************/
/*****************************/
/***** IMAGE RENDER AREA *****/
/*****************************/
/***** SPACE SHIP *****/
moveSpaceship() {
if (
this.keyState[37]
&& (
(
this.data.spaceship.position.x
- this.data.spaceship.moveSpeed
)
>= 0
)
) {
this.data.spaceship.position.x -= this.data.spaceship.moveSpeed;
}
else if (
this.keyState[39]
&& (
(
this.data.spaceship.position.x
+ this.data.spaceship.width
+ this.data.spaceship.moveSpeed
)
<= this.data.canvas.width
)
) {
this.data.spaceship.position.x += this.data.spaceship.moveSpeed;
}
}
drawSpaceship() {
this.context.drawImage(
this.data.spaceship.image,
this.data.spaceship.position.x,
this.data.spaceship.position.y,
this.data.spaceship.width,
this.data.spaceship.height
);
}
drawLife() {
if (this.data.spaceship.life === 0) {
this.data.end = true;
}
this.context.drawImage(
this.data.images.life.images[this.data.spaceship.life],
this.data.images.life.position.x,
this.data.images.life.position.y,
this.data.images.life.width,
this.data.images.life.height
);
}
drawScoreText() {
this.context.drawImage(
this.data.images.score.image,
this.data.images.score.position.x,
this.data.images.score.position.y,
this.data.images.score.width,
this.data.images.score.height
);
}
drawScore() {
var scoreText = this.data.spaceship.score.toString();
for (var i = 0; i < scoreText.length; i++) {
this.context.drawImage(
this.data.images.numbers.image,
this.data.images.numbers.clip[parseInt(scoreText[i])].x,
this.data.images.numbers.clip[parseInt(scoreText[i])].y,
this.data.images.numbers.width,
this.data.images.numbers.height,
this.data.images.numbers.position.x + (i * (this.data.images.numbers.width + 3)),
this.data.images.numbers.position.y,
this.data.images.numbers.width,
this.data.images.numbers.height
);
}
}
renderSpaceship() {
this.moveSpaceship();
this.drawSpaceship();
this.drawLife();
this.drawScoreText();
this.drawScore();
}
/***** SPACE SHIP END *****/
/***** FIRE *****/
makeFire() {
if (script.keyState[32]) {
script.keyState[32] = false;
if (this.data.spaceship.fire.spawn.length < 2) {
this.data.spaceship.fire.spawn.push({
x: this.data.spaceship.position.x + Math.ceil((this.data.spaceship.width / 2)) - Math.ceil(this.data.spaceship.fire.width / 2),
y: this.data.spaceship.position.y - this.data.spaceship.fire.height
});
}
}
}
moveFire(i) {
if (typeof this.data.spaceship.fire.spawn[i] === "undefined") {
return;
}
this.data.spaceship.fire.spawn[i].y -= this.data.spaceship.fire.moveSpeed;
if (this.data.spaceship.fire.spawn[i].y + this.data.spaceship.fire.height < 0) {
this.data.spaceship.fire.spawn.splice(i, 1);
}
}
drawFire() {
for (var i = 0; i < this.data.spaceship.fire.spawn.length; i++) {
if (typeof this.data.spaceship.fire.spawn[i] === "undefined") {
continue;
}
this.context.fillStyle = "white";
this.context.fillRect(
this.data.spaceship.fire.spawn[i].x,
this.data.spaceship.fire.spawn[i].y,
this.data.spaceship.fire.width,
this.data.spaceship.fire.height
);
this.moveFire(i);
this.fireToBacteriaCollision(i);
}
}
renderFire() {
this.makeFire();
this.drawFire();
}
/***** FIRE END *****/
/***** BACTERIA *****/
getBacteriaSummonTime() {
return Math.round(
this.random(
this.data.bacteria.summonTime.min,
this.data.bacteria.summonTime.max
)
) * this.data.bacteria.summonTime.multiple;
}
getBacteriaMovespeed() {
return this.random(
this.data.bacteria.moveSpeed.min,
this.data.bacteria.moveSpeed.max
);
}
getBacteriaSummonCount() {
return Math.round(
this.random(
this.data.bacteria.summonCount.min,
this.data.bacteria.summonCount.max
)
);
}
summonBacteria() {
this.data.bacteria.timer++;
// PATCH 4
// Added summon count
for (let i = 0; i < this.getBacteriaSummonCount(); i++) {
// PATCH 3
// Bacteria has random summon time
if (this.data.bacteria.timer % this.getBacteriaSummonTime() === 0) {
this.data.bacteria.spawn.push({
x: this.helper.randNumGen(
Math.ceil(this.data.bacteria.width / 2),
this.data.canvas.width
- this.data.bacteria.width
- Math.ceil(this.data.bacteria.width / 2)
),
y: 0 - this.data.bacteria.height,
// PATCH 2
// Bacteria has random speed
moveSpeed: this.getBacteriaMovespeed()
});
}
}
}
moveBacteria(i) {
if (typeof this.data.bacteria.spawn[i] === "undefined") {
return;
}
if (this.data.bacteria.spawn[i].y > this.data.canvas.height) {
this.data.bacteria.spawn.splice(i, 1);
// PATCH 1
// Space ship will not lose life anymore when the bacteria passed through.
// this.data.spaceship.life--;
}
else {
this.data.bacteria.spawn[i].y += this.data.bacteria.spawn[i].moveSpeed;
this.bacteriaToSpaceShipCollision(i);
}
}
drawBacteria() {
for (var i = 0; i < this.data.bacteria.spawn.length; i++) {
try {
this.moveBacteria(i);
this.context.drawImage(
this.data.bacteria.image,
this.data.bacteria.spawn[i].x,
this.data.bacteria.spawn[i].y,
this.data.bacteria.width,
this.data.bacteria.height
);
}
catch (ex) {
continue;
}
}
}
renderBacteria() {
this.summonBacteria();
this.drawBacteria();
}
/***** BACTERIA END *****/
clearCanvas() {
this.context.clearRect(0, 0, this.data.canvas.width, this.data.canvas.height);
}
render() {
this.clearCanvas();
this.renderSpaceship();
this.renderFire();
this.renderBacteria();
}
/*********************************/
/***** IMAGE RENDER AREA END *****/
/*********************************/
/************************************/
/***** DATA INITIALIZATION AREA *****/
/************************************/
/***** IMAGES *****/
setTitle() {
this.context.drawImage(
this.data.images.title.image,
this.data.images.title.position.x,
this.data.images.title.position.y,
this.data.images.title.width,
this.data.images.title.height
);
}
setControls() {
this.context.drawImage(
this.data.images.controls.image,
this.data.images.controls.position.x,
this.data.images.controls.position.y,
this.data.images.controls.width,
this.data.images.controls.height
);
}
setBegin() {
this.context.drawImage(
this.data.images.begin.image,
this.data.images.begin.position.x,
this.data.images.begin.position.y,
this.data.images.begin.width,
this.data.images.begin.height
);
}
setGameOver() {
this.context.drawImage(
this.data.images.gameover.image,
this.data.images.gameover.position.x,
this.data.images.gameover.position.y,
this.data.images.gameover.width,
this.data.images.gameover.height
);
script.data.start = false;
this.gameOverCount++;
}
setImages() {
this.context.clearRect(0, 0, this.data.canvas.width, this.data.canvas.height);
if (!script.data.start && !script.data.end) {
this.setTitle();
this.setControls();
this.setBegin();
}
else if (
(script.data.start && script.data.end) ||
(!script.data.start && script.data.end)
) {
this.setGameOver();
this.drawLife();
this.drawScoreText();
this.drawScore();
}
}
/***** IMAGES END *****/
/***** SPACE SHIP *****/
setSpaceshipPosition() {
this.data.spaceship.position.x = Math.round((this.data.canvas.width / 2) - (this.data.spaceship.width / 2));
this.data.spaceship.position.y = this.data.canvas.height - this.data.spaceship.height - 25;
}
/***** SPACE SHIP END *****/
initialize(callback) {
this.setEvents();
this.setImages();
this.setSpaceshipPosition();
callback();
}
/****************************************/
/***** DATA INITIALIZATION AREA END *****/
/****************************************/
}
var script = new Script();
script.initialize(function startAnimation() {
requestAnimationFrame(startAnimation);
if (script.data.start && !script.data.end) {
script.render();
}
else if (!script.data.start || script.data.end) {
script.setImages();
}
});
This Pen doesn't use any external CSS resources.
This Pen doesn't use any external JavaScript resources.