百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

使用canvas实现简单的贪吃蛇游戏,html+css+js

yund56 2025-07-01 01:21 1 浏览

一.话不多,先瞅效果:

又在别的地方嫖到了这个效果研究了亿下下,制作过程如下(超详细):

二.实现过程(源码在最后):

1.定义canvas标签:

 <canvas id="canvas"></canvas>

2.基本css样式:

#canvas{
            position: absolute;
            top: 50%;
            left: 50%;
            transform: translate(-50%,-50%);
            box-shadow:  0 0 10px rgb(150, 150, 150);     
        }

position: absolute; 绝对定位。
top: 50%;
left: 50%;
transform: translate(-50%,-50%); 居中。
box-shadow: 0 0 10px rgb(150, 150, 150); 阴影。

3.开始js部分,获取标签:

var canvas = document.querySelector("#canvas");
var ctx = canvas.getContext('2d');            

4.定义基本变量:

        //画布宽
        var wide=600;
        //画布高
        var high=600;
        // 变量,判断一次渲染中只识别按键一次
        var kd = 0;
        //当前分数
        var fraction =0;
        //速度,就是执行定时器的时间参数
        var speed = 250;
        // 蛇的初始颜色 红色
        var yanse = `red`;
        // 蛇数组,组成蛇的每一个方块
        var snake = [];
        // 食物数组
        var food = {};
        // 蛇的移动方向,x轴:1为向右,-1为向左;y轴:1为向下,-1为向上 。不能斜着走,所以0为某轴无方向。
        var diretion = {
            x:-1,
            y:0
        }
        // 给画布宽高赋值 打算画一个长宽都是30个20px的方块画布
        canvas.width = wide;
        canvas.height = high;

5. 初始化:

function chushi(){            
        //蛇初始长度为3个方块,位置如下(这个随意)
            for(let i =0;i<3;i++){
                snake.push({
                    x: i+10,
                    y: 10
                })
            }
            // 给食物一个随机位置和随机颜色
            food = {
                x: parseInt(Math.random()*30),
                y: parseInt(Math.random()*30),
                color:`rgb(${Math.random()*255},${Math.random()*255},${Math.random()*255})`
            }
        }

6. 绘制图形:

  // 绘制图形
        function draw(){
            // 绘制显示当前分数的文字
            ctx.fillStyle = 'rgba(255,255,255,0.5)';
                ctx.font="50px 仿宋";
                ctx.textAlign = 'center'; 
                ctx.fillText("你的分数为:"+fraction+" 分",300,300);

             // 绘制方格,长宽都是30个,都是19px*19px的方格   
            for(let i=0;i<30;i++){
                for(let j=0;j<30;j++){
                    ctx.fillStyle = 'rgba(255, 255, 255,.3)';
                    ctx.fillRect(i*20,j*20,19,19);
                }
            }
              // 绘制蛇
            for(let i=0;i<snake.length;i++){
                temp = snake[i];
                ctx.fillStyle = yanse;
                ctx.fillRect(temp.x*20,temp.y*20,19,19);
                // 判断蛇头(第一个方块)是否与身体某个方块重合 ,就是头撞到身体
                if(temp.x==snake[0].x&&temp.y==snake[0].y&&i!=0){
                    // 游戏结束,重新给初始化
                    alert('游戏结束~点击确认再来一次~'); 
                    fraction = 0;
                    snake.length=0;  
                    chushi();
                    
                }
            }
               // 绘制食物,绘制一个圆形
                ctx.beginPath();
                ctx.fillStyle = food.color;
                ctx.arc(food.x*20+9.5,food.y*20+9.5,7,0,Math.PI*2,false);
                ctx.stroke();
                ctx.fill();
                ctx.closePath();
      
               // 给蛇头绘制一个字符,☆ ,好区分头尾 ,也可省略
                ctx.fillStyle = 'yellow';
                ctx.font="15px 仿宋";
                ctx.textAlign = "start";
                ctx.fillText("☆",snake[0].x*20+2,snake[0].y*20+14.5);
        }

7.更新位置:

 //更新
        function update(){
            // 建一个对象head,这个为蛇的新头,通过绘制新头,去掉尾部实现移动效果
            var head = {};
            //判断蛇头是否遇到边界,到边界则在另一边重新绘制 x轴
            switch (snake[0].x+diretion.x){
                case -1: head.x=29;break;
                case 30: head.x=0;break;
                // 没到边界则为当前位置加方向
                default:  head.x = snake[0].x+diretion.x;
            }
           //判断蛇头是否遇到边界,到边界则在另一边重新绘制 y轴
            switch (snake[0].y+diretion.y){
                case -1: head.y=29;break;
                case 30: head.y=0;break;
                // 没到边界则为当前位置加方向
                default:  head.y = snake[0].y+diretion.y;
            }
            // 判断新蛇头是否与食物重合,就是吃到食物
           if(head.x==food.x&&head.y==food.y){
               //蛇的颜色为吃到食物的颜色
            yanse = food.color;
             // 重新给食物初始化
            food = {
                x: parseInt(Math.random()*30),
                y: parseInt(Math.random()*30),
                color:`rgb(${Math.random()*255},${Math.random()*255},${Math.random()*255})`
            }
            //在蛇尾添加一节
            let temp = snake[length-1];
            snake.push(temp);
            fraction+=1;
            // 吃完食物速度加快
            if(speed>80){
                //定时器间隔减10
                speed = speed-10;
                // 清除原来定时器,重新绘制
                clearInterval(time);
                 time = setInterval(function () {
                    kd = 0;
                    ctx.clearRect(0, 0, wide, high);
                    update();
                    draw();

                }, speed);
            
            }
           
           }
            //添加新头
            snake.splice(0,0,head);
            //去掉尾部
            snake.pop();
                     
        }

8.判断点击键盘事件:

 //判断点击事件 
        document.addEventListener('keydown', event=>{
              switch (event.keyCode){
                  // 按了向上键
                  case 38:
                      // 判断当前不是向下移动与还没按过键,否则蛇会重叠
                      if(diretion.y!=1&&kd==0){
                          // 重新给移动方向赋值
                        diretion.x=0;
                        diretion.y=-1;
                        kd=1;
                      }
                      break;
                      // 下面以此类推一样的原理
                  case 39:
                      if(diretion.x!=-1&&kd==0){
                        diretion.x=1;
                        diretion.y=0;
                        kd=1;
                      }
                      break;
                  case 40:
                      if(diretion.y!=-1&&kd==0){                     
                      diretion.x=0;
                      diretion.y=1;
                      kd=1;
                      }
                      break; 
                  case 37:
                      if(diretion.x!=1&&kd==0){
                        diretion.x=-1;
                        diretion.y=0;
                        kd=1;
                      }
                      break;        

              }
        })

9.设置定时器,开始动画:

 chushi();    
            var time = setInterval(function(){
            kd=0;
            ctx.clearRect(0,0,wide,high);
            update();
            draw();      
        
        },speed);

三.完整代码:

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <style>
        *{
            margin: 0;
            padding: 0;
            box-sizing: border-box;
        }
        body{
            height: 100vh;
        }
        video{
            position: fixed;
            z-index: -10;
            width: 100%;
            height: 100%;
            object-fit: cover;
        }
        #canvas{
            position: absolute;
            top: 50%;
            left: 50%;
            transform: translate(-50%,-50%);
            box-shadow:  0 0 10px rgb(150, 150, 150);     
        }
    </style>
</head>
<body>
    <video src="video/rain.mp4" muted autoplay loop></video>
    <canvas id="canvas"></canvas>
    <script>
        var canvas = document.querySelector("#canvas");
        var ctx = canvas.getContext('2d');

        //画布宽 
        var wide=600;
        //画布高
        var high=600;
        // 变量,判断一次渲染中只识别按键一次
        var kd = 0;
        //当前分数
        var fraction =0;
        //速度,就是执行定时器的时间参数
        var speed = 250;
        // 蛇的初始颜色 红色
        var yanse = `red`;
        // 蛇数组,组成蛇的每一个方块
        var snake = [];
        // 食物数组
        var food = {};
        // 蛇的移动方向,x轴:1为向右,-1为向左;y轴:1为向下,-1为向上 。不能斜着走,所以0为某轴无方向。
        var diretion = {
            x:-1,
            y:0
        }
        // 给画布宽高赋值 打算画一个长宽都是30个20px的方块画布
        canvas.width = wide;
        canvas.height = high;

        function chushi(){            
        //蛇初始长度为3个方块,每个位置如下(这个随意)
            for(let i =0;i<3;i++){
                snake.push({
                    x: i+10,
                    y: 10
                })
            }
            // 给食物一个随机位置和随机颜色
            food = {
                x: parseInt(Math.random()*30),
                y: parseInt(Math.random()*30),
                color:`rgb(${Math.random()*255},${Math.random()*255},${Math.random()*255})`
            }
        }

        // 绘制图形
        function draw(){
            // 绘制显示当前分数的文字
            ctx.fillStyle = 'rgba(255,255,255,0.5)';
                ctx.font="50px 仿宋";
                ctx.textAlign = 'center'; 
                ctx.fillText("你的分数为:"+fraction+" 分",300,300);

             // 绘制方格,长宽都是30个,都是19px*19px的方格   
            for(let i=0;i<30;i++){
                for(let j=0;j<30;j++){
                    ctx.fillStyle = 'rgba(255, 255, 255,.3)';
                    ctx.fillRect(i*20,j*20,19,19);
                }
            }
              // 绘制蛇
            for(let i=0;i<snake.length;i++){
                temp = snake[i];
                ctx.fillStyle = yanse;
                ctx.fillRect(temp.x*20,temp.y*20,19,19);
                // 判断蛇头(第一个方块)是否与身体某个方块重合 ,就是头撞到身体
                if(temp.x==snake[0].x&&temp.y==snake[0].y&&i!=0){
                    // 游戏结束,重新给初始化
                    alert('游戏结束~点击确认再来一次~'); 
                    fraction = 0;
                    snake.length=0;  
                    chushi();
                    
                }
            }
               // 绘制食物,绘制一个圆形
                ctx.beginPath();
                ctx.fillStyle = food.color;
                ctx.arc(food.x*20+9.5,food.y*20+9.5,7,0,Math.PI*2,false);
                ctx.stroke();
                ctx.fill();
                ctx.closePath();
                /* var img = new Image();
                img.src = "img/snake/orange.png";
                img.onload = function(){
                    ctx.drawImage(img,food.x*20,food.y*20,19,19);
                }  */
      
               // 给蛇头绘制一个字符,☆ ,好区分头尾 ,也可省略
                ctx.fillStyle = 'yellow';
                ctx.font="15px 仿宋";
                ctx.textAlign = "start";
                ctx.fillText("☆",snake[0].x*20+2,snake[0].y*20+14.5);
        }

        //更新
        function update(){
            // 建一个对象head,这个为蛇的新头,通过绘制新头,去掉尾部实现移动效果
            var head = {};
            //判断蛇头是否遇到边界,到边界则在另一边重新绘制 x轴
            switch (snake[0].x+diretion.x){
                case -1: head.x=29;break;
                case 30: head.x=0;break;
                // 没到边界则为当前位置加方向
                default:  head.x = snake[0].x+diretion.x;
            }
           //判断蛇头是否遇到边界,到边界则在另一边重新绘制 y轴
            switch (snake[0].y+diretion.y){
                case -1: head.y=29;break;
                case 30: head.y=0;break;
                // 没到边界则为当前位置加方向
                default:  head.y = snake[0].y+diretion.y;
            }
            // 判断新蛇头是否与食物重合,就是吃到食物
           if(head.x==food.x&&head.y==food.y){
               //蛇的颜色为吃到食物的颜色
            yanse = food.color;
             // 重新给食物初始化
            food = {
                x: parseInt(Math.random()*30),
                y: parseInt(Math.random()*30),
                color:`rgb(${Math.random()*255},${Math.random()*255},${Math.random()*255})`
            }
            //在蛇尾添加一节
            let temp = snake[length-1];
            snake.push(temp);
            fraction+=1;
            // 吃完食物速度加快
            if(speed>80){
                //定时器间隔减10
                speed = speed-10;
                // 清除原来定时器,重新绘制
                clearInterval(time);
                 time = setInterval(function () {
                    kd = 0;
                    ctx.clearRect(0, 0, wide, high);
                    update();
                    draw();

                }, speed);
            
            }
           
           }
            //添加新头
            snake.splice(0,0,head);
            //去掉尾部
            snake.pop();
                     
        }

       //判断点击事件 
        document.addEventListener('keydown', event=>{
              switch (event.keyCode){
                  // 按了向上键
                  case 38:
                      // 判断当前不是向下移动与还没按过键,否则蛇会重叠
                      if(diretion.y!=1&&kd==0){
                          // 重新给移动方向赋值
                        diretion.x=0;
                        diretion.y=-1;
                        kd=1;
                      }
                      break;
                      // 下面以此类推一样的原理
                  case 39:
                      if(diretion.x!=-1&&kd==0){
                        diretion.x=1;
                        diretion.y=0;
                        kd=1;
                      }
                      break;
                  case 40:
                      if(diretion.y!=-1&&kd==0){                     
                      diretion.x=0;
                      diretion.y=1;
                      kd=1;
                      }
                      break; 
                  case 37:
                      if(diretion.x!=1&&kd==0){
                        diretion.x=-1;
                        diretion.y=0;
                        kd=1;
                      }
                      break;        

              }
        })

        chushi();    
            var time = setInterval(function(){
            kd=0;
            ctx.clearRect(0,0,wide,high);
            update();
            draw();      
        
        },speed);
        
    </script>
</body>
</html>

相关推荐

豆包编程能力升级:支持HTML代码实时预览、交互

IT之家3月19日消息,IT之家从豆包官方获悉,豆包宣布AI编程功能迎来三项升级,包括HTML预览、Python运行、生成完整项目。据介绍,目前豆包支持HTML代码实时预览和交互...

1898款游戏!80、90回忆杀,重温旧梦,快速搭建中文DOS游戏服务

本内容来源于@什么值得买APP,观点仅代表作者本人|作者:羊刀仙大家好,我是羊刀仙。本期来介绍一个特别情怀向的游戏项目:chinese-dos-games。这套包含1898款经典中文DOS游戏的合集...

利用 SVG 文件内的 HTML 代码进行网络钓鱼攻击

随着时间的推移,网络钓鱼攻击的技术越来越精妙,旨在欺骗用户并规避安全措施。攻击者会使用欺骗性的URL重定向策略,例如将恶意网站地址附加到看似安全的链接后,在PDF中嵌入链接,以及发送HTML...

aardio + AI 大模型自动编写 Python 代码、网页前端代码的经验与技巧

在AI时代,老式的编程习惯完全被颠覆。原来可能要一大堆插件或工具辛苦堆出来的程序,现在只要把AI调教好了就行。aardio支持调用十几种编程语言,这很适合发挥AI大模型的优势。对于AI...

用AI制作游戏就是如此简单!

很多人不知道如何利用AI提高效率,不知道AI能帮我们做什么,其实可以让我们实现很多自己根本不懂的领域取得直观体验,比如利用DS或者豆包,输入“我想做一个简单的单机俄罗斯方块游戏”,AI会给出phtho...

不会写代码?教你用DeepSeek 三步做出小游戏

如今,借助人工智能技术,哪怕你完全看不懂代码,也能通过DeepSeek制作出属于自己的网页版大鱼吃小鱼游戏。接下来,就为大家详细介绍制作过程。第一步、向DeepSeek描述需求为何要做网页版的...

《暗黑1》被移植成网页游戏 可在浏览器上玩了

《暗黑1》,这款1996年发售的“鼠标杀手”砍杀游戏,现在可以在浏览器上玩了。国外专注暴雪游戏的Rivsoft分享了一个《暗黑1》的共享版本,该版本只包含地下城的头2个地区和三个角色职业中的一个。不...

网页代码过滤 轻松获取专辑目录

通过过滤网页代码,可以将网页上显示不全的长文件名列表完整地提取出来。我有一个含有75个视频文件的《中医诊断学》课件,文件名是以01.RMVB、02.RMVB……75.RMVB这种格式命名的。我希望能找...

IDEA 2021首个大版本发布,Java开发者感动哭了(附新亮点演示)

工欲善其事,必先利其器!就在不久之前,Java领域的开发神器IntelliJIDEA终于迎来2021年的一个重要的大版本更新:IntelliJIDEA2021.1。现如今大量的Java开发者深度...

View Source:在 iOS 上轻松查看网页源代码

在移动互联网时代,移动端的应用和web体验都尤为重要,在PC上有很多web前端工具可以选择,而在移动端貌似就少之又少了,在NEXT出现的ViewSource能帮你在iOS上查看...

当我们《寻找房祖名》,我们能找到什么?

游戏葡萄原创专稿,未经允许请勿转载柯震东,因为在九把刀电影《那些年我们追过的女孩》中饰演男主角柯景腾而走红的台湾影星,在昨天被爆出了和著名演员成龙之子房祖名吸毒被抓的丑闻,一时间相关讨论席卷社交网络。...

多用途游戏娱乐新闻网站HTML5模板

Retnews是一个响应式的HTML新闻,博客,杂志网站模板,可以使用这套前端模板简约很多设计的工作。模板有许多特性适合流行的主题商业、时尚,游戏,娱乐,生活方式、体育、科技、政治、旅行、天气、视频等...

简约好看的个人引导页HTML源码下载

源码介绍一款非常简约好看的个人引导页HTML源码,非常适合个人主页以及个人导航使用,纯HTML不需要数据库,上传服务器即可使用!...

教你如何在微信公共平台上插入小游戏(图文教程)

很多玩微信公共平台的朋友都想在公共平台上面插入几个小游戏,用来跟用户之间互动,这里花生来分享一下如何在微信公共平台上插入游戏,以及如何制作html5微信小游戏。首先是找游戏,总共有三个方法,本人比较倾...

html5重力感应剖析附源码

下面是测试html5重力感应的demohttp://bbs.qietu.com/html/zhongli/http://www.qietu.com/html/f2/qqqianbao/demo2是切图...