51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

JavaScript之贪吃的滑稽

这玩意光修bug都花了三小时 :kano-2d-19: 最终还是做出来了

效果(WASD操控)


代码

<!DOCTYPE html>
<html lang="en">

&lt;head&gt; &lt;meta charset="UTF-8"&gt; &lt;meta http-equiv="X-UA-Compatible" content="IE=edge"&gt; &lt;meta name="viewport" content="width=device-width, initial-scale=1.0"&gt; &lt;title&gt;贪吃蛇&lt;/title&gt; &lt;link rel="stylesheet" href="./css/style.css"&gt; &lt;style&gt; 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;
    }
&amp;lt;/style&amp;gt;

&lt;/head&gt;

&lt;body&gt; &lt;button id="start"&gt;开始游戏&lt;/button&gt; &lt;div class="map"&gt;

    &amp;lt;!-- &amp;lt;span class=&quot;snake&quot;&amp;gt;&amp;lt;/span&amp;gt; --&amp;gt;
&amp;lt;/div&amp;gt;
&amp;lt;script&amp;gt;
    //随机工具类
    var Tools = {
        getRandom: function(min, max) {
            //min和max都能取到
            return Math.floor(Math.random() * (Math.floor(max) - Math.ceil(min) + 1)) + min
        }

    };
&amp;lt;/script&amp;gt;
&amp;lt;script&amp;gt;
    //独立作用域
    (() =&amp;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 &amp;gt; 0) {
                elements[elements.length - 1].parentNode.removeChild(elements[elements.length - 1]);
                elements.splice(elements.length - 1, 1);
            }
        }
        //把构造函数添加到window对象下,外部就可以访问了
        window.Food = Food;
    })();
&amp;lt;/script&amp;gt;
&amp;lt;script&amp;gt;
    //蛇对象
    (() =&amp;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 &amp;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 &amp;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 &amp;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 &amp;gt;= food.x) &amp;amp;&amp;amp; (snakeX &amp;lt;= (food.x + food.width))) &amp;amp;&amp;amp; ((snakeY &amp;gt;= food.y) &amp;amp;&amp;amp; (snakeY &amp;lt;= (food.y + food.height)))
            let rightPaddingIsTrue = ((snakeX + this.width / 2 &amp;gt;= food.x) &amp;amp;&amp;amp; (snakeX + this.width / 2 &amp;lt;= (food.x + food.width))) &amp;amp;&amp;amp; ((snakeY + this.height / 2 &amp;gt;= food.y) &amp;amp;&amp;amp; (snakeY + this.height / 2 &amp;lt;= (food.y + food.height)))
            if (leftPaddingIsTrue || rightPaddingIsTrue) {
                //蛇加一节
                //复制蛇的最后一节
                let lastNode = this.body[this.body.length - 1];
                for (let i = 0; i &amp;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;
    })();
&amp;lt;/script&amp;gt;
&amp;lt;script&amp;gt;
    //主游戏对象
    (() =&amp;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 &amp;gt;= maxX || headY &amp;gt;= maxY) {
                    //设置异步,避免alert卡帧
                    setTimeout(() =&amp;gt; {
                        // alert('你创到墙啦,游戏结束');
                        document.querySelector('.snake:nth-child(2)').classList.add('sad');
                        btn.style.display = 'block';
                    }, 100);
                    clearInterval(timer);
                }
                if (headX &amp;lt;= 0 || headY &amp;lt;= 0) {
                    setTimeout(() =&amp;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 &amp;lt; snakeBody.length; i++) {
                    if ((snakeBody[0].x == snakeBody[i].x) &amp;amp;&amp;amp; (snakeBody[0].y == snakeBody[i].y)) {
                        setTimeout(() =&amp;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;
    })();
&amp;lt;/script&amp;gt;
&amp;lt;script&amp;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();
    })
&amp;lt;/script&amp;gt;

&lt;/body&gt;

&lt;/html&gt;


赞(2)
未经允许不得转载:工具盒子 » JavaScript之贪吃的滑稽