翻译: 星球生成 II

翻译: 星球生成 II

本文翻译自Planet Generation - Part II

  • 译者: FreeBlues

以下为译文:

概述

前一章 我解释了如何为星球创建一个几何球体.

在本文中, 我将会解释如何为即将用来创建陆地,岛屿和海洋的顶点加上高度数据.

第一幅星球截图:

计划

本文的计划是为球体的每个面创建高度数据, 置换位置来创建陆地和海洋.

一个重要性质是生成应该是伪随机的(pseudo random), 它允许创建很多外观不同的星球, 并且也能通过提供一个唯一的 种子数(seed)来完全复现相同的生成.

已有的几何体

当前的几何体由可以折叠成一个球体的 6 个网格组成. 这个事实为我们简化了大量不得不做的工作. 如果有人想要创建一个纹理贴图, 他们可以很容易地保持住一个网格的 UV 映射并且为每个面附着一个纹理贴图或者只用立方体纹理映射.

立方体纹理贴图截图:

出于为了得到更高级别细节的原因, 我决定把我的几何体分到 6 个不同的顶点缓冲区中, 它允许我把表面上看不出的独立的位面完全捡出, 并且对每一个位面实现 LOD 算法, 就好像它是一个孤立的地形块.

for ( int j = 0; j < numRows; ++j )
{
    for ( int i = 0; i < numCols; ++i )
    {
        vertices[  j * width + i ].UV.x = (float)i / (float)(numCols - 1);
        vertices[  j * width + i ].UV.y = (float)j / (float)(numRows - 1);
    }
}

现在我们有了 UV 映射, 颜色和高度纹理能被应用了.

生成高度图

正如我上一个帖子中提到过的, 我选择过程内容生成的部分原因是用它补偿我艺术技能方面的欠缺. 换句话说, 对我来说创建一个用于星球的纹理贴图的最好的方法是创建一个将会为我创建纹理贴图的程序. 他们还说, 程序员是安全的, 而人工智能会抢走我们的工作...

我们有 6 个网格, 每个都需要高度数据, 一个特殊需求是数据要很好地沿着网格每一面的边缘包回来. 内容生成的 过往经验 让我以 perlin noise terrain generation 为关键字进行搜索, 结果找到这个教程Tutorial 8: Creating spherical planetary terrain

Libnoise 是一个可移植的, 开源的, 连贯的噪声生成 C++ 库. 对我的需求来说, 这一点听起来很不错, 当我在 Nexus 5 Android 手机上运行这些代码时, 并且这个库还支持一个 SetSeed 函数, 因此它满足了我给定一个种子能够复现相同结果的需求

生成的图可以很容易地切成立方体纹理或者对球体额每一面都独立的纹理贴图(译者注:那就是6个小纹理贴图). 这就是根据上面教程生成的纹理贴图的一个示例:

地球高度截图:

它在地形生成的教程中使用了 Perlin 噪声. Perlin 噪声作为一种很好的用于生成分形并被用于模拟云,烟雾和地形之类的自然景物的梯度噪声生成器而广为人知.

柏林噪声截图:

不幸的是, 用它经过一段时间的试验后, 我发现它生成数据太慢了, 接着我寻找了替代品. 这把我引向简单噪声(Simplex Noise), 跟 Perlin 噪声相比, 它更快并且提供了和 Perlin 噪声相似的结果. 它也可以 GPU 上运行.

使用简单噪声

现在我决定选择的噪声算法需要一个将会满足我们限制的实现. 我发现了作为一个轻量级的 C++ 解决方案的实现 Simplex Noise for C++ and Python 并且开始试验结果.

译者注: 这里是简单噪声和 Perlin 噪声的效果对比图:

效果对比图:

simplexnoise.h 文件中对我们目前情况有用的函数是 scaled_octave_noise_3d, 它允许为三维坐标生成噪声, 充分满足我们沿着星球交点包裹纹理贴图的需求.

float scaled_octave_noise_3d( const float octaves,
                            const float persistence,
                            const float scale,
                            const float loBound,
                            const float hiBound,
                            const float x,
                            const float y,
                            const float z );

对每一个把位置映射到球体的顶点, 我用下面的函数为它应用了偏移(displacement):

void CalculateHeight( Vector3& vPosition )
{
    // Get the direction vector from the center of the sphere
    Vector3 vNormalFromCenter = vPosition;
    vNormalFromCenter.Normalize();

    // Variables for the noise
    static const float HEIGHT_MAX = 24.5f; // Planet Radius is 1,000.
    static const float HEIGHT_MIN = -31.0f;
    static const float NOISE_PERSISTENCE = 0.6f;
    static const float NOISE_OCTAVES = 8.0f;
    static const float NOISE_SCALE = 1.0f;

    // Generate the noise for the position
    float fNoise = scaled_octave_noise_3d( HEIGHT_OCTAVES, HEIGHT_PERSISTENCE, NOISE_SCALE, HEIGHT_MIN, HEIGHT_MAX, vPosition.x, vPosition.y, vPosition.z );

    // Keep ocean level as base level
    if ( fNoise <= -0.0f )
        fNoise = 0.0f;

    // Displace the position
    vPosition += vNormalFromCenter * fNoise;
}

这导致一个如本页第一幅图片所示的结果. 我在 0.0f 或比它小的位置对所有高度应用了 RGB(28, 107, 160), 对所有陆地层应用了 RGB(61, 182, 29).

注意高度的最小值和最大值. 因为地球大概 1/3 是陆地, 2/3 是水, 我决定用将会产生 33%-40% 陆地的数值来试验. 在试验过数值之后, 我发现 -31.024.5 产生很好的结果. feel free to, 另外, 我的星球半径是 1000, 因此这些值存在某种相关比例.

用这样一个快速解决方案来开始是相当令人满意的. 将来我希望能增加细节通过应用第二个噪声图, 更少的存留状态, 和/或 增加海平面以上的高度比例. 这将会生成额外的山脉和更少连贯的陆地板块(land masses).

下面是一个星球的示例, NOISE_SCALE = 2.5f:

截图:

应用一个种子数

正如我在本文开头所说, 生成器的一个需求是给定一个种子能够复现相同的结果. 当 libnoise 提供了一个 SetSeed 函数, 简易噪声(SimplexNoise)没有已实现的方法可用.

幸运的是实现一个方法非常容易. 在 simplexnoise.h 头文件中, 有一个数字数组用作一个排列表.

static const int perm[512] = { ... }

这个数组基本上包含了从 0255 随机生成的所有数字. 这个序列接着被复制一次来填满整个数组

通过从一个常量改变数组, 并且应用了下面这个函数, 你能轻易地为一个星球设置种子并且复制相同的结果:

// Variables
// perm - SimplexNoise's Permutation table
void set_seed( unsigned int seed )
{
    // Set the seed for random generator
    srand( seed );

    // Set up the random numbers table
    for ( int i = 0; i < 256; ++i )
    {
        // Put each number in once
        perm[ i ] = i;
    }

    // Randomize the random numbers table
    for ( int i = 0; i < 256; ++i )
    {
        // Replace current value with random value from the table
        int k = perm[ i ]; // current value

        // Random table cell
        int j = (int)random( 256 );

        // Replace two values
        perm[ i ] = perm[ j ];
        perm[ j ] = k;

        // The table repeats itself so just assign repeating values the same way
        perm[ 256 + i ] = perm[ j ];
        perm[ 256 + j ] = k;
    }
}

后续

这就是这篇文章的全部了, 在接下来的文章中, 我会关注光照, 大气散射, 生物群落生成, 星球颜色以及细节级别(Level of Details).

参考

原文: Planet Generation - Part II
Libnoise: Tutorial 8: Creating spherical planetary terrain
Perlin noise
Simplex Noise for C++ and Python
Simplex Noise Demystified pdf
GPU Gems 2: Chapter 26. Implementing Improved Perlin Noise

posted on 2016-07-21 01:39  自由布鲁斯  阅读(1192)  评论(0编辑  收藏  举报