简单自定义实体
在知道一些 mod 制作中最为基础的东西后, 我们首先把兴趣点转向自定义实体, 相信相对于修改已有的东西并伴随着一定的烧脑(我该怎么改? 这么改会不会破坏官图元素?), 你可能更喜欢自己制作一个实体, 那就对了! 在这一节, 我们会了解 loenn 的基础使用、添加一个自定义实体的通用步骤, 以及一些游戏常用的方法!
我们的自定义实体类
它能干什么
在开始制作这个自定义实体之前, 我们先明白一下这个实体做出来是什么效果的.
那么, 为了简单起见, 这里就做一个简单的功能, 它大概是:
- 它叫做
PassByRefill
- 长度和宽度是随意的
- 它看起来是个透明的红色物体
- 玩家与其碰撞时会锁定为某个冲刺数
- 这个冲刺数可以在作图软件中设置
是的就这样简单, 那么我们开始吧.
声明这个类
Warning
在前面我们钩取了 Player.Update()
并且在一直修改冲刺数为 1!
不要忘记把钩子函数以及钩取和取消钩取的那些代码删掉,
最后你的 Module 类应该只包含两个空的 Load
和 Unload
方法!
首先在代码中, 我们需要声明这个类, 也就是创建一个 .cs 文件, 然后命个名.
在这里你还需要继承一下 Entity
, 在前面我们已经了解过了它是 Monocle
的一个类, 表示场景中的实体.
那么你应该会写出如下的代码:
PassByRefill.cs | |
---|---|
1 2 3 4 5 6 |
|
ok, 接下来, 我们加一个表示冲刺数的字段, 然后搞一个构造函数接收并赋值它:
PassByRefill.cs | |
---|---|
1 2 3 4 5 6 7 |
|
这里的 position
参数会被用于设置它的 Position
字段, 这个字段从 Monocle.Entity
继承而来, 它表示这个 Entity
在场景中所在的位置(世界坐标).
Vector2
是一个 XNA 中的结构体, 包含一个 X
和一个 Y
的 float
字段, 它用于描述一个平面向量, 就像你在平面直角坐标系中所做的一样!
不过这里需要注意, 游戏中的坐标系和我们通常数学中的坐标系不同, 它的原点在左上角, X 轴水平向右但是 Y 轴竖直向下! 记住不要搞混了.
你可能注意到我们并没有理 size
这个表示大小参数, 稍等一下, 我们等会就会使用它了.
接下来, 我们需要声明一个特殊的构造函数, 以让 Everest 反射调用并使得我们的实体可以正常获取地图的数据:
PassByRefill.cs | |
---|---|
1 2 3 |
|
EntityData data
储存了作图软件保存的相关数据, 我们要提取它们很简单.比如说我们要提取一个名为
dashes
的 int
类型的数据, 我们就简单地调用它的方法 Int(string name)
,
然后是它的大小数据, 大小在这是个特殊的东西, 得需要通过 Width
和 Height
字段来提取, 然后我们扔进 Vector2
里并传递给我们上面的构造函数.
对于更多方法以及这部分数据该如何自定义我们会在本节后半部分说明.Vector2 offset
参数表示这一面的最左上角的世界坐标, EntityData
的 Position
字段表示物体相对于这一面最左上角的位置,
所以我们需要把它相加来得到世界坐标. (因为 Entity.Position
通常只允许世界坐标!)
Info
这个构造函数的签名也能为其他的样子, 但是我们最最最常用的一个版本就是上面这个
让 Everest 找到它
现在我们只定义了一个类 Everest 那边是啥也不知道的, 所以我们得告诉它, 而这个过程我们一般就叫 注册
.
要注册一个自定义实体非常简单, 我们只需要将 CustomEntity
(在命名空间 Celeste.Mod.Entities
中) 装饰到我们的类上并传入我们实体的 "名称 ID". 就像:
PassByRefill.cs | |
---|---|
1 2 3 4 |
|
MyCelesteMod
, 类名为 PassByRefill
, 实体 "名称 ID" 就是 MyCelesteMod/PassByRefill
. 这里我建议你记住它, 待会我们会在作图软件配置的时候用到.
Loenn
作图软件
蔚蓝的作图软件最流行的目前有 ahorn
与 loenn
, 不过我更推荐 loenn
, 因为 ahorn
实在是 太!!卡!!了!!
基础使用
Loenn 是一个蔚蓝的作图软件, 当然这里我们使用它不是制作出精美的地图, 而是仅仅放置一下我们的自定义实体!
_(:з」∠)_, 这里我不怎么想重复它的基础使用, 所以我推荐你去以下几个视频中了解:
版本
如果你之前已经下载了 Loenn, 我建议你确保它升级到最新版了, 因为接下来的某些东西我可能不太记得是哪个版本引入的了.
让 Loenn 找到我们的实体
相信你现在已经懂的 Loenn 基本的使用用法了, 那么现在, 我们要给 Loenn 写一些东西让它知道我们这个实体 mapper 们能用了.
首先我们在 ModFolder 下新建一个叫 Loenn
的文件夹, 再在里面新建一个叫 entities
的文件夹, 再在里面新建一个叫 PassByRefill.lua
(也就是{实体类名}.lua
) 的文本文件.
现在你的目录结构应该是这样的:
- ModFolder
- Loenn
- entities
- PassByRefill.lua
- entities
- Loenn
ok, 然后我们打开 PassByRefill.lua
, 这是一个 lua
语言的源文件, 这里我们不需要很多 lua
知识, 直接照着抄就好了:
PassByRefill.lua | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
还记得我们之前记住的实体 "名称 ID" 吗? 在这里你就需要将其赋值给 entity.name
.
- 然后是
entity
的placements
, 它表示这个实体的放置选项, 在这里我们先不管, 你就理解为你希望 Loenn 怎么把你这个实体拆分为多个放置选项, 这里我们的写法表示不希望 Loenn 拆分它, 如果这里的placements
是个数组的话那么 Loenn 就会拆分它了.- 继续到
name
属性, 它表示这个放置选项的名称, 这个名称是任意的, 它只被用来作为本地化键名(本地化就包括我们常说的"汉化"). - 然后是
data
属性, 它表示这个实体的数据.- 在这里我们声明了三个数字类型的属性:
width
,height
,dashes
, 前两个属性是特殊的, 在代码中提取只需要对EntityData
获取Width
/Height
字段即可. - 第三个
dashes
属性就是我们的自定义属性
- 在这里我们声明了三个数字类型的属性:
- 继续到
在 data
对象里的属性, 后面的等号就表示它的默认值, 比如 width
默认为 16, dashes
默认为 1.
Note
如果你不声明 width
和 height
属性并且不加贴图的话 Loenn 似乎会直接禁止你放置这个实体.
好吧看起来上面这一坨*非常难懂*, 不过没关系, 只要你会直接复制上面的东西, 改一下 entity.name
, 向 data
里加一些属性就行了. 实际上我也是这么做的.
那么, 重新编译我们的项目, 让 msbuild
把我们的东西复制过去, 这时打开我们的 Loenn (或者重启 Loenn), 搜索一下 @MyCelesteMod
.
那么列表上应该只会出现一个名字很奇怪的选项(因为我们还没配置本地化相关的东西!), 现在把它放到任何你喜欢的地方, 打开它的属性框, 我们就能看到一个大大的 dashes
属性在上面允许我们改啦. 你可能注意到它是全白的! 因为我们还没告诉 Loenn 它长什么样! 不过这个等以后我们再来搞.
回到代码
现在在游戏中进入放了我们那个实体的那张图, 然后到之前你喜欢的那个地方, 然后... 是的, 你啥也不会看见, 因为我们也没告诉游戏它长什么样! 同时你走到它附近, 也不会发生任何事, 因为我们也没告诉游戏这个实体应该干什么!
告诉游戏它该做什么
那么, 就像前面描述的一样, "玩家与其碰撞时会锁定为某个冲刺数", 那么我们需要做一些碰撞. 在 Monocle
里这项工作很简单, 我们首先在构造函数里去 new
一个长方形的碰撞箱:
PassByRefill.PassByRefill | |
---|---|
1 2 3 4 |
|
Info
我们在这里用到了之前的 size
参数
Hitbox
就是我们想要的长方形碰撞箱, 它的唯一的构造器接收四个参数, 前两个参数为它的宽高, 后两个参数为这个碰撞箱的偏移(因为碰撞箱我们要附加到 Entity
身上, 所以会有偏移这个东西),
这两个参数默认都为 0.
然后设置到 Entity
身上:
PassByRefill.PassByRefill | |
---|---|
1 |
|
Collider
是 Entity
的一个属性, 它表示这个 Entity
自身的碰撞箱.Collider
属性是 Collider
类型的, 它是一个抽象类表示一个碰撞体, 这里我们的 Hitbox
就是它的一个实现, 也就是一种长方形的实现.
碰撞箱设置完后, 我们就该在 Update()
检测碰撞了, 这很简单~
PassByRefill.Update() | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
相信你看到第五行一定会被吓一跳! 相信那一串东西对于新人来说一定很复杂, 不过没关系, 你只需要知道那一串会返回在场的那个玩家就行(不在场时为 null).
然后我们用 Entity
的 CollideCheck
检查我们有没有与玩家发生碰撞, 如果有则强制设置它的冲刺数.
由于我们每一帧都在检查, 都在设置, 所以最终的效果就是玩家一旦进入这个区域, 冲刺数被锁定为 Dashes
.
Info
实际上这里有更好的方法来单独检测与玩家的碰撞, 为了简单起见这里没有采用. 不过我们依然会在后面提到("常见 Celeste, Monocle 类"节).
第一行的 base.Update()
会遍历调用该 Entity
的所有 Component
的 Update()
, 通常我们需要在开头就调用它. 后面的 Render()
也是.
Info
在玩家死亡后的"烟花"并不是 Player
实例, 所以此时我们会得到一个 null
的结果, 如果你不想让你的游戏崩溃的话记得检查它是否为 null
.
告诉游戏它长什么样
那么, 功能做好后, 得让玩家看见, 为了简单起见这里我们不打算使用图片, 只是像前面描述的一样, 它是个"透明的红色物体",
在这里我们需要重写 Entity
的 Render()
方法来绘制一个纯色长方形,
它会在游戏中每帧都会被调用, 就像 Update()
一样,
不过切记不要在这两个方法里干不相关的事! 你应该在 Update()
里只更新你的逻辑, 而在 Render()
里只做绘制.
这里我们选择绘制一个透明的红色长方形, 我们需要借助这个函数:
Monocle.Draw.Rect | |
---|---|
1 |
|
- 第一个参数是我们希望绘制的位置(世界坐标)
- 第二个参数是我们希望绘制的宽度
- 第三个参数是我们希望绘制的高度
- 第四个参数是我们希望绘制的颜色
我们在这里会这样用它:
PassByRefill.Render() | |
---|---|
1 2 3 4 |
|
首先我们获取一个红色的 Color
, 然后设置透明度为 127
, 然后调用这个方法绘制.
你可能注意到 Width
和 Height
这两个东西, 它是 Entity
的两个属性, 默认它们都是 0, 在你加入碰撞箱的瞬间他们会变成碰撞箱的宽和高,
在这里它们的值就是我们之前设置的 Hitbox
的宽高.
准备就绪
那么, 一切就绪, 编译你的项目, 到那个地方, 在半透明红色的区域里享受锁定冲刺数的快乐吧!
如果你遇到了困难, 你可以对比一下最终的代码:
PassByRefill.cs | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 |
|
更多
在 Loenn 中禁止设置 dashes 为小数
在 Loenn 中如果你没有显式指定某个数字 data 的类型的话它默认会是浮点数, 也就是你能输入小数, 不过这不会很影响(还是有的!)代码那边, 所以我们得跟 Loenn 说一下它是个整数! 那么我们在代码的 return 之前这样设置一下:
PassByRefill.lua | |
---|---|
1 2 3 4 5 6 |
|
fieldInformation
, 然后在里面告诉 Loenn 我们的 dashes
这个属性的 fieldType
是 integer
, 也就是整数.那么现在再重新编译, 重启 Loenn, 你应该会看到 Loenn 只允许你输入整数了.
为 Loenn 侧配置本地化
现在我们在 Loenn 侧我们的实体放置选择的名称非常奇怪! 这肯定不是你想要的结果, 所以这里我们将为其配置本地化.
那么, 常规地, 我们需要在 ModFolder/Loenn
这个文件夹下再新建一个叫 lang
文件夹, 然后在里面创建一个叫 en_gb.lang
的空文件,
它是 Loenn 默认读取的本地化文件. 现在你的目录结构应该像:
- ModFolder
- Loenn
- entities
- PassByRefill.lua
- lang
- en_gb.lang
- entities
- Loenn
在 en_gb.lang
文件里, 我们写下:
1 |
|
啊, 这一串可能有亿点长, 我们慢慢来解释一下:
首先这是一个实体, 所以以 entities
开头, 然后我们需要键入实体的"名称 ID", 然后再依次键入 placements
和 name
,
这表示我们想本地化的内容是那个实体的放置选项的名称, 最后键入放置选项的名字 "normal"(它应该会在你的 lua 的第五行附近).
ok 现在我们已经指定了我们想本地化什么东西了, 那么接下来简单的 ={内容}
, 在这里是 PassByRefill
. 最后, 编译复制, 重启 Loenn,
你应该就会看到我们的实体终于有个正常名字了:
Info
对于它的贴图我们之后再说, 我知道你很急但是你先别急.