说明
本系列文章是对<3D Apple Games by Tutorials>一书的学习记录和体会此书对应的代码地址
最终的实时渲染效果图
data:image/s3,"s3://crabby-images/81a2d/81a2de4b8d266ecc8ee2b91a5844cffa916f2e91" alt=""
Lighting models(照明模型)
data:image/s3,"s3://crabby-images/99329/99329061622513477f0954cfca31cfb37d1f5512" alt=""
- Constant:使用一个扁平的照明模型,在计算渲染像素的颜色时,只包含ambient信息.
- Lambert:在计算渲染像素的颜色时,只包含ambient信息和diffuse信息.
- Blinn:在计算渲染像素的颜色时,只包含ambient信息,diffuse信息,specular信息,而且specular高光区是用Blinn-Phong法则计算的.
- Phong:在计算渲染像素的颜色时,只包含ambient信息,diffuse信息,specular信息,而且specular高光区是用Phong法则计算的.
- PBR:即Physically Based Rendering基于物理的渲染,包含了真实情况下,物理光源和材质对光的漫反射和吸收.
Materials(材质)
一般也就是textures(纹理).
纹理实际是2D图片按照几何体内部的纹理坐标系展开,包裹在3D几何体表面.所有Scene Kit内部自带的初始形状已经包含了这个坐标信息.
-
Diffuse map(颜色贴图,漫反射贴图):给几何体一个基本的颜色纹理,不考虑灯光和特效:
QQ20170409-111646@2x.png
-
Normal map(法线贴图):在上篇文章的灯光里讲过,灯光是使用形状表面的法向量来决定照亮哪个面的.系统自带形状是使用单一的整个面的向量,而法线贴图则以RGB值定义了精确到每个像素的法向量,这样每个像素对灯光的反应都不同,形成表面崎岖不平的灯光效果.
QQ20170409-111705@2x.png
QQ20170409-111814@2x.png
-
Reflective map(反射贴图):以黑白图片精确定义了材质每个像素的反光程度.就是周围环境的光线在物体表面映射出的图像(实际就是天空盒子图像在物体表面的反光).
QQ20170409-111848@2x.png
QQ20170409-111903@2x.png
需要注意的是
Xcode8之前,反射贴图和前面两个贴法不同,不是直接拉伸包裹在几何体上面的,而是用cube mapping(立方体贴图)来完成的.想像一个立方体,六个面的纹理并排水平放置,就形成了立方体贴图:
data:image/s3,"s3://crabby-images/7ea98/7ea9883750cf519716921b8e60aac191be4ffd6c" alt=""
data:image/s3,"s3://crabby-images/890ae/890ae4208479b57092fbca0f9617a95ce5851178" alt=""
Xcode8之后别的形式立方贴图也可以,:
data:image/s3,"s3://crabby-images/cc1d5/cc1d5e716791cf83947db7dc9565072895210157" alt=""
Xcode8还支持了1:2的球面贴图:
data:image/s3,"s3://crabby-images/01a77/01a770e3497c5d7602a3c417966968424984e056" alt=""
- Occlusion map(闭塞贴图):也就是ambient occlusion map(AO贴图,环境光闭塞贴图),只有当场景中有ambient light环境光时才有作用,精确定义了每个像素在环境光作用下的被照亮程度.也就是让几何体的黑色部分不被环境光照亮而变浅.
data:image/s3,"s3://crabby-images/d91e0/d91e08c426ce3705a281acda35320ce3ad7db56a" alt=""
data:image/s3,"s3://crabby-images/0d732/0d73236d8f438943942c4d3cbff708236d80b153" alt=""
-
Specular map(镜面贴图,高光贴图):镜面贴图决定了几何体的镜面程度,黑色部分就是不光滑,白色就是光滑反光.会影响Normal map(法线贴图)外部光线照射反光和Reflective map(反射贴图)外部天空盒子图像反光的清晰程度
QQ20170409-111949@2x.png
QQ20170409-112002@2x.png
-
Emission map(发光贴图):在没有光线时,如果物体表面有荧光涂料,就会发光.发光贴图可以用来模拟这种物体.彩色贴图中,黑色不发光,亮色发光强,暗色发光弱.
QQ20170409-112012@2x.png
QQ20170409-112024@2x.png
需要注意的是
在Scene Kit中Emission map(发光贴图)并不真正发光,只是模拟发光效果而已.就是说不能照亮其他物体,不能产生阴影.这点与其他3D创作工具不同. -
Multiply map(乘法贴图,正片叠底贴图):会影响其他所有效果.一般用来给最后的效果调整色彩或者亮度.
QQ20170409-120113@2x.png
data:image/s3,"s3://crabby-images/3f4be/3f4be7858ca6d28d9e5ec11cedceaa3950009529" alt=""
-
Transparency map(透明贴图):黑色部分不透明,白色透明.
注意:球体内部需要开启double-sided mode才能看到
QQ20170409-120138@2x.png
data:image/s3,"s3://crabby-images/bbd17/bbd175b80cbcb62f7730d59854f17efb0b5452ae" alt=""
- Metalness and Roughness maps(光泽度和粗糙度贴图):Xcode8引入的新特性,Physically Based Rendering (PBR)灯光模型可以使用Metalness和Roughness贴图.
WX20171203-142500@2x.png
data:image/s3,"s3://crabby-images/96cab/96cabf7b1b1a74a0b4da04fb0ba31de49461dfb8" alt=""
skybox天空盒子
设置方法如图,也可以用代码设置
data:image/s3,"s3://crabby-images/cbf1b/cbf1b40d601d8155af1ecce82e6608052e884207" alt=""
let scene = SCNScene()
scnView.scene = scene
scene.background.contents = "skybox01_cube.png"
当使用Cube map立方体贴图时,Scene Kit支持以下几种图像模式
1. A horizontal strip image where `6 * image.height == image.width`水平排列宽高6:1
2. A vertical strip image where ` image.height == 6 * image.width`竖直排列宽高1:6
3. A horizontal cross image where `4 * image.height == 3 * image.width`水平十字交叉宽高3:4
4. A vertical cross image where `3 * image.height == 4 * image.width`竖直十字交叉宽高3:4
5. A lat/long image where ` image.height == 2 * image.width`经纬度图片宽高1:2
6. A NSArray of 6 images. This array must contain images of the exact same dimensions, in the following order, in a left-handed coordinate system: +X, -X, +Y, -Y, +Z, -Z (or Right, Left, Top, Bottom, Front, Back).图片数组6张
data:image/s3,"s3://crabby-images/25b61/25b611ccf3b3cc5a57b9fa1571823c358105969d" alt=""
data:image/s3,"s3://crabby-images/fe688/fe688cb20e3f77ea69770c46a200e1c942581464" alt=""
网友评论
let gemSous = SCNGeometrySource.init(vertices: poinS)
let gemElem = SCNGeometryElement.init(indices: indexs, primitiveType: .triangleStrip)
let gem = SCNGeometry.init(sources: [gemSous,gemSous1], elements: [gemElem])
你研究下这几个类 应该就可以了