LOADING

MiniKano的小窝


 

JavaScript之贪吃的滑稽

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

效果(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;
        }
    </style>
</head>

<body>
    <button id="start">开始游戏</button>
    <div class="map">

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

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

</html>
点赞

发表回复

电子邮件地址不会被公开。必填项已用 * 标注