本系列文章是对 metalkit.org 上面MetalKit内容的全面翻译和学习.html
让咱们从第12部分 Part 12继续.使用上次咱们工做的同一个playground,咱们今天将学习光照和3D物体.记得前几周咱们作出的日食吗?它又回来了! 好吧,此次咱们将移除太阳,只关注行星.git
首先,让咱们清理内核,使其只包含下面的代码:github
int width = output.get_width();
int height = output.get_height();
float2 uv = float2(gid) / float2(width, height);
uv = uv * 2.0 - 1.0;
float radius = 0.5;
float distance = length(uv) - radius;
output.write(distance < 0 ? float4(1) : float4(0), gid);
复制代码
你必定认出几周前的这些代码了.咱们唯一改变的是将圆外面的颜色替换为黑色
,并将圆内部改成白色
.输出的黑乎乎应该是这个样子: 函数
目前还不错.行星看起来至关扁平,光照分布的太均匀看起来不真实.让我接下来修复它.几何学告诉咱们,为了找到球面上的点,咱们须要球体公式: post
在咱们的例子中,x0, y0和z0都是0由于咱们的球体在屏幕中间.算出z值就能够获得planet行星
颜色的值,因此让咱们用下面这几行来替换内核中的最后一行:学习
float planet = float(sqrt(radius * radius - uv.x * uv.x - uv.y * uv.y));
planet /= radius;
output.write(distance < 0 ? float4(planet) : float4(0), gid);
复制代码
输出图像应该像起来像这样:ui
正如你期待的那样,颜色从中间的纯白色变为圆外面的纯黑色.为此,咱们必须用颜色除以radius半径
,来使z
值规范化到 [0,1] 区间内,它能给咱们全范围的光照效果.咱们实际伪造了一个灯光源放在 (0,0,1).让引出了新话题:lighting灯光
.spa
lighting灯光让咱们的颜色真正活起来.为了在咱们的场景中有一个灯光,咱们须要计算每一个坐标的normal法线
.法向量是垂直于表面,告诉咱们表面"指向"哪一个坐标.用下面几行替换最后两千:翻译
float3 normal = normalize(float3(uv.x, uv.y, planet));
output.write(distance < 0 ? float4(float3(normal), 1) : float4(0), gid);
复制代码
注意,咱们在planet行星
变量中已经有z
值了.输出的图片看起来应该这样:
这可能不是咱们想要看到的,但至少咱们知道在每一个规格化坐标处计算颜色时法线看起来是什么样子了.下一步,让咱们建立一个光源放置在咱们左侧(负x
),后面(正z
)一点.用下面几行替换最后一行:
float3 source = normalize(float3(-1, 0, 1));
float light = dot(normal, source);
output.write(distance < 0 ? float4(float3(light), 1) : float4(0), gid);
复制代码
咱们采用了一种基本的光照模型叫作朗伯特 Lambertian(漫反射)光照,其中咱们将法线乘以规格化光源.咱们将在之后的文章中学习更多光照知识,可是若是你对学习光照模型颇有兴趣,能够参考这里 here的众多资源.输出的图片看起来应该这样:
还记得上一次内核给咱们了一个计时器uniform吗?让咱们用起来玩玩!用这行替换source
行:
float3 source = normalize(float3(cos(timer), sin(timer), 1));
复制代码
经过使用cos
和sin
函数,咱们给光源一个圆周运动.x
和y
都是按圆的参数方程从 -1到1.输出的图片看起来应该这样:
咱们来仔细看看,场景中的物体被照亮(天空中的行星),然而,物体仍然呈现出单一的表面.咱们有两种方法可让它看起来更真实:使用纹理,或者给plante
颜色加上一些噪点. 源代码source code 已发布在Github上.
下次见!