【Unity】RPG2D龙城纷争(二)关卡、地块

更新日期:2024年6月12日。
项目源码:在第四章发布

索引

  • 简介
    • 地块(Block)
      • 一、定义地块类
      • 二、地块类型
      • 三、地块渲染
      • 四、地块索引
    • 关卡(Level)
      • 一、定义关卡类
      • 二、关卡基础属性
      • 三、地块集合
      • 四、关卡初始化
      • 五、关卡销毁
      • 六、回合制逻辑

简介

本章我们将从零开始实现关卡,不过,我们的目的是实现关卡与地块的基本逻辑,更复杂的功能我们打算在开发关卡编辑器时再做涉及,正所谓一步一个坎,就没有迈不过去的高山。

地块(Block)

地块作为关卡的组成元素,他是一个正方形的格子(当然也可以是菱形,针对斜视角游戏,不过这不在框架的支持范畴内,但做出一定的修改即可实现),每一个地块都有属于他自己的一些属性。

一、定义地块类

首先,我们定义地块类Block

    /// <summary>
    /// 地块
    /// </summary>
    [DisallowMultipleComponent]
    public class Block : HTBehaviour
    {
        
    }

Block继承至HTBehaviour,使得它可以挂载到游戏物体上,然后作为一个地块对象(为不需要一个物体上多次挂载的组件定义DisallowMultipleComponent是一个好习惯,这将提高容错率,如果你的代码会给别人使用的话)。

很多同学热衷于在类继承链上规避MonoBehaviour,其实完全没有这个必要,只要你不滥用MonoBehaviour的生命周期,它将提供给你几大优势:
1.属性可编辑(Inspector面板);
2.属性可调试(Inspector面板);
3.对象可追踪(Scene里面有则有,无则无);
4.对象可销毁(当确定不再使用时,Destroy它,而不用置null后交给GC)。

二、地块类型

经过深思熟虑,我们将地块类型划分为如下几种:

  • 地面
  • 山体
  • 森林
  • 湖泊
  • 雪地
  • 障碍

编写代码:

    /// <summary>
    /// 地块
    /// </summary>
    [DisallowMultipleComponent]
    public class Block : HTBehaviour
    {
        /// <summary>
        /// 类型
        /// </summary>
        [Label("类型")] public BlockType Type;
    }

/// <summary>
/// 地块类型
/// </summary>
public enum BlockType
{
    /// <summary>
    /// 地面
    /// </summary>
    [Remark("地面")]
    Ground = 0,
    /// <summary>
    /// 山体
    /// </summary>
    [Remark("山体")]
    Moutain = 1,
    /// <summary>
    /// 森林
    /// </summary>
    [Remark("森林")]
    Forest = 2,
    /// <summary>
    /// 湖泊
    /// </summary>
    [Remark("湖泊")]
    Water = 3,
    /// <summary>
    /// 雪地
    /// </summary>
    [Remark("雪地")]
    Snow = 4,
    /// <summary>
    /// 障碍
    /// </summary>
    [Remark("障碍")]
    Obstacle = 5
}

然后,我们设计每种类型的地块拥有的交互权限如下:

类型角色可行走角色可攻击(站在上面的敌人或穿过它攻击其他敌人)
地面
山体受限(拥有飞檐走壁可行走,行走速度减1/2
森林受限(行走速度减1/2
湖泊受限(拥有踏水神行可行走,行走速度减1/2
雪地受限(行走速度减2/3
障碍受限(拥有隔山打牛可穿透障碍进行攻击

需注意的是,任意地块,当上面站有敌人时,将不可行走,不可跨越,当站有队友时,不可行走,可跨越。

也即是说,我们可以使用角色摆出特定的阵型,以拦住敌方的行走路线,或达到包围的效果。

三、地块渲染

地块的渲染我们决定选择SpriteRenderer,且一个地块仅渲染一张图片,那么,在Block类中需要持有地块渲染器的引用,我们添加代码:

    /// <summary>
    /// 地块
    /// </summary>
    [DisallowMultipleComponent]
    public class Block : HTBehaviour
    {
        /// <summary>
        /// 目标渲染器
        /// </summary>
        [Label("目标渲染器")] public SpriteRenderer Target;
        /// <summary>
        /// 类型
        /// </summary>
        [Label("类型")] public BlockType Type;
    }

四、地块索引

然后,我们想一想地块还需要些什么属性,哦对了,我想我们在后续一定会需要检索地块(也即是根据索引找到一个地块),而我们的关卡采用二维平铺布局,所有地块存储的数据结构应当是一个二维数组最合适,所以地块的索引我们定义为二维的下标Vector2Int

    /// <summary>
    /// 地块
    /// </summary>
    [DisallowMultipleComponent]
    public class Block : HTBehaviour
    {
        /// <summary>
        /// 目标渲染器
        /// </summary>
        [Label("目标渲染器")] public SpriteRenderer Target;
        /// <summary>
        /// 类型
        /// </summary>
        [Label("类型")] public BlockType Type;
        /// <summary>
        /// 位置
        /// </summary>
        [Label("位置")] public Vector2Int Pos;
    }

地块类的属性暂时就想到这么多,不必追求一次就考虑全面,后续根据情况补充即可,接下来我们定义关卡类。

关卡(Level)

关卡用于容纳并绘制一系列地块,以及后期容纳角色等其他的一系列东西。

一、定义关卡类

我们定义关卡类Level

    /// <summary>
    /// 关卡
    /// </summary>
    [DisallowMultipleComponent]
    public class Level : SingletonBehaviourBase<Level>
    {
    }

Level继承至单例行为基类SingletonBehaviourBase,使得它可以挂载到游戏物体上,并作为单例始终全局唯一(同一时刻,场景中的关卡肯定只能有一个)。

二、关卡基础属性

我们先为关卡设计一些基础的属性:

        /// <summary>
        /// 关卡索引
        /// </summary>
        [Label("关卡索引")] public int Index;
        /// <summary>
        /// 关卡名称
        /// </summary>
        [Label("关卡名称")] public string Name;
        /// <summary>
        /// 关卡背景音乐
        /// </summary>
        [Label("关卡背景音乐")] public AudioClip BGAudio;
        /// <summary>
        /// 地图
        /// </summary>
        [Label("地图")] public AStarGrid Map;
        /// <summary>
        /// 地图尺寸
        /// </summary>  
        [Label("地图尺寸")] public Vector2Int MapSize;
        /// <summary>
        /// 角色根节点
        /// </summary>
        [Label("角色根节点")] public Transform RolesRoot;
        /// <summary>
        /// 特效根节点
        /// </summary>
        [Label("特效根节点")] public Transform EffectsRoot;
属性名称属性详解
关卡索引关卡唯一标识符,也用作保存、加载关卡时的索引
地图所有地块对象的根节点,此处类型为AStarGrid(A*寻路网格),兼并寻路功能
地图尺寸地图的尺寸,用于描述地图的宽、高
角色根节点所有角色对象的根节点
特效根节点所有特效对象的根节点

地块、角色、特效都归于单一的根节点,即方便管理所有对象,又方便统一划层,也即是规定渲染器的遮挡层,这三者的遮挡层关系应当是:特效 > 角色 > 地块,具体如何实现遮挡,我们先不考虑这么多。

完事后我们的Level在层级面板应当是这样的(三个子对象也即是地块、角色、特效的根节点):
在这里插入图片描述

三、地块集合

定义一个二维数组存储所有地块:

        /// <summary>
        /// 所有的地块
        /// </summary>
        public Block[,] Blocks { get; private set; }

此处注意,将其定义为property的原因是,使其规避序列化功能(因为不需要序列化,关卡在初始化时搜寻所有地块即可),且提升访问安全性。

四、关卡初始化

然后,定义一个初始化方法,用于在关卡加载到场景中后,执行他自身的所有初始化操作,此处我们避开MonoBehaviour的生命周期方法AwakeStart,因为在此处他们是不受控的,这也是我上面所提到的不滥用生命周期的另一个意思。

        /// <summary>
        /// 初始化
        /// </summary>
        public virtual void Initialize()
        {
        	//Map为所有地块根节点,即可从Map搜寻所有地块
            Blocks = new Block[MapSize.x, MapSize.y];
            Block[] blocks = Map.GetComponentsInChildren<Block>(true);
            for (int i = 0; i < blocks.Length; i++)
            {
            	//地块的Pos下标,即代表了在二维数组中的索引,我们后续会开发关卡编辑器,Pos的赋值交由编辑器来完成,所以这里只管取Pos值
                Block block = blocks[i];
                Blocks[block.Pos.x, block.Pos.y] = block;
            }
        }

Initialize方法作为主动调用方法,在我们自行加载关卡完成后主动调用即可。

五、关卡销毁

同理,应当定义一个销毁方法,用于在关卡销毁时执行一些操作,虽然我们现在还没有需要做的(地块、角色、特效等都属于关卡物体子节点,会跟着一起销毁,无需额外操作),但事先将其规划好总没错。

        /// <summary>
        /// 销毁
        /// </summary>
        public virtual void Dispose()
        {
            
        }

六、回合制逻辑

为了实现回合制逻辑,我们先定义如下几个属性:

        /// <summary>
        /// 当前的回合
        /// </summary>
        [PropertyDisplay("当前的回合")]
        public int Round { get; private set; } = 1;
        /// <summary>
        /// 当前回合的行动阵营
        /// </summary>
        [PropertyDisplay("当前回合的行动阵营")]
        public RoleCamp RoundCamp { get; private set; } = RoleCamp.Player;
        /// <summary>
        /// 关卡状态
        /// </summary>
        [PropertyDisplay("关卡状态")]
        public LevelState State { get; private set; } = LevelState.InProgress;

    /// <summary>
    /// 角色阵营
    /// </summary>
    public enum RoleCamp
    {
        /// <summary>
        /// 玩家
        /// </summary>
        Player = 0,
        /// <summary>
        /// 敌人
        /// </summary>
        Enemy = 1
    }
    
    /// <summary>
    /// 关卡状态
    /// </summary>
    public enum LevelState
    {
        /// <summary>
        /// 进行中
        /// </summary>
        InProgress,
        /// <summary>
        /// 已通关
        /// </summary>
        Passed,
        /// <summary>
        /// 已失败
        /// </summary>
        Failed
    }

此处应该很好理解了,我们按字面意思来就行了,根据最初的设计,每一个回合:玩家先行动,然后是敌人行动,敌人行动完毕后此回合结束,进入下一回合(循环往复)

此时,我们发现,完整的回合制逻辑在未编写角色(Role)类前,并不太好写出来,所以我们先放下,将复杂的事情留到后面一步步拆解。

接下来我们准备引入角色(Role)类,不过,看了一眼窗外,今日天色已晚,不宜working…

那么,择日再战吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/703488.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

EDEX-UI这个终端模拟器

eDEX-UI 是一款开源、免费、跨平台的全屏终端模拟器和系统监视器&#xff0c;外观和操作界面极其科幻&#xff0c;灵感来自电影《创战纪》的会议室特效场景。作者倾注了大量心血&#xff0c;使得它不仅拥有酷炫的操作界面&#xff0c;还具备清晰爽脆的音效。 优点&#xff1a; …

使用 PNPM 从 0 搭建 monorepo,测试并发布

1 目标 通过 PNPM 创建一个 monorepo&#xff08;多个项目在一个代码仓库&#xff09;项目&#xff0c;形成一个通用的仓库模板。 这个仓库既可以用于公司存放和管理所有的项目&#xff0c;也可以用于将个人班余的所有积累整合其中。 2 环境要求 核心是 PNPM 和 Node.js&…

万字长文讲解Linux内存管理:伙伴系统

1. buddy system简介&#xff1a; 伙伴系统是内核中用来管理物理内存的一种算法&#xff0c;我们知道内存中有一些是被内核代码占用&#xff0c;还有一些是被特殊用途所保留&#xff0c;那么剩余的空闲内存都会交给内核内存管理系统来进行统一管理和分配。 内核中会把内存按照…

nodejs——原型链污染

一、引用类型皆为对象 原型和原型链都是来源于对象而服务于对象的概念&#xff0c;所以我们要先明确一点&#xff1a; JavaScript中一切引用类型都是对象&#xff0c;对象就是属性的集合。 Array类型、Function类型、Object类型、Date类型、RegExp类型等都是引用类型。 也就…

Codeforces Round 950 (Div. 3) A~F

A.Problem Generator&#xff08;遍历&#xff09; 题意&#xff1a; 弗拉德计划在下个月举行 m m m轮比赛。每轮比赛应包含一个难度为"A"、“B”、“C”、“D”、“E”、"F"和"G"的问题。 弗拉德已经有了一个 n n n个问题的问题库&#xff0…

easyrecovery专业版破解无需注册绿色版免费下载 easyrecovery16数据恢复软件永久激活码密钥百度网盘crack文件

EasyRecovery &#xff08;易恢复中国&#xff09;是由全球著名数据厂商Ontrack 出品的一款数据文件恢复软件。支持恢复不同存储介质数据&#xff1a;硬盘、光盘、U盘/移动硬盘、数码相机、Raid文件恢复等&#xff0c;能恢复包括文档、表格、图片、音视频等各种文件。 开发背景…

解决uview2中u--input输入框禁用状态下click事件不生效

需求&#xff1a;想要点击输入框&#xff0c;展示下拉内容 之前使用uview1是可以直接在input上添加click事件&#xff08;禁用和只读情况下都不影响&#xff09; 但是在uview2上直接写click不生效 解决方式&#xff1a;直接在写click.native"xxx" 代码部分&#x…

什么是有限状态机

标准答案 有限状态机表示有限个状态以及在这些状态之间的转移和动作等行为的数学模型 我自己的理解 有限状态机是状态的改变&#xff0c;比如门的开关、灯的开关等。 执行某些操作&#xff0c;比如推门或按下开关&#xff0c;状态会发生切换。 在这里&#xff0c;我编写一…

LIUNX系统编程:可重入函数volatile

目录 1.概念 2.volatile关键字 1.概念 在执行流执行到mian函数&#xff0c;insert函数中的1号位置的时候&#xff0c;突然就陷入内核&#xff0c;处理信号&#xff0c;执行信号自定义方法&#xff0c;这个方法调用的也是insert&#xff0c;执行完之后&#xff0c;导致了n2的节…

音视频文件格式转换神器(常用JPG、PNG、MP4、MP3转换等)

一、简介 1、一款完全免费、无广告且开源的格式转换工具&#xff0c;支持超过200种文件格式的转换。它能够处理视频、音频、图像、文档、电子书等多种类型的文件&#xff0c;功能非常强大。该软件由GitHub上的一位开发者发布&#xff0c;目的是为了让用户能够轻松地完成文件转换…

C++升级软件时删除老版本软件的桌面快捷方式(附源码)

删除桌面快捷方式其实是删除桌面上的快捷方式文件,那我们如何去删除桌面快捷方式文件呢?软件可能已经发布过多个版本,其中的一些版本的快捷方式文件名称可能做了多次改动,程序中不可能记录每个版本的快捷方式名称,没法直接去删除快捷方式文件。本文就给出一种有效的处理办…

25.入口点注入

钩子注入是利用SetWindowsHookEx函数这是一个被动的注入方式&#xff0c;入口点注入是一个主动注入&#xff0c;就是做这件事什么都不为就是为了注入&#xff0c;入口点注入有很多优势比如说做一个游戏的多开器&#xff0c;多开的检测事情是在游戏一启动的时候完成的&#xff0…

流程控制基本概念

流程控制基本概念 默认情况下程序运行后&#xff0c;系统会按书写顺序从上至下依次执行程序中的每一行代码。但是这并不能满足我们所有的开发需求, 为了方便我们控制程序的运行流程&#xff0c;C语言提供3种流程控制结构&#xff0c;不同的流程控制结构可以实现不同的运行流程…

Meta Llama 3 前馈层

Meta Llama 3 前馈层 flyfish 图片来自论文 http://arxiv.org/pdf/2304.13712 因为树根是Transformer&#xff0c;所以这里会将 Llama 3 与Transformer比较下 Transformer的前馈层 在Transformer模型中&#xff0c;每个编码器和解码器层中都包含一个前馈神经网络&#xff0…

MySQL-子查询(DQL 结束)

054-where后面使用子查询 什么是子查询 select语句中嵌套select语句就叫做子查询。select语句可以嵌套在哪里&#xff1f; where后面、from后面、select后面都是可以的。 select ..(select).. from ..(select).. where ..(select)..where后面使用子查询 案例&#xff1a;找…

C++中stack和queue

前言 在 C 中&#xff0c;stack&#xff08;栈&#xff09;和 queue&#xff08;队列&#xff09;是两种常用的容器适配器&#xff0c;分别用于管理数据的后进先出&#xff08;LIFO&#xff09;和先进先出&#xff08;FIFO&#xff09;访问模式。本文将详细介绍这两种数据结构的…

C#开源软件:OneNote组件oneMore轻松打造自己的公众号编辑器

OneMore是一款为Microsoft OneNote设计的插件&#xff0c;它提供了许多扩展功能来增强OneNote的使用体验。 插件功能概述&#xff1a; OneMore插件拥有多达一百多个扩展功能&#xff0c;这些功能覆盖了笔记编辑、搜索、导出等多个方面&#xff0c;为用户提供了更加便捷和高效的…

【项目实战】--云备份系统

1、云备份认识 自动将本地计算机上指定文件夹中需要备份的文件上传备份到服务器中。并且能够随时通过浏览器进行查看并且下载&#xff0c;其中下载过程支持断点续传功能&#xff0c;而服务器也会对上传为文件进行热点管理&#xff0c;将非热点文件进行压缩存储&#xff0c;节省…

openGauss 6.0.0 一主二备集群安装及使用zcbus实现Oracle到openGauss的数据同步

一、前言 openGauss 6.0.0-RC1是openGauss 2024年3月发布的创新版本&#xff0c;该版本生命周期为0.5年。根据openGauss官网介绍&#xff0c;6.0.0-RC1与之前的版本特性功能保持兼容,另外&#xff0c;在和之前版本兼容的基础上增加了很多新功能&#xff0c;比如分区表性能优化…

Java集合自测题

文章目录 一、说说 List , Set , Map 三者的区别&#xff1f;二、List , Set , Map 在 Java 中分别由哪些对应的实现类&#xff1f;底层的数据结构&#xff1f;三、有哪些集合是线程不安全的&#xff1f;怎么解决呢&#xff1f;四、HashMap 查询&#xff0c;删除的时间复杂度五…