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

Posted 凯小默

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了TypeScript教程# 16:ts + webpack + less实现贪吃蛇小游戏相关的知识,希望对你有一定的参考价值。

说明

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

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

项目搭建

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

在这个基础上添加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;

项目界面

界面效果如下:

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 = `$leftpx`;
        this.element.style.top = `$toppx`;
    

export default Food;

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

完成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 = `$valuepx`;
        // 检查蛇头是否撞到身体
        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 = `$valuepx`;
        // 检查蛇头是否撞到身体
        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 = `$Xpx`;
            (this.bodies[i] as HTMLElement).style.top = `$Ypx`;
        
    
    // 检查蛇头是否撞到身体
    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 
       

以上是关于TypeScript教程# 16:ts + webpack + less实现贪吃蛇小游戏的主要内容,如果未能解决你的问题,请参考以下文章

TypeScript教程# 1:TS简介

TypeScript系列教程04编译参数

TypeScript教程# 2:TS开发环境搭建

TypeScript系列教程04编译参数

typescript写给JS老鸟的TS速成教程

TypeScript教程# 5:TS编译选项