原生js实现canvas星际飞船射击类游戏代码
代码语言:html
所属分类:游戏
代码描述:原生js实现canvas星际飞船射击类游戏代码,按住键盘的上下左右键移动位置,按空格键开火。
下面为部分代码预览,完整代码请点击下载或在bfwstudio webide中打开
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<style>
@import url("https://fonts.googleapis.com/css2?family=Fredericka+the+Great&display=swap");
* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
html {
font-size: 62.5%;
}
body {
background: #221d38;
color: whitesmoke;
}
canvas {
background: #867443;
background: radial-gradient(circle at 33% 115%, #867443 4%, #805a42 15%, #221d38 34%);
background-position-y: 147px;
}
.wrapper {
font-family: "Fredericka the Great", cursive;
position: relative;
}
.game_space {
background-color: #5f5f5f;
background: radial-gradient(circle, #867443 4%, #805a42 15%, #221d38 34%);
border: 1px solid #666;
color: whitesmoke;
left: 50%;
margin-top: -6px;
position: fixed;
top: 21rem;
transform: translate(-50%, -50%);
width: 720px;
}
.game_space.playing {
cursor: none;
}
.canvasWrapper {
animation: ship_move_y 3s cubic-bezier(0.34, 1.56, 0.64, 1) forwards, ship_move_x 2.5s cubic-bezier(0.34, 1.2, 0.64, 1) forwards;
background: url("//repo.bfw.wiki/bfwrepo/webp/ship_dark.webp");
background-repeat: no-repeat;
background-position: -210px 190px;
height: 300px;
position: relative;
width: 720px;
z-index: 0;
}
@keyframes ship_move_x {
0% {
background-position-x: -210px;
}
50%, 100% {
background-position-x: 350px;
}
}
@keyframes ship_move_y {
0% {
background-position-y: 190px;
}
50%, 100% {
background-position-y: 75px;
}
}
.messaging {
font-size: 4rem;
left: 50%;
pointer-events: none;
position: absolute;
text-align: center;
text-shadow: 1px 1px 10px whitesmoke;
top: 50%;
transform: translate(-50%, -50%);
z-index: 1;
}
.messaging span {
display: inline-block;
}
.messaging span.h2 {
font-size: 2.5rem;
text-shadow: 1px 1px 10px firebrick;
}
.messaging span.h3 {
font-size: 1.9rem;
margin-top: 10px;
text-align: left;
text-shadow: 1px 1px 10px #241e9e;
}
.messaging span.h3 .action {
margin-top: 0;
width: 55px;
}
.level_display,
.score_display {
animation: reveal_score_display 1.5s ease-out forwards;
animation-delay: 1.5s;
font-size: 1.6rem;
height: 2.5rem;
margin: -8rem auto 0;
overflow: hidden;
padding: 0 8px;
text-align: right;
width: 720px;
}
.level_display {
animation: reveal_level_display 1.5s ease-out forwards;
text-align: left;
}
.level_display.hidden,
.score_display.hidden {
margin-top: -8rem;
}
@keyframes reveal_score_display {
0% {
margin-top: -8rem;
}
50%, 100% {
margin-top: 3rem;
}
}
@keyframes reveal_level_display {
0% {
margin-top: -8rem;
}
50%, 100% {
margin-top: -2.5rem;
}
}
.loading_font {
position: fixed;
top: -9999px;
}
.hidden {
display: none;
}
</style>
</head>
<body>
<div class="wrapper">
<div class="game_space loading">
<div class="canvasWrapper"></div>
<div class="messaging"></div>
</div>
<div class="score_display hidden">Score: <span class="score">0</span></div>
<div class="level_display hidden">Level: <span class="level">_</span></div>
<div class="loading_font" aria-hidden="true"> </div>
</div>
<script>
// Pen-ified version of this project:
// https://codepen.io/2Mogs/project/editor/DddNpw
/***
CLASSES FIRST
Main code all the way down at the bottom (approx line 1200)
***/
/* CONSTANTS */
class EventIDs {
constructor() {
this.ENTER_PRESSED = 'enter key pressed';
this.EXPLODE = 'explode';
this.FIRE = 'fire';
this.GAME_OVER = 'game over';
this.GET_READY = 'Get next level ready';
this.LEVEL_COMPLETE = 'level complete';
this.RESET_GAME = 'reset game';
this.START = 'start';
}}
class Messaging {
constructor() {
this.gameOver = 'Game Over';
this.levelOver = 'You beat the level';
this.start = 'Click to Play';
}}
class MyMath {
constructor() {
this.mFloor = Math.floor;
this.mRound = Math.round;
this.mRandom = Math.random;
this.mSqrt = Math.sqrt;
this.tau = Math.PI * 2;
}}
class EnemyPatterns {
constructor() {
this.baseRules = {
baseX: 535,
baseY: 44,
scalar: .5,
spaceY: 40,
spaceX: 50,
speed: .6 };
this.patterns = [
{
id: 'code',
rows: ['0111001100111001111',
'1000010010100101000',
'1000010010100101110',
'1000010010100101000',
'0111001100111001111'] },
{
id: 'pen',
rows: ['111001111010001',
'100101000011001',
'100101110010101',
'111001000010011',
'100001111010001'] },
this.generatePattern()];
}
generatePattern() {
return {
id: 'generated',
baseX: 750,
speed: 1.2,
rows: [this.generateRow(),
this.generateRow(),
this.generateRow(),
this.generateRow(),
this.generateRow()] };
}
generateRow(size = 10) {
let row = '';
for (let i = 0; i < size; i++) {
row += math.mRandom() < .85 ? '1' : '0';
}
return row;
}}
let eventIDs = new EventIDs();
let math = new MyMath();
let messaging = new Messaging();
let formations = new EnemyPatterns();
/* CONSTANTS - END */
/* MANAGERS & WATCHERS */
class EventManager {
constructor() {
this.subscribers = {};
}
subscribe(event, handler) {
if (this.subscribers[event]) {
this.subscribers[event].push(handler);
} else
{
this.subscribers[event] = [handler];
}
}
dispatch(event, data) {
if (this.subscribers[event]) {
this.subscribers[event].forEach(function (handler) {
handler(data);
});
}
}}
let eventManager = new EventManager();
class KeyWatcher {
constructor() {
this.goRIGHT = false;
this.goLEFT = false;
this.goUP = false;
this.goDOWN = false;
this.SPACEBAR = false;
this.keycodeSPACE = 32;
this.keycodeRIGHT = 39;
this.keycodeLEFT = 37;
this.keycodeUP = 38;
this.keycodeDOWN = 40;
this.keycodeW = 87;
this.keycodeA = 65;
this.keycodeS = 83;
this.keycodeD = 68;
this.keycodeENTER = 13;
this.allowEnter = false;
this.playing = false;
this.gameOver = false;
this.keyDownListener = this.handleKeyDown.bind(this);
this.keyUpListener = this.handleKeyUp.bind(this);
document.addEventListener('keydown', this.keyDownListener);
document.addEventListener('keyup', this.keyUpListener);
eventManager.subscribe(eventIDs.GAME_OVER, () => this.handleGameOver());
eventManager.subscribe(eventIDs.GET_READY, () => this.allowEnter = true);
eventManager.subscribe(eventIDs.LEVEL_COMPLETE, () => this.stopWatching());
eventManager.subscribe(eventIDs.START, () => this.startWatching());
}
startWatching() {
this.resetAllActions();
this.playing = true;
}
stopWatching() {
this.resetAllActions();
this.allowEnter = false;
this.playing = false;
}
resetAllActions() {
this.goRIGHT = false;
this.goLEFT = false;
this.goUP = false;
this.goDOWN = false;
this.SPACEBAR = false;
}
handleGameOver() {
this.stopWatching();
this.gameOver = true;
this.allowEnter = true;
}
handleKeyDown(e) {
if (!this.playing) return;
if (e.keyCode === this.keycodeRIGHT || e.keyCode === this.keycodeD) {
this.goRIGHT = true;
}
if (e.keyCode === this.keycodeLEFT || e.keyCode === this.keycodeA) {
this.goLEFT = true;
}
if (e.keyCode === this.keycodeUP || e.keyCode === this.keycodeW) {
this.goUP = true;
}
if (e.keyCode === this.keycodeDOWN || e.keyCode === this.keycodeS) {
this.goDOWN = true;
}
if (e.keyCode === this.keycodeSPACE) {
this.SPACEBAR = true;
}
}
handleKeyUp(e) {
if (this.gameOver) {
this.gameOver = false;
}
if (!this.playing && this.allowEnter) {
if (e.keyCode === this.keycodeENTER) {
eventManager.dispatch(eventIDs.ENTER_PRESSED);
}
} else
{
if (e.keyCode === this.keycodeRIGHT || e.keyCode === this.keycodeD) {
this.goRIGHT = false;
}
if (e.keyCode === this.keycodeLEFT || e.keyCode === this.keycodeA) {
this.goLEFT = false;
}
if (e.keyCode === this.keycodeUP || e.keyCode === this.keycodeW) {
this.goUP = false;
}
if (e.keyCode === this.keycodeDOWN || e.keyCode === this.keycodeS) {
this.goDOWN = false;
}
if (e.keyCode === this.keycodeSPACE) {
this.SPACEBAR = false;
}
}
}}
let keyWatcher = new KeyWatcher();
class CollisionManager {
constructor({ enemies, missile, ship }) {
this.ship = ship;
this.missile = missile;
this.enemies = enemies;
this.running = false;
eventManager.subscribe(eventIDs.LEVEL_COMPLETE, () => this.stopTesting());
eventManager.subscribe(eventIDs.START, () => this.startTesting());
}
startTesting() {this.running = true;}
stopTesting() {this.running = false;}
testCollision({ objA, objB, radius }) {
const dx = objA.x - objB.x;
const dy = objA.y - objB.y;
return math.mSqrt(dx * dx + dy * dy) < radius ? true : false;
}
update() {
if (!this.running) return;
const missilePt = { x: this.missile.sprite.x, y: this.missile.sprite.y };
const shipPt = { x: this.ship.sprite.x, y: this.ship.sprite.y };
let enemyPt;
let hit = false;
this.enemies.enemies.map((enemy, i) => {
if (!enemy.alive) return;
enemyPt = { x: enemy.sprite.x, y: enemy.sprite.y };
// Test against missile
if (this.missile.active && enemyPt.x < 700) {
hit = this.testCollision({ objA: missilePt, objB: enemyPt, radius: enemy.radius });
if (hit) {
this.missile.active = false;
this.enemies.killSprite(i);
eventManager.dispatch(eventIDs.EXPLODE, { x: enemyPt.x, y: enemyPt.y, type: explosions.types.missile, value: enemy.variant + 1 });
}
}
// Test against ship
hit = this.testCollision({ objA: shipPt, objB: enemyPt, radius: ship.radius });
if (hit) {
this.enemies.killSprite(i);
this.ship.takeDamage(1);
eventManager.dispatch(eventIDs.EXPLODE, { x: enemyPt.x, y: enemyPt.y, type: explosions.types.missile, value: (enemy.variant + 1) * 2 });
}
});
}}
/* MANAGERS & WATCHERS - END */
/* CONFIGS */
class EnemiesConfig {
constructor() {
this.height = 60;
this.radius = 34;
this.width = 68;
this.winLineX = 120;
}}
class ExplosionsConfig {
constructor() {
this.durations = {
missile: 15,
ship: 30 };
this.hsl = {
missile: '0,100%,100%',
ship: '10,70%,60%' };
this.types = {
missile: 'missile',
ship: 'ship' };
}}
class GlowsConfig {
constructor() {
this.colours = [
'silver', 'yellowgreen', 'darkorange', 'firebrick'];
this.groundBurn = '#521';
this.missile = '#fff';
this.glowSizes = {
ship: 20,
enemy: 7,
ground: 12,
missile: 3 };
}}
class MissileConfig {
constructor() {
this.height = 4;
this.radius = 6;
this.speed = 8;
this.variants = 4;
this.width = 20;
this.xAdjust = 46;
this.yAdjust = -2;
}}
class RocketFlamesConfig {
constructor() {
this.baseDepth = 14;
this.colours = [
'hsla(0,70%,60%,.2)',
'hsla(45,70%,60%,.2)',
'hsla(190,70%,60%,.2)',
'hsla(310,70%,60%,.2)'];
this.colourArrayLength = this.colours.length - 1;
this.flickerBase = 2;
this.flickerVariance = 10;
this.maxLengthVariance = 20;
this.minLength = 12;
this.layers = 6;
this.xAdjust = -42;
this.yAdjust = 6.5;
}}
class ShipConfig {
constructor() {
this.damageLevel = 0;
this.dead = false;
this.health = 0;
this.healthBase = 10;
this.height = 67;
this.maxX = 500;
this.maxY = 210;
this.minX = 85;
this.minY = 50;
this.radius = 40;
this.width = 102;
this.initialX = -200;
this.initialY = 150;
this.autoFly = {
status: true,
state: 0, // 0/1
steps: 30, // Equates to speed - bigger slower,
tolerance: 1, // How close is 'Arrived'
dest: [
{ x: 120, y: 120 }, // Flying in
{ x: 1000, y: 110 } // Flying off
] };
}}
let enemies = new EnemiesConfig();
let explosions = new ExplosionsConfig();
let glows = new GlowsConfig();
let missile = new MissileConfig();
let rocketFlames = new RocketFlamesConfig();
let ship = new ShipConfig();
/* CONFIGS - END */
class GameCanvas {
constructor() {
this.canvas = null;
this.ctx = null;
this.id = '';
this.height = 0;
this.width = 0;
}
createCanvas({ domElement, id, w, h }) {
this.canvas = document.createElement('canvas');
this.resize(w, h);
this.id = this.canvas.id = id;
domElement.appendChild(this.canvas);
this.ctx = this.canvas.getContext('2d');
}
resize(w, h) {
this.width = this.canvas.width = w;
this.height = this.canvas.height = h;
}}
class Backgrounds {
constructor({ speed, ctx, x = 0, y = 0, h, w, initialVariant = 0 }) {
this.bgSprites = [];
this.spriteData = new SpriteData();
this.ctx = ctx;
this.x = x;
this.y = y;
this.h = h;
this.w = w;
this.speed = speed;
this.previousBgId = initialVariant;
this.numBgVariants = 5;
this.lastBgX = 0;
this.minLastBgX = w;
this.init();
}
init() {
while (this.lastBgX < this.minLastBgX) {
this.addBg();
}
}
addBg() {
this.bgSprites.push(
new SpriteSheet(
{ spriteData: this.spriteData.scenes[this.previousBgId++],
w: this.w,
h: this.h,
ctx: this.ctx,
x: this.lastBgX,
y: this.y,
centerX: false }));
this.lastBgX += this.minLastBgX;
if (this.previousBgId >= this.numBgVariants) {this.previousBgId = 0;}
}
update() {
// Move bgs
this.bgSprites.map(bg => {
bg.x -= this.speed;
});
// Crop bgs outside view
this.bgSprites = this.bgSprites.filter(bg => bg.x > -this.w);
// Add new bg if last bg is in view
this.lastBgX -= this.speed;
if (this.lastBgX < this.minLastBgX) {
this.addBg();
}
}
draw() {
this.bgSprites.map(bg => {
bg.draw();
});
}}
class Ground {
constructor({ ctx, w, h, x, y }) {
this.ctx = ctx;
this.spriteData = new SpriteData();
this.sprite = new SpriteSimple({ spriteData: this.spriteData.ground, w: w, h: h, ctx: ctx, x: x, y: y });
}
update() {return;}
draw() {
this.ctx.shadowBlur = glows.glowSizes['ground'];
this.ctx.shadowColor = glows.groundBurn;
this.sprite.draw();
this.ctx.shadowBlur = 0;
}}
class Plants {
constructor({ speed, ctx, x = 0, y = 0, w, minY = 15, maxY = 40, minGap = 120, maxGap = 230, scalar = 1 }) {
this.plantSprites = [];
this.spriteData = new SpriteData();
this.ctx = ctx;
this.x = x;
this.y = y;
this.speed = speed;
this.minY = minY;
this.maxY = maxY - this.minY;
this.previousPlantId = 0;
this.numPlantVariants = 8;
this.lastPlantX = 55;
this.minLastPlantX = w + 50;
this.minGap = minGap;
this.maxGap = maxGap - this.minGap;
this.scalar = scalar;
this.init();
}
init() {
while (this.lastPlantX < this.minLastPlantX) {
this.addPlant();
}
}
addPlant() {
this.plantSprites.push(
new SpriteSheet(
{ spriteData: this.spriteData.plants,
w: 40,
h: 53,
ctx: this.ctx,
x: this.lastPlantX,
y: this.getY(),
variant: this.getVariant(),
scalar: this.scalar }));
this.lastPlantX += math.mRandom() * this.maxGap + this.minGap;
}
getVariant() {
const nextPlantId = math.mFloor(math.mRandom() * this.numPlantVariants);
if (nextPlantId !== this.previousPlantId) {
this.previousPlantId = nextPlantId;
return nextPlantId;
} else
{
this.getVariant();
}
}
getY() {
return this.y - math.mRandom() * this.maxY + this.minY;
}
update() {
// Move plants
this.plantSprites.map(plant => {
plant.x -= this.speed;
});
// Crop plants outside view
this.plantSprites = this.plantSprites.filter(plant => plant.x > -30);
// Add new plant if last plant is in view
this.lastPlantX -= this.speed;
if (this.lastPlantX < this.minLastPlantX) {
this.addPlant();
}
}
draw() {
this.plantSprites.map(plant => {
plant.draw();
});
}}
class WinLine {
constructor({ ctx, h }) {
this.ctx = ctx;
this.height = h;
}
update() {return;}
draw() {
this.ctx.setLineDash([5, 10]);
// Home base line
this.ctx.strokeStyle = 'hsla(0,60%,50%,.4)';
this.ctx.beginPath();
this.ctx.moveTo(enemies.winLineX, 0);
this.ctx.lineTo(enemies.winLineX, this.height);
this.ctx.stroke();
// Max ship range line
this.ctx.strokeStyle = 'hsla(0,100%,100%,.4)';
this.ctx.beginPath();
this.ctx.moveTo(ship.maxX, 0);
this.ctx.lineTo(ship.maxX, this.height);
this.ctx.stroke();
}}
class UI {
constructor() {
this.levelEl = document.querySelector('.level');
this.messageEl = document.querySelector('.messaging');
this.scoreEl = document.querySelector('.score');
this.gameSpace = document.querySelector('.game_space');
this.level = 0;
this.score = 0;
this.scoreMultiplier = 9;
this.gameRunning = false;
this.listening = false;
this.firstMessage = true;
this.startMessages = [
'Defender 404<br><span class="h2">Stop the invasion before<br>it crosses your home line</span><br><span class="h3"><span class="action">Start</span>: Click / ENTER<br><span class="action">Move</span>: Arrow Keys / WASD<br><span class="action">Fire</span>: SPACEBAR</span>',
'Beat the next wave'];
this.levelClear = 'Level cleared';
this.gameOver = 'GAME OVER<br>Play again?';
eventManager.subscribe(eventIDs.ENTER_PRESSED, () => this.handleStart());
eventManager.subscribe(eventIDs.EXPLODE, data => this.handleExplode(data));
eventManager.subscribe(eventIDs.GAME_OVER, () => this.handleGameOver());
eventManager.subscribe(eventIDs.GET_READY, () => this.handleReady());
eventManager.subscribe(eventIDs.LEVEL_COMPLETE, () => this.handleLevelOver());
this.startListener = this.handleStart.bind(this);
this.gameSpace.addEventListener('click', this.startListener);
}
handleReady() {
this.listening = true;
this.setMessage(this.firstMessage ? this.startMessages[0] : this.startMessages[1]);
this.levelEl.innerHTML = ++this.level;
this.firstMessage = false;
this.gameRunning = true;
}
handleStart() {
if (!this.gameRunning && !this.listening) {
this.reset();
return;
}
this.toggleCursor({ hide: true });
eventManager.dispatch(eventIDs.START);
this.listening = false;
this.setMessage('');
}
handleLevelOver() {
this.toggleCursor({ hide: false });
this.setMessage(this.levelClear);
}
handleExplode(data) {
if (!data.value) return;
this.score += data.value * this.scoreMultiplier;
this.scoreEl.innerHTML = this.score;
}
handleGameOver() {
this.toggleCursor({ hide: false });
this.setMessage(this.gameOver);
this.gameRunning = false;
}
setMessage(message) {
this.messageEl.innerHTML = message;
}
toggleCursor({ hide }) {
hide ? this.gameSpace.classList.add('playing') : this.gameSpace.classList.remove('playing');
}
reset() {
eventManager.dispatch(eventIDs.RESET_GAME);
this.level = 0;
this.score = 0;
this.levelEl.innerHTML = this.level;
this.scoreEl.innerHTML = this.score;
this.gameRunning = true;
this.firstMessage = true;
this.handleReady();
}}
class SpriteData {
constructor() {
this.ground = '.........完整代码请登录后点击上方下载按钮下载查看
















网友评论0