原生JavaScript实现的贪吃蛇
github代码地址:https://github.com/McRayFE/snake
涉及到的知识点:
- 键盘事件
- setInterval()定时器
- javascript中数组的使用
- 碰撞的检测
- offsetLeft,offsetWidth,offsetTop,offsetHeight
实现的思路
- 让小蛇动起来
- 让食物随机生成
- 检测碰撞
- 让身体跟随
HTML主体代码
<div class="container">
       <!--小蛇移动的操场-->
       <div id="playground">
           <!--小蛇-->
           <div id="snake"></div>
           <!--食物-->
           <div id="food"></div>
       </div>
       <!--记录得分-->
       <div id="menu">
          <div>得分<span id="score"></span></div>
       </div>
   </div>
一个小蛇移动的操场playground,里面包含了小蛇snake和食物food,还有一个记录得分的menu
CSS代码
*{
    padding:0;
    margin:0;
}
.container{
    width:600px;
    height:500px;
    margin:0 auto;
}
#playground{
    width:450px;
    height:100%;
    background-color:pink;
    float:left;
    position:relative;
}
#menu{
    width:150px;
    height:100%;
    background-color:blue;
    float:left;
}
#snake{
    width:20px;
    height:20px;
    background-color:red;
    position:absolute;
    left:0;
    top:0;
}
#food{
    width:20px;
    height:20px;
    background-color:yellow;
    position:absolute;
}
.body{
    width:20px;
    height:20px;
    background-color:green;
    position:absolute;;
    top:0;
    left:0;
}
#score{
    font-size:30px;
    font-weight:bold;
    color:black;
}
#menu div{
    font-size:20px;
    font-weight:bold;
    margin-left:20px;
}
此时界面的效果如下:
获取元素和定义全局变量
    var snake = document.getElementById("snake");
    var food = document.getElementById("food");
    var playground = document.getElementById("playground");
    var score = document.getElementById('score');
    /*设置全局变量*/
    var timer;
    var srr = [];//存放蛇的位置的数组
    var num = 0;//数组的长度
    var snakeBody;//每次吃调一个食物,增加的身体
    var timer1 = setInterval(eat,10);//设置一个碰撞的时间器
让小蛇动起来
监听键盘事件,判断上下左右键的ASCII码值来执行不同方向的移动函数,其中每移动1px,就要将小蛇的位置存入到arr[]数组中,因为后面实现身体跟随的时候,是根据数组里面的位置来动态变化的。
document.onkeydown = function(e){
        var evt = window.evnet || e;
        switch(evt.keyCode){
            case 37://左
                clearInterval(timer);
                timer = window.setInterval(runLeft,10);//向左移动的时间器
              function runLeft(){
                if(snake.offsetLeft > 0){
                    snake.style.left = snake.offsetLeft - 1 + 'px';
                    snake.style.top = snake.offsetTop + 'px';
                    srr.push([snake.offsetLeft,snake.offsetTop]);//每移动1px,就将位置存进数组中
                    num++;//相应的数组长度加1
                }else{
                    clearInterval(timer);
                    console.log("failed!");
                }
               }
                break;
            case 38://上
                clearInterval(timer);
                timer = window.setInterval(runTop,10);
                function runTop(){
                    if(snake.offsetTop > 0){
                        snake.style.top = snake.offsetTop - 1 + 'px';
                        snake.style.left = snake.offsetLeft + 'px';
                        srr.push([snake.offsetLeft,snake.offsetTop]);
                        num++;
                    } else{
                        clearInterval(timer);
                        console.log("failed!");
                    }
                }
                break;
            case 39://右
                clearInterval(timer);
                timer = window.setInterval(runRight,10);
                function runRight(){
                    if(snake.offsetLeft < 430){
                        snake.style.left = snake.offsetLeft + 1 + 'px';
                        snake.style.top = snake.offsetTop + 'px';
                        srr.push([snake.offsetLeft,snake.offsetTop]);
                        num++;
                    } else{
                        clearInterval(timer);
                        console.log("falied!");
                    }
                }
                break;
            case 40://下
                clearInterval(timer);
                timer = window.setInterval(runBottom,10);
                function runBottom(){
                    if(snake.offsetTop < 480) {
                        snake.style.top = snake.offsetTop + 1 + 'px';
                        snake.style.left = snake.offsetLeft + 'px';
                        srr.push([snake.offsetLeft,snake.offsetTop]);
                        num++;
                    }else{
                        clearInterval(timer);
                        console.log("falied!");
                    }
                }
                break;
        }
    }
让食物随机产生
运用了Math类里面的random方法实现食物的随机产生,代码如下,配合碰撞检测使用
function pos(){
    food.style.left = parseInt(Math.random()*430) + 'px';
    food.style.top = parseInt(Math.random()*480) + 'px';
}
碰撞检测
小蛇和食物不碰撞有四种可能,图中黄色是食物,红色是小蛇:
第一种情况:
第二种情况:
第三种情况:
第三种情况
除了以上四种情况以外的都是碰撞的情况,所以代码如下
/*检测碰撞函数*/
function eat(){
    snakeCrashFood(snake,food);
    function snakeCrashFood(obj1,obj2){
        var obj1Left = obj1.offsetLeft;
        var obj1Width = obj1.offsetWidth + obj1.offsetLeft;
        var obj1Top = obj1.offsetTop;
        var obj1Height = obj1.offsetHeight + obj1.offsetTop;
        var obj2Left = obj2.offsetLeft;
        var obj2Width = obj2.offsetWidth + obj2.offsetLeft;
        var obj2Top = obj2.offsetTop;
        var obj2Height = obj2.offsetHeight + obj2.offsetTop;
        if(!((obj1Width<obj2Left)||(obj2Width<obj1Left)||(obj1Height<obj2Top)||(obj2Height<obj1Top))){
            snakeBody = document.createElement("div");
            snakeBody.setAttribute("class","body");
            playground.appendChild(snakeBody);//添加一节新的身体
            pos();//让食物重新随机出现
            setInterval(follow,10);//动态地改变新的身体的位置
        }
    }
}
让身体跟随
在讲让小蛇动起来的那一节的时候,已经提到过了,小蛇每移动1px,都会将此时的位置存入到位置数组中,这个位置数组就是在这一节里面起作用的,能让身体动态地跟随,看下面的图片
红色的头节点移动了40px,此时arr[]数组的长度就为40,那么第二节身体的位置刚好就是arr[20]的位置,第三节的位置就是arr[40]的位置,依次加等20。。。实现的代码如下
/*身体跟随函数*/
function follow(){
    /*获得增加的身体的数组*/
    var bodyNum = document.getElementsByClassName("body");
    score.innerHTML = bodyNum.length;
    var place = 0;
    /*数组每移动1px,新的身体的位置就是相对于前一个身体的第20个数组的位置,后面依次加等*/
    for(var i=0;i<bodyNum.length;i++){
        place += 20;
        bodyNum[i].style.left = srr[num-place][0] + 'px';
        bodyNum[i].style.top = srr[num-place][1] + 'px';
    }
}
原生JavaScript实现的贪吃蛇的更多相关文章
- 原生js写的贪吃蛇网页版游戏特效
		<meta http-equiv="Content-Type" content="text/html; charset=utf-8"> <bo ... 
- javascript 编写的贪吃蛇
		<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ... 
- JavaScript实现的--贪吃蛇
		总的实现思路: 一.效果部分: 1.编写html代码,加上样式. 二.JavaScript部分: 1.利用dom方法创建一块草坪,即活动区域: 2.创建一条蛇,并设置其初始位置: ... 
- JavaScript 小游戏 贪吃蛇
		贪吃蛇 代码: <!DOCTYPE html><html><head> <meta charset="UTF-8"> <met ... 
- JavaScript新手入门    贪吃蛇
		从小就在玩贪吃蛇,但是知道今天自己做了一遍才知道原理的具体的实现步骤. 刚进入界面时显示开始游戏(不重要,本人比较喜欢吹毛求疵) 中间黑色部分为游戏的主要展示部分 主要步骤及源码: body中代码,红 ... 
- javascript实现游戏贪吃蛇
		1.设计蛇:属性有宽.高.方向.状态(有多少节),方法:显示,跑 2.设计食物:属性宽.高 3.显示蛇:根据状态向地图里加元素 4.蛇跑起来:下一节到前一节的位置,蛇头根据方向变,删除原来的蛇,新建蛇 ... 
- JavaScript 面向对象思想 贪吃蛇游戏
		js代码: 游戏的对象 ,食物,蛇 ,游戏控制思路如下 (完整代码在https://github.com/774044859yf/ObjectSnakeGame下载) var snake = { aS ... 
- canvas原生js写的贪吃蛇
		<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ... 
- js贪吃蛇-简单版
		分享个用原生js写的贪吃蛇,最近在学java,按照当年写的 js的思路,转换成java,换汤不换药 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1 ... 
随机推荐
- shadowssock+openvpn 2.2.2 成功*** -- 好文档 - 原理理解+架构
			[root@iZrj9j4lxhjopzii4vhw0lZ 2.0]# grep -v '^#' vars |grep -v '^$'export EASY_RSA="`pwd`" ... 
- 非const引用参数传入不同类型编译不过的理解(拒绝将临时对象绑定为非const的引用的形参是有道理的)
			int f (int & I) { cout<<I<<std::endl; } void main() { long L; f(L); // 编译不过 f((int)L ... 
- unlink与close关系
			close和unlink.以前时候总是不太理解两者的区别,最近看到一篇博客比较详细地描述了二者的本质区别,这里我引用了它的原文. “每一个文件,都可以通过一个struct stat的结 ... 
- ThreadLocal理解
			ThreadLocal 概述 ThreadLocal实例仅作为线程局部变量的==操作类==,以及==线程存储局部变量时的Key==.真正的线程局部变量是存储在各自线程的本地,通过Thread类中的Th ... 
- swoft orm中的坑(针对实体类的属性名称和数据库字段不相等)
			最近在用swoft的orm,发现了一些问题: 首先看下实体类的定义 它的属性名称和所映射的数据库字段名不一致,这个就会导致蛋疼的问题,首先,在我们使用orm的时候,应该使用哪个字段? 我直接说结论,在 ... 
- C语言学习记录_2019.02.08
			\n:换行: \t:制表符,相当于大空格: a[5]={2};<------->a[5]={2,0,0,0,0}; 数组初始化的方法:a[5]={0};即全部初始化为0: 数组初始化的 ... 
- Datax3.0使用说明
			一.datax3.0介绍 1.DataX 是一个异构数据源离线同步工具,致力于实现包括关系型数据库(MySQL.Oracle等).HDFS.Hive.ODPS.HBase.FTP等各种异构数据源之间稳 ... 
- Scala的类继承
			Scala的类继承 extend Scala扩展类的方式和java一样使用extends关键字 class Employee extends Person { } 与java一样,可以在定义的子类重写 ... 
- 做题记录 To 2019.2.13
			2019-01-18 4543: [POI2014]Hotel加强版:长链剖分+树形dp. 3653: 谈笑风生:dfs序+主席树. POJ 3678 Katu Puzzle:2-sat问题,给n个变 ... 
- USACO Section1.2
			section1.1主要包括四道题和两个编程知识介绍.下面将对这6个部分内容进行学习. Your Ride Is Here 这道题没什么难度,读懂题目意思就行:把两个字符串按照题目要求转换成数字,然后 ... 
