翻译: 星球生成 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

翻译: 星球生成 II的更多相关文章

  1. 翻译: 星球生成 I

    翻译: 星球生成 I 本文翻译自Planet Generation - Part I 译者: FreeBlues 以下为译文: 概述 我一直是一个过程内容生成的爱好者, 它允许你创建一个甚至不断改变的 ...

  2. 【翻译】生成 Timestamps / Watermarks

    本文翻译自flink官网:https://ci.apache.org/projects/flink/flink-docs-release-1.7/dev/event_timestamps_waterm ...

  3. 机器指令翻译成 JavaScript —— 终极目标

    上一篇,我们顺利将 6502 指令翻译成 C 代码,并演示了一个案例. 现在,我们来完成最后的目标 -- 转换成 JavaScript. 中间码输出 我们之所以选择 C,就是为了使用 LLVM.现在来 ...

  4. TensorFlow练习24: GANs-生成对抗网络 (生成明星脸)

    http://blog.topspeedsnail.com/archives/10977 从2D图片生成3D模型(3D-GAN) https://blog.csdn.net/u014365862/ar ...

  5. qt中qlineedit和qtextedit右键菜单翻译成中文

    没有linguist和lupdate等命令需要安装Linguist: 在Terminal中输入: sudo apt-get install qt4-dev-tools qt4-doc qt4-qtco ...

  6. Qt 本地化(翻译)

    Qt 本地化(翻译) 翻译流程大致是这样的:首先源代码产生 ts 文件,然后送给 Qt Linguist(Qt 语言家)这个 Qt 自带的小工具进行处理产生 qm 翻译文件,最后源代码里加载这个 qm ...

  7. Linux内核调试方法总结

    Linux内核调试方法总结 一  调试前的准备 二  内核中的bug 三  内核调试配置选项 1  内核配置 2  调试原子操作 四  引发bug并打印信息 1  BUG()和BUG_ON() 2   ...

  8. EF里Guid类型数据的自增长、时间戳和复杂类型的用法

    通过前两章Lodging和Destination类的演示,大家肯定基本了解Code First是怎么玩的了,本章继续演示一些很实用的东西.文章的开头提示下:提供的demo为了后面演示效果,前面代码有些 ...

  9. 解析大型.NET ERP系统 多国语言实现

    实现多国语言有许多种实现方案,无外乎是一种字符串替换技术,将界面控件的文本标签替换成相应语言的文字..NET Windows Forms实现多国语言的方法有以下几种: 1 .NET的方案,使用资源文件 ...

随机推荐

  1. POJ 3744 Scout YYF I 概率dp+矩阵快速幂

    题目链接: http://poj.org/problem?id=3744 Scout YYF I Time Limit: 1000MSMemory Limit: 65536K 问题描述 YYF is ...

  2. week4d:个人博客作业

    7,程序结果的显示 1,界面 2,选第一选项. 3,输入3个数后. 4,选择第一个. 5,输入第4个数字. 6,再次进行一轮游戏. 7,选择是否要看历史记录. 8,进入下一轮游戏. 9,开始第二轮数字 ...

  3. PHP面试题一

    http://www.viphper.com/?p=28 1.用PHP打印出前一天的时间格式是2006-5-10 22:21:21(2分) $a = date("Y-m-d H:i:s&qu ...

  4. 3、第一个Python程序

    现在,了解了如何启动和退出Python的交互式环境,我们就可以正式开始编写Python代码了. 在写代码之前,请千万不要用“复制”-“粘贴”把代码从页面粘贴到你自己的电脑上.写程序也讲究一个感觉,你需 ...

  5. C++编译与链接(2)-浅谈内部链接与外部链接

    发现每次写技术博客时,都会在文章开头处花费一番功夫 ...从前,有一个程序员....他的名字叫magicsoar 为什么有时会出现aaa已在bbb中重定义的错误? 为什么有时会出现无法解析的外部符号? ...

  6. 读书笔记-《Java核心技术卷I-基础知识》

    1.定时器Timer类 构造定时器时,需要设置一个时间间隔,并告知定时器,当到达时间间隔时需要做什么操作.定时器需要知道调用哪一个方法,并要求传递的对象所属的类实现了java.awt.event包的A ...

  7. 在linux中安装jdk以及tomcat并shell脚本关闭启动的进程

    在命令行模式中输入uname -a ,如下图,当界面展示i386就说明本linux系统为32版本,就在官网下载对应jdk版本,或者直接到我的网盘上下载http://pan.baidu.com/s/1c ...

  8. java poi给sheet表格中的某个单元格添加批注

    Label l = , , "A cell with a comment"); WritableCellFeatures cellFeatures = new WritableCe ...

  9. EasyUseCase 一款脑图转化 Excel 测试用例工具 (1.2 版本升级)

    EasyUseCase 本工具由本人自主开发.经过内部实践有效提升测试用例编写效率200% 覆盖率可度量.利用读取xmind软件图表转换符合国人基本需求的测试用例,让手动写Excel用例的日子过去,发 ...

  10. Log4j读取配置文件并使用

    /** 设置配置路径从环境变量读取     * PropertyConfigurator类加载.properties文件的配置    * DOMConfigurator加载.xml文件的配置     ...