酷站(www.ku0.com)-致力于为互联网从业者提供动力!

热门关键词:  企业  as  baidu  c4rp3nt3r  美女
【阿里云】采购季上云仅¥223/3年

js实现贪吃蛇小游戏教程

来源:互联网搜集 作者:秩名 人气: 发布时间:2019-10-29
本篇文章主要介绍了js实现贪吃蛇小游戏教程,对大家的学习或者工作具有一定的参考学习价值,感兴趣的小伙伴们可以参考一下,也感谢大家对酷站(ku0.com)的支持。

方法如下:

index.html


<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>贪吃蛇</title>
  <link rel="stylesheet" href="style.css" >
</head>
<body>
  <div id="game">
    <div id="explain">
      <div>操作WASD 空格开始/暂停 R重新开始</div>
      <div>当前分数:<span id="grade">0</span>分</div>
    </div>
    <canvas id="snakegame" width="500" height="500">
    </canvas>
  </div>
  <script type="text/javascript" src="main.js" charset="UTF-8"></script>
</body>
</html>

style.css

*{
  padding: 0;
  margin: 0;
}
#game {
  width: 500px;
  margin: auto;
}
#explain {
  width: 500px;
}
#explain div{
  width: 500px;
  height: 30px;
}
#snakegame {
  background: green;
}
 

main.js

/**
 * el 挂载的元素
 * attribute 贪吃蛇的属性
 */
class Game {
  constructor (el, attribute) {
    this.el = document.getElementById(el);
    // 获取画布的宽高
    this.el.elW = parseInt(window.getComputedStyle(this.el).width);
    this.el.elH = parseInt(window.getComputedStyle(this.el).height);
    this.init(attribute);
    this.keyListening();
  }
  // 初始化
  init(attribute) {
    this.attribute = {
      color: "red", // 颜色
      direction: "rigth", // 移动方向
      state: "pause", // 状态 run pause end
      grade: 0, // 分数
      body: [{x: 20, y: 0}, {x: 0, y: 0}], // 贪吃蛇身体
      wh: 20, // 矩形的宽高
      speed: 200 // 速度
    };
    if (attribute) {
      this.newAttribute = attribute;
      Object.keys(attribute).forEach(key => {
        this.attribute[key] = attribute[key];
      });
    }
    this.food ={
      x: 0,
      y: 0,
      color: 'red'
    }
    this.draw();
    this.foodDraw();
  }
  // 绘制贪吃蛇
  draw() {
    let el = this.el;
    let { body, wh, color } = this.attribute;
    // 确定浏览器是否支持canvans元素
    if (el.getContext) {
      let context = snakegame.getContext("2d");
      context.fillStyle = color;
      body.forEach( key => {
        context.fillRect(key.x, key.y, wh, wh);
      });
    }
  }
  // 随机生成食物
  foodDraw() {
    let el = this.el, wh = this.attribute.wh;
    this.food.x = Math.floor(Math.random()*(el.elW - wh)),
    this.food.y = Math.floor(Math.random()*(el.elH - wh));
    while (this.isOverlap()) {
      this.food.x = Math.floor(Math.random()*(el.elW - wh)),
      this.food.y = Math.floor(Math.random()*(el.elH - wh));
    }
    if (el.getContext) {
      let context = snakegame.getContext("2d");
      context.fillStyle = this.food.color;
      context.fillRect(this.food.x, this.food.y, wh, wh);
    }
  }
  // 判断食物是否与贪吃蛇的身体重叠
  isOverlap() {
    let { wh } = this.attribute;
    let food = this.food;
    let flag = false;
    function isIn(key, x, y) {
      if (key.x <= x && key.x + wh >= x && key.y <= y && key.y + wh >= y) {
        return true;
      } else {
        return false;
      }
    }
    this.attribute.body.forEach(key => {
      // 食物的上下左右四个点一个点在贪吃蛇的身体内就判断为重叠
      if (isIn(key, food.x, food.y) || isIn(key, food.x, food.y + wh) || isIn(key, food.x + wh, food.y) || isIn(key, food.x + wh, food.y + wh)) {
        flag = true;
      }
    });
    return flag;
  }
  // 清除图形
  clear(x, y, width, height) {
    // 确定浏览器是否支持canvans元素
    if (this.el.getContext) {
      let context = snakegame.getContext("2d");
      context.clearRect(x, y, width, height);
    }
  }
  // 游戏状态更新
  updateState(state) {
    this.attribute.state = state;
    if (state === "run") {
      this.run();
    } 
  }
  // 游戏线程
  run() {
    let { body, wh, speed} = this.attribute;
    let time = setInterval(() => {
      // 判断游戏线程是否在运行
      if (this.attribute.state !== 'run') {
        clearInterval(time);
      }
      let obj = {};
      switch(this.attribute.direction) {
        case 'left':
          obj['x'] = body[0].x - wh;
          obj['y'] = body[0].y;
          break;
        case 'rigth':
          obj['x'] = body[0].x + wh;
          obj['y'] = body[0].y;
          break;
        case 'up':
          obj['x'] = body[0].x;
          obj['y'] = body[0].y - wh;
          break;
        case 'down':
          obj['x'] = body[0].x;
          obj['y'] = body[0].y + wh;
          break;
      }
      body.unshift(obj);
      // 判断是否吃到食物 
      if (this.isOverlap()) {
        this.clear(this.food.x, this.food.y, wh, wh);
        this.attribute.grade++; 
        this.foodDraw();
        this.draw();
      } else {
        if (this.end()) {
          alert("游戏结束");
          this.updateState('end');
          clearInterval(time);
        } else {
          let item = body.pop();
          this.clear(item.x, item.y, wh, wh);
          this.draw();
        }
      }
    }, speed);
  }
  // 键盘事件监听
  keyListening() {
    document.onkeydown = (event) => {
      let e = event || window.event || arguments.callee.caller.arguments[0];
      if (e && e.keyCode === 87 && this.attribute.direction !== 'down') { // 按下W
        this.attribute.direction = 'up';
      }
      if (e && e.keyCode === 65 && this.attribute.direction !== 'rigth') { // 按下A
        this.attribute.direction = 'left';
      }
      if (e && e.keyCode === 68 && this.attribute.direction !== 'left') { // 按下D
        this.attribute.direction = 'rigth';
      }
      if (e && e.keyCode === 83 && this.attribute.direction !== 'up') { // 按下W
        this.attribute.direction = 'down';
      }
      if (e && e.keyCode === 32) { // 按下空格 
        let state;
        if (this.attribute.state === 'pause') {
          state = 'run';
        } 
        if (this.attribute.state === 'run') {
          state = 'pause';
        }
        this.updateState(state);
      }
      if (e && e.keyCode === 82) { // 按下R键
        this.reStart();
      }
    } 
  }
  // 是否死亡
  end() {
    let body = [...this.attribute.body];
    let obj = body.shift();
    let flag = false;
    if (obj.x < 0 || obj.x >= this.el.elW || obj.y < 0 || obj.y >= this.el.elH) {
      flag = true;
    }
    body.forEach(key => {
      if (key.x === obj.x && key.y === obj.y) {
        flag = true;
      }
    });
    return flag;
  }
  // 重新开始
  reStart() {
    // 清除整个画布
    this.clear(0, 0, this.el.elW, this.el.elH);
    // 重新开始
    this.init(this.newAttribute);
  }
}
let game = new Game("snakegame", {color: "yellow"});
let grade = document.getElementById("grade");
let oldGrade = game.attribute.grade;
setInterval(() => {
  if (oldGrade !== game.attribute.grade) {
    oldGrade = game.attribute.grade;
    grade.innerText = game.attribute.grade;
  }
})

效果如下:


版权声明:本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 959677720#qq.cn(#换@) 举报,一经查实,本站将立刻删除。
原文链接:https://blog.csdn.net/qq_37473645/article/details/102758190

相关文章

  • Javascript执行流程细节的介绍

    Javascript执行流程细节的介绍

    Javascript从定义到执行,JS引擎在实现层做了很多初始化工作,因此在学习JS引擎工作机制之前,我们需要引入几个相关的概念:执行环境栈、全局对象、执行环境、变量对象、活动对象、作用域和作用域链等,这些概念正是JS引擎工作的核心组件......
    05-15
  • javascript代码是如何被压缩

    javascript代码是如何被压缩

    随着前端的发展,特别是 React , Vue 等构造单页应用的兴起,前端的能力得以很大提升,随之而来的是项目的复杂度越来越大。 此时的前端的静态资源也越来越庞大,而毫无疑问 javascript 资源已是前端的主体资源,对于压缩它的体积至为重......
    05-06
  • 解决React在安装antd之后出现的Can't resolve './locale'问题

    解决React在安装antd之后出现的Can't resolve './locale'问题

    React在安装antd之后出现的Cant resolve ./locale问题,是因为moment在2.2之后的版本都有问题,而react默认使用了最新的moment,但是在moment@2.18.1中是没有问题的。 解决方案就是配置webpack的alias,将所有的 moment 路径引用导入到 mo......
    05-03
  • jQuery实现移动端图片上传预览组件的方法

    jQuery实现移动端图片上传预览组件的方法

    我们先来捋一捋想要实现的功能: 预览 删除 压缩 上传到服务器 基本机构 这样,我们的组件结构就有了: ;!function(window, $, undefined){ function Upload() { }; Upload.prototype.change = function() { }; Upload.prototype.del = f......
    05-01
  • js解析JSON数据的详解

    js解析JSON数据的详解

    JSON 数据如下: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 { name : mkyong , age : 30, address : { streetAddress : 88 8nd Street , city : New York }, phoneNumber : [ { type : home , number : 111 111-1111 }, { type : ......
    04-21
  • js中间件设计模式的深入探讨与实例分析

    js中间件设计模式的深入探讨与实例分析

    中间件作为一些辅助处理功能,应用非常广泛,例如express中间件,redux中间件,koa中间件,那么中间件的设计模式到底是怎样的呢。结合中间件的使用实例探讨和总结一下中间件的设计思想和一般实现模式。 仿照redux中间件实现模式,看如下......
    04-11
  • Quasar Input:type="number" 去掉上下小箭头 实现加减按钮样式

    Quasar Input:type="number" 去掉上下小箭头 实现加减按钮样式

    实现效果 UI组件依然是使用 Quasar Framework。 先来看一下效果: 加减.gif input type=number 去掉上下小箭头 去掉上下小箭头,主要是css: input::-webkit-outer-spin-button,input::-webkit-inner-spin-button {-webkit-appearance: no......
    04-09
  • js实现登录时记住密码

    js实现登录时记住密码

    常见的很多网站登录,都有记住密码功能,下面是用js实现的记住密码功能(代码用的源生js,不用引入任何插件,大家如果引入了jQuery,可以进行修改,优化) js部分 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 ......
    04-05
  • jQuery实现鼠标放置名字上显示详细内容气泡提示框效果

    jQuery实现鼠标放置名字上显示详细内容气泡提示框效果

    本文实例讲述了jQuery实现鼠标放置名字上显示详细内容气泡提示框效果的方法。分享给大家供大家参考,具体如下: 实现效果如上图,当鼠标放置到名字上时,则显示出内容详情。 实现具体过程如下: 1、需要加这句js ? 1 2 3 4 5 6 !--实现鼠......
    04-04
  • Javascript组合继承方法的代码

    Javascript组合继承方法的代码

    组合继承 组合继承,指将原型链和借用构造函数的技术组合到一块,从而发挥二者之长的一种继承模式。其背后思路使用用原型链实现对原型属性和方法的继承,而通过构造函数来实现对实例属性的继承。这样,即通过在原型上定义方法实现了函数......
    04-02

最新更新