原生js实现canvas星际飞船射击类游戏代码

代码语言:html

所属分类:游戏

代码描述:原生js实现canvas星际飞船射击类游戏代码,按住键盘的上下左右键移动位置,按空格键开火。

代码标签: 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">&nbsp;</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