【TypeScript教程】# 16:ts + webpack + less实现贪吃蛇小游戏

简介: 【TypeScript教程】# 16:ts + webpack + less实现贪吃蛇小游戏

说明

尚硅谷TypeScript教程(李立超老师TS新课)学习笔记。

贪吃蛇源码笔记:https://github.com/kaimo313/typescript-demo/tree/main/greedy-snake



项目搭建

我们以demo3的项目为基础,可以复制一份过来

b9359b63c89042bd864c0fa2c63d2e5c.png


在这个基础上添加less相关的处理

npm i -D less less-loader css-loader style-loader


然后添加postcss处理兼容性问题

npm i -D postcss postcss-loader postcss-preset-env


最后配置webpack

// 设置less文件的处理
{
     test: /\.less$/,
     use: [
         "style-loader",
         "css-loader",
         {
             loader: "postcss-loader",
             options: {
                 postcssOptions: {
                     plugins: [
                         [
                             "postcss-preset-env",
                             {
                                 browsers: "last 2 versions"
                             }
                         ]
                     ]
                 }
             }
         },
         "less-loader"
     ]
 }


能将我们写的less文件编译成css即可:

body {
    background-color: #eee;
    display: flex;
}


c001430494ff4427b5ac4a3fcdb82a40.png


项目界面

界面效果如下:

html的代码

<!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>
</head>
<body>
    <div id="app">
        <div id="stage">
            <div id="snake">
                <div></div>
            </div>
            <div id="food">
                <div></div>
                <div></div>
                <div></div>
                <div></div>
            </div>
        </div>
        <div id="score-panel">
            <div>SCORE: <span id="score">0</span></div>
            <div>LEVEL: <span id="level">1</span></div>
        </div>
    </div>
</body>
</html>


样式代码

@bg-color: #ebebeb;
* {
    margin: 0;
    padding: 0;
    box-sizing: border-box;
}
body {
    font: 20px bold "Courier";
}
#app {
    display: flex;
    flex-flow: column;
    align-items: center;
    justify-content: space-around;
    width: 360px;
    height: 420px;
    background-color: @bg-color;
    margin: 100px auto;
    border-radius: 6px;
    box-shadow: 0 2px 10px 0 rgba(0, 0, 0, 0.4);
    #stage {
        width: 304px;
        height: 304px;
        border: 2px solid #c9c9c9;
        position: relative;
        #snake {
            &>div {
                width: 10px;
                height: 10px;
                background-color: blueviolet;
                border: 1px solid @bg-color;
                position: absolute;
            }
        }
        #food {
            width: 10px;
            height: 10px;
            position: absolute;
            top: 100px;
            left: 40px;
            display: flex;
            flex-flow: row wrap;
            justify-content: space-between;
            align-content: space-between;
            &>div {
                width: 4px;
                height: 4px;
                background-color: green;
                transform: rotate(45deg);
            }
        }
    }
    #score-panel {
        display: flex;
        justify-content: space-between;
        width: 304px;
    }
}


完成Food类

class Food{
    // 定义一个属性表示食物所对应的元素
    element: HTMLElement;
    constructor() {
        // 变量后使用 !:表示类型推断排除null、undefined
        this.element = document.getElementById("food")!;
    }
    // 定义一个获取食物的X轴坐标的方法
    get X() {
        return this.element.offsetLeft;
    }
    // 定义一个获取食物的Y轴坐标的方法
    get Y() {
        return this.element.offsetTop;
    }
    // 修改食物位置
    change() {
        // 生成一个随机的位置
        // 食物的位置最小是0,最大是290,一格是10
        let left = Math.round(Math.random() * 29) * 10;
        let top = Math.round(Math.random() * 29) * 10;
        this.element.style.left = `${left}px`;
        this.element.style.top = `${top}px`;
    }
}
export default Food;


每次调用change都可以修改到食物的位置,让其在stage里随机显示

24ec0b047ba54b5ab538ac018333c4d2.png



完成ScorePanel类

// 定义表示记分牌的类
class ScorePanel{
    score = 0;
    level = 1;
    scoreEle: HTMLElement;
    levelEle: HTMLElement;
    // 最高等级
    maxLevel: number;
    // 多少分升级
    upScore: number;
    constructor(maxLevel: number = 10, upScore: number = 10) {
        this.scoreEle = document.getElementById("score")!;
        this.levelEle = document.getElementById("level")!;
        this.maxLevel = maxLevel;
        this.upScore = upScore;
    }
    // 设置一个加分方法
    addScore() {
        this.scoreEle.innerHTML = `${++this.score}`;
        if(this.score % this.upScore === 0) {
            this.levelUp();
        }
    }
    // 提升等级
    levelUp() {
        if(this.level < this.maxLevel) {
            this.levelEle.innerHTML = `${++this.level}`;
        }
    }
}
export default ScorePanel;

完成Snake类

// 定义蛇类
class Snake{
    // 表示蛇头所对应的元素
    head: HTMLElement;
    // 蛇的身体(包含蛇头)
    bodies: HTMLCollection;
    // 获取蛇的容器
    element: HTMLElement;
    constructor() {
        // 变量后使用 !:表示类型推断排除null、undefined
        this.element = document.querySelector("#snake")!;
        this.head = document.querySelector("#snake > div") as HTMLElement;
        this.bodies = this.element.getElementsByTagName("div");
    }
    // 获取蛇头的X轴坐标
    get X() {
        return this.head.offsetLeft;
    }
    // 获取蛇头的Y轴坐标
    get Y() {
        return this.head.offsetTop;
    }
    // 设置蛇头的X轴坐标
    set X(value: number) {
        if(this.X === value) {
            return;
        }
        // 撞墙
        if(value < 0 || value > 290) {
            throw new Error("蛇撞墙了");
        }
        // 修改x时,是在修改水平坐标, 蛇在左右移动,蛇在向左移动时,不能向右掉头, 反之亦然
        if(this.bodies[1] && (this.bodies[1] as HTMLElement).offsetLeft === value) {
            console.log("水平方向发生了调头");
            value = value > this.X ? this.X - 10 : this.X + 10;
        }
        // 移动身体
        this.moveBody();
        this.head.style.left = `${value}px`;
        // 检查蛇头是否撞到身体
        this.checkHeadBody();
    }
    // 设置蛇头的Y轴坐标
    set Y(value: number) {
        if(this.Y === value) {
            return;
        }
        // 撞墙
        if(value < 0 || value > 290) {
            throw new Error("蛇撞墙了");
        }
        // 修改y时,是在修改垂直坐标, 蛇在上下移动,蛇在向上移动时,不能向下掉头, 反之亦然
        if(this.bodies[1] && (this.bodies[1] as HTMLElement).offsetTop === value) {
            console.log("垂直方向发生了调头");
            value = value > this.Y ? this.Y - 10 : this.Y + 10;
        }
        // 移动身体
        this.moveBody();
        this.head.style.top = `${value}px`;
        // 检查蛇头是否撞到身体
        this.checkHeadBody();
    }
    // 蛇增加身体的方法
    addBody() {
        // 向element添加一个div
        // insertAdjacentElement 将一个给定的元素节点插入到相对于被调用的元素的给定的一个位置。
        // beforeend 只在该元素当中,在该元素最后一个子孩子后面。
        this.element.insertAdjacentElement("beforeend", document.createElement("div"));
    }
    // 蛇身体移动:将后边的身体设置为前边的身体的位置
    moveBody() {
        for(let i = this.bodies.length - 1; i > 0; i--) {
            // 获取前边身体的位置
            let X = (this.bodies[i - 1] as HTMLElement).offsetLeft;
            let Y = (this.bodies[i - 1] as HTMLElement).offsetTop;
            // 将值赋值到当前的身体上面
            (this.bodies[i] as HTMLElement).style.left = `${X}px`;
            (this.bodies[i] as HTMLElement).style.top = `${Y}px`;
        }
    }
    // 检查蛇头是否撞到身体
    checkHeadBody() {
        // 检查所有身体是否和蛇头坐标发生重叠
        for(let i = 1; i < this.bodies.length; i++) {
            let bd = this.bodies[i] as HTMLElement;
            if(this.X === bd.offsetLeft && this.Y === bd.offsetTop) {
                throw new Error("撞到自己了!");
            }
        }
    }
}
export default Snake;


GameControl键盘事件以及使蛇移动

import Food from "./Food";
import Snake from "./Snake";
import ScorePanel from "./ScorePanel";
// 游戏控制器,控制其他的所有类
class GameControl {
    food: Food;
    snake: Snake;
    scorePanel: ScorePanel;
    // 方向
    direction: String = "";
    // 是否结束
    isLive = true;
    constructor() {
        this.food = new Food();
        this.snake = new Snake();
        this.scorePanel = new ScorePanel(10, 1);
        this.init();
    }
    // 游戏初始化
    init() {
        document.addEventListener("keydown", this.keydownHandler.bind(this));
        this.run();
    }
    /**
     * 按下事件
     * ArrowUp Up
     * ArrowRight Right
     * ArrowDown Down
     * ArrowLeft Left
    */
    keydownHandler(event: KeyboardEvent) {
        console.log(event);
        // 检查是否合法
        this.direction = event.key;
    }
    // 控制蛇移动,根据方向移动
    run() {
        /**
         * ArrowUp Up:top -
         * ArrowRight Right:left +
         * ArrowDown Down:top +
         * ArrowLeft Left:left -
        */
        // 蛇现在的坐标
        let X = this.snake.X; 
        let Y = this.snake.Y;
        switch(this.direction){
            case "ArrowUp":
            case "Up":
                Y -= 10;
                break;
            case "ArrowRight":
            case "Right":
                X += 10;
                break;
            case "ArrowDown":
            case "Down":
                Y += 10;
                break;
            case "ArrowLeft":
            case "Left":
                X -= 10;
                break;
        }
        // 检查是否吃到食物
        this.checkEat(X, Y);
        // 修改XY
        try {
            this.snake.X = X;
            this.snake.Y = Y;
        } catch (e) {
            alert(e);
            this.isLive = false;
        }
        this.isLive && setTimeout(this.run.bind(this), 300 - (this.scorePanel.level - 1)*30);
    }
    // 检查是否吃到食物
    checkEat(X:number, Y:number) {
        if(X === this.food.X && Y === this.food.Y) {
            console.log("吃到食物了!");
            // 食物重置
            this.food.change();
            // 分数增加
            this.scorePanel.addScore();
            // 蛇增加一节
            this.snake.addBody();
        }
    }
}
export default GameControl;



实例化GameControl

最后引入控制器实例化即可。

import "./style/index.less";
import GameControl from "./modules/GameControl";
new GameControl();


蛇撞墙和吃食检测效果

c89371be9e15435cb286c0a3a9986b14.png


7504de794342402589eadacbcee548d8.png




目录结构

整体的目录结构如下:

5769c5c55e6f4f4ba6f057653b80a6e7.png


目录
相关文章
|
5天前
|
缓存 JavaScript 前端开发
探讨如何通过一系列优化策略来提升TypeScript与Webpack的构建性能。
【6月更文挑战第11天】本文探讨了优化TypeScript与Webpack构建性能的策略。理解Webpack的解析、构建和生成阶段是关键。优化包括:调整tsconfig.json(如关闭不必要的类型检查)和webpack.config.js选项,启用Webpack缓存,实现增量构建,代码拆分和懒加载。通过这些方法,可以提升构建速度,提高开发效率。
31 0
|
13天前
|
JavaScript 前端开发
37.【TypeScript 教程】TSLint 与 ESLint
37.【TypeScript 教程】TSLint 与 ESLint
11 0
|
13天前
|
JavaScript 编译器 IDE
36.【TypeScript 教程】tsconfig.json 配置
36.【TypeScript 教程】tsconfig.json 配置
10 0
|
13天前
|
JavaScript 编译器
35.【TypeScript 教程】编译选项
35.【TypeScript 教程】编译选项
11 2
|
13天前
|
JavaScript 前端开发 编译器
34.【TypeScript 教程】声明合并
34.【TypeScript 教程】声明合并
14 0
|
13天前
|
JavaScript 编译器
33.【TypeScript 教程】命名空间
33.【TypeScript 教程】命名空间
17 2
|
13天前
|
JavaScript 编译器 开发者
32.【TypeScript 教程】模块
32.【TypeScript 教程】模块
10 0
|
13天前
|
JavaScript 编译器
31.【TypeScript 教程】混入(Mixins)
31.【TypeScript 教程】混入(Mixins)
15 3
|
13天前
|
JavaScript Java API
30.【TypeScript 教程】Reflect Metadata
30.【TypeScript 教程】Reflect Metadata
12 4
|
13天前
|
JavaScript 监控 编译器
29.【TypeScript 教程】装饰器(Decorator)
29.【TypeScript 教程】装饰器(Decorator)
9 0