本文经@a327ex授权翻译与转载, 原文链接1: GitHub,原文链接2: gamedeveloper

这篇文章阐释了一种生成随机地牢的技术,该技术首先由TinyKeepDev在此描述.我将比原始帖子中的步骤更详细地介绍它.此算法的实现流程如下:

生成房间

首先, 你需要在一个圆内生成随机的生成一些具有一定宽度和高度的房间.TKdev的算法通过正太分布来控制房间的大小, 我认为这是一个好主意, 因为它可以通过更多参数来进行控制.通过控制宽高比和标准方差可以生成外观不同的地牢.

一个你可能需要的方法getRandomPointInCircle

1
2
3
4
5
6
7
function getRandomPointInCircle(radius)
local t = 2*math.pi*math.random()
local u = math.random()+math.random()
local r = nil
if u > 1 then r = 2-u else r = u end
return radius*r*math.cos(t), radius*r*math.sin(t)
end

你可以通过这里获取有关此代码原理的相关信息. 这样你就能完成以下的内容:

有一件事是你需要考虑的, 由于你(至少想法中)处理的是网格, 因此你需要将所有的内容都对齐到相应的网格上.在上面的gif中, 图块的大小是4像素, 这意味着所有的房间的位置和大小都是4的倍数. 为此, 我们将宽度/高度的分配通过一个通过的函数进行处理, 这个函数将处理图块的大小已达到整体的统一:

1
2
3
4
5
6
7
8
9
function roundm(n, m)
return math.floor(((n + m - 1)/m))*m
end

-- Now we can change the returned value from getRandomPointInCircle to:
function getRandomPointInCircle(radius)
...
return roundm(radius*r*math.cos(t), tile_size), roundm(radius*r*math.sin(t), tile_size)
end

分离房间

现在我们将进入分离房间的部分, 许多的房间都不应该像现在这样重叠并混杂在一个地方. TKdev使用separation steering behavior(直译为分离转向行为)来做到这一点, 但我发现只使用物理引擎会容易的多. 添加完所有的房间后, 将物理引擎添加到每个房间上, 然后模拟运行纸质所有的房间都重新进入休眠状态.在gif中, 我们正常的运行并模拟, 但是当你在你的关卡之间进行此操作的时候, 你可以使用更快和简单的方法.

我们并没有将物理效果绑定到网格上, 但是在设置房间位置的时候, 你可以使用roundm将其包裹起来, 这样你就可以得到彼此不重叠并且也视频网格的房间了.下面的gif显示了这一点, 蓝色的轮廓是物理引擎得到的位置, 因为他们对位置进行了取整, 使得了他们和房屋之间的位置总有不匹配的情况.

可能会出现的一个问题, 如果你的房间是水平或垂直居多的时候, 比如我正在开发的游戏:

由于战斗是横向的, 所以我希望大多数的房间的宽度是大于高度的. 那么问题就变成了物理引擎如何解决长房间在互相重叠的时候解决碰撞:

如你所见, 地牢变得很高, 这样并不理想. 为了解决这个问题, 我们最初的时候在一个椭圆内而不是圆形上生成房间. 这将保证地牢本地有合适的宽高比:

要在此区域内随机生成, 我们可以更改getRandomPointInCircle函数以便可以在椭圆内生成(在上面的gif中, 我们使用了ellipse_width = 400ellipse_height = 20)

1
2
3
4
5
6
7
8
function getRandomPointInEllipse(ellipse_width, ellipse_height)
local t = 2*math.pi*math.random()
local u = math.random()+math.random()
local r = nil
if u > 1 then r = 2-u else r = u end
return roundm(ellipse_width*r*math.cos(t)/2, tile_size),
roundm(ellipse_height*r*math.sin(t)/2, tile_size)
end

主要房间

下一步是要确定哪些房间是主要/中心房间, 哪些不是. TKdev使用的方法十分可靠: 只需要宽度和高度高于某个阈值的房间.在下面的gif中, 我使用的是1.25*mean, 这意味着如果width_meanheight_mean为24, 那么将选择宽度和高度大于30的房间.

Delaunay三角分割+图

现在我们将选择所有房间的中电并将其输如到Delaunay过程中. 你也可以自己实现此过程, 或者去寻找他人完成并共享的代码.我很幸运Yonaba完成了它. 正如你从此页面中看到的, 它接受点并输出三角形:

有了三角形后, 你就可以生成图形了. 如果你的手头有图形化数据的结构/库, 这个过程应该很简单.如果你还没有这么做, 那么你可以使你的Room对象/结构具有唯一的id吗这样你就可以将这些id添加到图形中, 而不是四处复制它们.

最小生成树

在这之后, 我们从图中生成一个最小生成树. 同样, 要么自己实现, 要么找一个其他人完成的代码.

最小生成树将保证地牢中所有的主要房间都可以到达, 但也使得它们不像以前那样全部相连. 这是很有用的, 因为在默认的情况下我们不希望所有的地牢都互相连接, 同样我们也不想要存在无法到达的岛屿. 然而, 只有一条路线的地牢也不是我们想要的, 所以我们要做的是从Delaunay中添加几条边.

在这添加更多的路径和循环, 这会使得地牢更有趣.TKdev会将15%的路径加回, 我发现大于8-10%是一个更好的值. 当然,取决于你的地牢的连接程度, 这个值会有所不同.

走廊

对于最后的一部分, 我们要为地牢添加走廊. 为此我们需要遍历图中的每个节点, 然后连接在它到其它节点直接的连线. 如果节点在水平方向上足够近(他们的y位置相似), 那么我们创建一条水平线. 如果节点在垂直方向上足够接近, 那么我们创建一条垂直线. 如果节点在水平或垂直方向上都没有靠近, 那么我们创建2条线形成L形.

我使用的测试足够的接近, 这意味着计算两个节点的位置就应该检查中点的x或y属性是否在节点的边界内. 如果是的话, 那么我们从该中点的位置创建线. 如果不是, 那么创建两条线, 从源的中点到目标的中点, 但仅仅在一个轴上.

在上图中, 你可以看到所有案例的示例. 节点62和47之间有一条水平线, 节点60和125间有一个垂直线. 节点118和119具有L形. 同样重要的是这些并不是我创建的唯一线条, 他是是我绘制的唯一一条线, 但我还在每条线的侧面创建了2条额外的先, 已tile_size为间距, 因为我希望我的走廊在宽度或高度上至少具有3个单位的宽度.

无论如何, 在这之后我们要检查哪些不是主要/中心的房间并与线条直接发生冲突. 然后将碰撞的房间加入到您现在所容纳这些信息的结构中, 他们将作为走廊的骨架:

根据你最初设置的房间的均匀性和大小, 你会在这看到不同外观的地牢. 如果你想让你的走廊更统一, 看起来不那么奇怪, 那么你应该以降低偏差为目标, 你要做一些检查, 防止房间以某种方式太瘦.

对于最后一步, 我们需要添加一个标准大小的网格来弥补缺失的部分. 请注意, 你实际上不需要网格的数据结构或任何太花哨的东西, 你只需要根据图块的大小遍历每一行并将网格的圆角位置(将对应1个大小的单元格)添加到某个列表. 这个在3(或更多)行而不是只有一行的地方.

现在我们就完成了!🌀

最后

我从整个流程中返回的数据结构是: 房间列表(每个房间具有唯一的id, x/y位置和高度/宽度的结果); 图, 其中每个节点指向一个房间id, 边缘具有最小单位的房间的距离; 然后是一个实际的2D网格, 其实每个单元格都可以是空的(意味着它是空的), 可以指向主/中心房间, 可以指向走廊房间或走廊单元. 通过这三种结果, 我认为可以从数据中活动你想要的任何类型的数据, 然后你可以确认放置门, 敌人, 物品的位置, 哪些房间有Boss等等.