这玩意光修bug都花了三小时 最终还是做出来了
效果(WASD操控)
代码
<!DOCTYPE html> <html lang="en">
<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>贪吃蛇</title> <link rel="stylesheet" href="./css/style.css"> <style> body { padding: 0; margin: 0; overflow: hidden; }
.map { position: relative; background-color: #313131; width: 100vw; height: 100vh; } .map .snake, .map .food { position: absolute; z-index: 1; display: block; width: 20px; height: 20px; background-color: red; border-radius: 50%; overflow: hidden; transition: all 1s; } #start { z-index: 10000; left: 50%; top: 50%; transform: translate(-50%, -50%); width: 140px; height: 40px; background-color: #00AEEC; border: none; border-radius: 20px; color: #eee; position: absolute; } #start:active { background-color: #00a1ec; } .snake:nth-child(2) { z-index: 111; transform: scale(1.5); background-image: url(); background-size: contain; } .sad { background-image: url() !important; } .map .food { left: 30px; transform: translate(50%, 50%); background-image: url(); background-size: contain; } &lt;/style&gt;
</head>
<body> <button id="start">开始游戏</button> <div class="map">
&lt;!-- &lt;span class="snake"&gt;&lt;/span&gt; --&gt; &lt;/div&gt; &lt;script&gt; //随机工具类 var Tools = { getRandom: function(min, max) { //min和max都能取到 return Math.floor(Math.random() * (Math.floor(max) - Math.ceil(min) + 1)) + min } }; &lt;/script&gt; &lt;script&gt; //独立作用域 (() =&gt; { //记录上次创建的食物 let elements = []; // 食物对象 function Food(options) { //防止传参为空 options = options || {}; this.x = options.x || 0; this.y = options.y || 0; this.width = options.width || 40; this.height = options.height || 40; this.color = options.color || 'transparent'; } //渲染食物 Food.prototype.render = function(map) { //删除之前创建的食物 remove(); //随机设置食物位置 this.x = Tools.getRandom(0, (map.offsetWidth / this.width - 1) * this.width); this.y = Tools.getRandom(0, (map.offsetHeight / this.height - 1) * this.height); //动态创建元素,我选择用span来显示食物 let span = document.createElement('span'); //设置样式s span.style.left = this.x + 'px'; span.style.top = this.y + 'px'; span.style.backgroundColor = this.color; span.classList.add('food'); map.appendChild(span); //保存食物 elements.push(span); }; //删除食物 function remove() { while (elements.length &gt; 0) { elements[elements.length - 1].parentNode.removeChild(elements[elements.length - 1]); elements.splice(elements.length - 1, 1); } } //把构造函数添加到window对象下,外部就可以访问了 window.Food = Food; })(); &lt;/script&gt; &lt;script&gt; //蛇对象 (() =&gt; { //记录上次创建的蛇 let elements = []; function Snake(options) { //防止传参为空 options = options || {}; //蛇节 this.width = options.width || 20; this.height = options.height || 20; //蛇的移动方向 this.direction = options.direction || 'right'; //蛇身 this.body = [{ x: 12, y: 2, color: 'transparent' }, { x: 11, y: 2, color: `rgba(${Tools.getRandom(0,255)},${Tools.getRandom(0,255)},${Tools.getRandom(0,255)},1)` }, ]; }; //渲染 Snake.prototype.render = function(map) { //删除之前创建的蛇 remove(); //渲染蛇节 for (let i = 0, len = this.body.length; i &lt; len; i++) { //蛇节 let obj = this.body[i]; //创建元素 let span = document.createElement('span'); //设置样式 span.style.width = this.width + 'px'; span.style.height = this.height + 'px'; //obj.x * this.width 表示x个蛇节宽度这么多的距离 span.style.left = (obj.x * this.width) + 'px'; span.style.top = (obj.y * this.height) + 'px'; span.style.backgroundColor = obj.color; span.classList.add('snake'); map.appendChild(span); elements.push(span); } }; //删除蛇方法 function remove() { while (elements.length &gt; 0) { elements[elements.length - 1].parentNode.removeChild(elements[elements.length - 1]); elements.splice(elements.length - 1, 1); } } //蛇移动 Snake.prototype.move = function(food, map) { //控制蛇的身体移动(当前蛇节到上一个蛇节的位置) for (let i = this.body.length - 1; i &gt; 0; i--) { this.body[i].x = this.body[i - 1].x; this.body[i].y = this.body[i - 1].y; } //控制蛇头的移动 //判断蛇移动的方向 let head = this.body[0]; switch (this.direction) { case 'right': head.x += 0.2; break; case 'left': head.x -= 0.2; break; case 'up': head.y -= 0.2; break; case 'down': head.y += 0.2; break; } //判断蛇头是否在食物的坐标范围内 let snakeX = (head.x * (this.width)) let snakeY = (head.y * (this.height)); let leftPaddingIsTrue = ((snakeX &gt;= food.x) &amp;&amp; (snakeX &lt;= (food.x + food.width))) &amp;&amp; ((snakeY &gt;= food.y) &amp;&amp; (snakeY &lt;= (food.y + food.height))) let rightPaddingIsTrue = ((snakeX + this.width / 2 &gt;= food.x) &amp;&amp; (snakeX + this.width / 2 &lt;= (food.x + food.width))) &amp;&amp; ((snakeY + this.height / 2 &gt;= food.y) &amp;&amp; (snakeY + this.height / 2 &lt;= (food.y + food.height))) if (leftPaddingIsTrue || rightPaddingIsTrue) { //蛇加一节 //复制蛇的最后一节 let lastNode = this.body[this.body.length - 1]; for (let i = 0; i &lt; 10; i++) { this.body.push({ x: lastNode.x, y: lastNode.y, color: `rgba(${Tools.getRandom(0,255)},${Tools.getRandom(0,255)},${Tools.getRandom(0,255)},.7)` }); } //随机在地图上增加食物 food.render(map); } } //把构造函数添加到window对象下,外部就可以访问了 window.Snake = Snake; })(); &lt;/script&gt; &lt;script&gt; //主游戏对象 (() =&gt; { var that; //创建食物和蛇对象的构造函数 function Game(map) { this.food = new Food(); this.snake = new Snake(); this.map = map; that = this; } //开始游戏方法 Game.prototype.start = function() { //渲染食物和蛇 this.food.render(this.map); this.snake.render(this.map); runSnake(); //开始游戏逻辑 //让蛇动起来 //通过键盘控制蛇移动的方向 //当蛇遇到食物 做相应的处理 //当蛇遇到边界游戏结束 }; //让蛇润起来 function runSnake() { var timer = setInterval(function() { //这里请注意this指向 that.snake.move(that.food, that.map); that.snake.render(that.map); //当蛇遇到边界游戏结束 //只需要判断蛇头的坐标即可 let headX = that.snake.body[0].x; let headY = that.snake.body[0].y; let maxX = Math.floor(that.map.offsetWidth / that.snake.width) - 0.5; let maxY = Math.floor(that.map.offsetHeight / that.snake.height) - 0.5; if (headX &gt;= maxX || headY &gt;= maxY) { //设置异步,避免alert卡帧 setTimeout(() =&gt; { // alert('你创到墙啦,游戏结束'); document.querySelector('.snake:nth-child(2)').classList.add('sad'); btn.style.display = 'block'; }, 100); clearInterval(timer); } if (headX &lt;= 0 || headY &lt;= 0) { setTimeout(() =&gt; { // alert('你创到墙啦,游戏结束'); document.querySelector('.snake:nth-child(2)').classList.add('sad'); btn.style.display = 'block'; }, 100); clearInterval(timer); } //判断是否吃到了自己 snakeBody = that.snake.body; for (let i = 1; i &lt; snakeBody.length; i++) { if ((snakeBody[0].x == snakeBody[i].x) &amp;&amp; (snakeBody[0].y == snakeBody[i].y)) { setTimeout(() =&gt; { // alert('你咬到自己啦,游戏结束'); document.querySelector('.snake:nth-child(2)').classList.add('sad'); }, 100); //clearInterval(timer); break; } } }, 10); } //蛇控制 function keyControl() { document.addEventListener('keyup', function(e) { switch (e.code) { case 'KeyD': //蛇不能180度掉头 if (that.snake.direction != 'left') { that.snake.direction = 'right'; } break; case 'KeyA': if (that.snake.direction != 'right') { that.snake.direction = 'left'; } break; case 'KeyS': if (that.snake.direction != 'up') { that.snake.direction = 'down'; } break; case 'KeyW': if (that.snake.direction != 'down') { that.snake.direction = 'up'; } break; } }, false); //事件冒泡 } keyControl(); //当蛇遇到食物 做相应的处理 window.Game = Game; })(); &lt;/script&gt; &lt;script&gt; var btn = document.querySelector('#start'); btn.addEventListener('click', function() { this.style.display = 'none'; var map = document.querySelector('.map'); var game = new Game(map); game.start(); }) &lt;/script&gt;
</body>
</html>