SceneKit(1): 解析 WWDC2017 ARKit Demo

在WWDC2017上,苹果推出了ARKit,在介绍完基本概念和用法后,演示了一个很小的 demo,虽然比较简单,但是很有趣,只需要几分钟就可以写出来。但对 SceneKit 不熟悉的同学来说,可能还有几个迷惑的地方,我们正好可以通过这个小 demo,管中窥豹学习一些基本知识。

这个Demo在AR场景中显示了一个飞机模型,手指点击屏幕,会将当前看到的画面截屏,并固定在场景中,就像将拍摄的照片固定在真实世界中一样。

ARKit Demo

先来看一下核心代码:

guard let currentFrame = sceneView.session.currentFrame else {
    return
}

// Create an image plane using a snapshot of the view
let imagePlane = SCNPlane(width: sceneView.bounds.width / 6000, 
                          height: sceneView.bounds.height / 6000)
imagePlane.firstMaterial?.diffuse.contents = sceneView.snapshot()
imagePlane.firstMaterial?.lightingModel = .constant

// Create a plane node and add it to the scene
let planeNode = SCNNode(geometry: imagePlane)
sceneView.scene.rootNode.addChildNode(planeNode)

// Set transform of node to be 10cm in front of camera
var translation = matrix_identity_float4x4
translation.columns.3.z = -0.1
planeNode.simdTransform = matrix_multiply(currentFrame.camera.transform, translation)

我们来一句一句剖析这段代码:

let imagePlane = SCNPlane(width: sceneView.bounds.width / 6000, height: sceneView.bounds.height / 6000)
  • 为什么平面的宽和高要用sceneView的尺寸计算出来?
    注意到 snapshot() 这个方法是 SCNView 的一个方法,返回的 UIImage 是按照实际 SCNView 的大小输出的。虽然这个例子乃至于实际大多数的AR应用都是全屏的,但你还是能将 SCNView 设置成想要的任何尺寸。因此这里不应假设就是屏幕大小,使用 UIScreen.main.bounds 之类的方法来定义截图,而是要使用调用了 snapshot() 方法的那个 SCNView 对象的尺寸。

  • 为什么宽和高要保持等比例,都除以一个数(6000)?
    因为图片纹理的显示规则与 UIImageView 设置图片的机制是不一样的。在 UIImageView 中,可以通过设置 contentMode 属性在 UIViewContentMode 枚举中选择缩放和比例模式。但在纹理的设置中 ,并没有这样自动的方法来设置纹理,如果都使用默认属性,SCNPlane 的纹理设置为图片后,该图片会拉伸铺满整个平面。所以,要保持 SCNPlane 的长宽比与截图的长宽比相同,从而保证图片显示在平面上不会被拉伸。

  • 为什么这个比例数是6000?
    以 iphone7plus 为例,屏幕点数为 414x736,由于 sceneView 与屏幕大小相同,因此 sceneView.bounds 返回 CGRect(0, 0, 414, 736),于是宽和高经过计算分别为 414/6000=0.069, 736/6000=0.123。其实宽和高的具体的数字不重要,重要的是数量级,可以看到都是小于1的非常小的数。这是因为 ARKit 要保证虚拟物体与现实世界融合的时候,虚拟物体的尺寸与现实相符,因此 ARSCNView (ARKit 中的 SCNView 的子类,sceneView 的类型)规定所用的单位为米,也就是真实物理世界中的米单位。因此这两个数字可以理解为真实物理世界中的69毫米和123毫米,大概是手机屏幕那么大。想象一下如果除以600,最后的结果尺寸就会大10倍,放置到真实世界中就会是69厘米和123厘米,目测那张桌子也放不下。而这个数字6000也应该为了这个demo选择的一个数字,正好使得放置平面后差不多正好覆盖整个屏幕,达到视觉上分离出一张图片的效果。

imagePlane.firstMaterial?.diffuse.contents = sceneView.snapshot()
imagePlane.firstMaterial?.lightingModel = .constant
  • sceneView.snapshot() 方法返回了什么?
    通过查看文档得知返回的是 UIImage,那么这个 UIImage 的尺寸是多少呢,文档并没有说明尺寸,那就只有通过调试来得知了。以 iPhone7+ 为例,返回的是1080x1920大小的 UIImage,再修改为放大模式后,仍然是1080x1920,这说明返回的是物理像素大小的图片,与 point 无关。
  • 如何设置纹理?
    • 首先,纹理是设置在 SCNGeometry 上的,与 SCNNode 无关(SCNPlaneSCNGeometry 的子类)
    • 内置的所有 SCNGeometry 的子类都定义了纹理和几何形状的对应关系,可以通过查看文档得知。因此它们至少都有一个纹理可以设置,imagePlane.firstMaterial 总是有值的。因此最简单的纹理使用方法就是设置firstMaterial 属性返回的 SCNMaterial 对象。
    • diffuse 属性定义纹理本身的颜色,相当于简单地将图片贴到几何形状表面,diffuse.contents 则定义使用什么内容来指定纹理,这个例子用的是 UIImage,更简单的还可以使用 UIColor,有很多种类型的数据都可以设置给 contents,甚至可以设置为 AVPlayer 或者 SKScene,可谓是很强大了。
let planeNode = SCNNode(geometry: imagePlane)
sceneView.scene.rootNode.addChildNode(planeNode)
  • SCNNode 是 SceneKit 用来指定三维物体空间属性的,按照一种树形的层级结构来组织物体。
  • 创建 SCNNode 有三种目的,一是做为几何对象,显示几何图形,需要设置一个 geometry 属性;二是做为容器,将其他 SCNNode 添加进来组合到一起;三是放置一些其他对象,比如 SCNCamera SCNLight
  • SCNNode 作为几何对象时,决定一个几何实体的位置、缩放、旋转等对外的属性。
  • SCNNode 作为容器时,其内部的所有对象组合起来做为一个整体,外面并不用关心 SCNNode 内部到底是有一个 SCNGeometry 还是由多个 SCNNode 组合而成的。
  • SCNNode 放置其他对象,与几何对象一样,一些其他对象也需要位置、缩放、旋转,比如摄像机和点光源。
  • 一个 SCNNode 可以同时具有以上三种用法。一种有趣的用法是在持有 SCNCameraSCNNode 中添加子节点,就会使这个子节点相对于摄像头不动,像固定在屏幕中一样。
  • SCNNode 是按照树形结构存储的,必然有一个根节点,这个根节点是预定义好的, rootNode 属性返回这个场景的根节点。自定义的节点通过添加到根节点来添加到场景中。SCNNodeparent 属性,指示上层节点,rootNodeparent 属性值为 nil
var translation = matrix_identity_float4x4
translation.columns.3.z = -0.1
planeNode.simdTransform = matrix_multiply(currentFrame.camera.transform, translation)
  • 前面讲到 SCNNode 具有位置、缩放、旋转等属性,这几个属性可以通过单独的属性设置,可以通过一个矩阵来表示它们的组合。这些属性所表示的都是相对于父节点坐标系的。

    • 位置:position : SCNVector3,表示这个 SCNNode 所在的位置。
    • 缩放:scale : SCNVector3,表示这个 SCNNode 自身三个维度的缩放比例。
    • 旋转:旋转是最有趣的,有3个属性可以用来表示旋转。注意 SceneKit 所有 API 涉及到的角度都是弧度制。
      • eulerAngles : SCNVector3,欧拉角,表示节点分别围绕三个坐标轴旋转的角度,是一种比较简单的表示方法,这种方法有一个缺点就是万向节死锁。简单讲就是使用欧拉角时在特定的组合下回丢失某个轴的旋转角度,因此只建议在固定角度经过试验的情况下或者只有一个轴旋转时使用。
      • rotation : SCNVector4,向量的 x, y, z 表示一个三维的方向向量,w 表示围绕这个向量旋转的角度。这个属性是很好理解的,对于任意两个向量,总能找到一个向量和一个角度使其中一个变换为另一个。
      • orientation : SCNQuaternion,其实 SCNQuaternion 类型是 SCNVector4 的别名,但表示的含义不一样,是比较难理解的一个数学概念……wiki传送门,适合在不需要直观理解旋转角度的情况下使用,比如一个节点要用动画的方式旋转到与另一个节点同样的朝向,不用关心具体的值,只需要使用动画改变这个属性即可。
    • 矩阵:可以表达以上三种属性的组合,它是节点从默认状态变换为当前状态的一个变换矩阵,设置这个属性相当于先将节点变换回默认状态,再执行这个矩阵变换。那么矩阵与上面三种属性有什么关系呢?分别设置三种属性的先后顺序不会对结果产生影响,是因为三种属性的设置等价于更新变换矩阵,而这个变换矩阵的构造顺序是固定的,即先缩放和旋转(都围绕中心所以这两种不互相影响),再位移。
  • simd:指的是 intel 的一个向量指令集,定义了丰富的向量结构与运算,可以直接在 SceneKit 中使用,上面所说的属性都有另一个以 simd 为前缀的版本用来返回或者设置 simd 类型。SceneKit 也定义了带有 SCN 前缀的一些结构和方法来进行向量和矩阵的运算,但运算方法比较少,有时还需要借助 simd 的一些方法。

  • 矩阵乘法:代码中使截图浮在摄像头前面并朝向摄像头,达到这样效果是因为将图片做了几次变换。

    • 首先 SCNPlane 默认是一个垂直于z轴的平面,将它向z轴反方向位移0.1米,就可以在 (0, 0, 0) 的位置看到平面后移了一些,对应的代码是 translation.columns.3.z = -0.1,如果将矩阵 translation 设置给平面节点,那么它只会固定在 (0, 0, -0.1) 的位置,多次截图的位置都相同,与摄像头当前的位置没有关系。
    • 然后,假设只需要把截图平面变换到与摄像头相同的状态,只需要将摄像头的矩阵 currentFrame.camera.transform 设置给平面节点即可。但这样平面就与摄像头重合了,我们最终要的效果是在摄像头前面0.1米放置截图,这样就要找到将位移和摄像头状态组合起来的方法。
    • 两个变换矩阵结合是有先后顺序的,根据我们的需求可得知应该先将平面移动一下,再将它变换为摄像头的状态。因此就需要将两个矩阵相乘 translation * currentFrame.camera.transform,将结果作为平面节点的矩阵。
    • 注意 simd 提供的 matrix_multiply 方法的两个参数的顺序与它的数学含义的顺序是相反的,即A * B 应写成 matrix_multiply(B, A)。再注意:SCNMatrix4 也提供了一个矩阵乘法运算 SCNMatrix4Mult,它的顺序是正常的,即 A * B 应写成 SCNMatrix4Mult(A, B)

以上代码解析完毕,这个 demo 虽然小,但是涉及到了很多 SceneKit 的概念和用法。比较难以理解的是最后的矩阵变换,可能需要复习一下数学知识。矩阵使用熟练以后可以实现很多有趣的功能,后面的文章中会包含一些相关的例子。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • 简介 增强现实技术(Augmented Reality,简称 AR),是一种实时地计算摄影机影像的位置及角度并加上...
    牛奈奈阅读 1,265评论 1 3
  • 引言ARKit 为开发 iPhone 和 iPad 增强现实(AR)app 提供了一个前沿平台。本文为你介绍 AR...
    蚂蚁安然阅读 9,257评论 0 14
  • ARKit ARKit框架通过集成iOS设备摄像头和运动功能,在您的应用程序或游戏中产生增强现实体验。 概述 增强...
    暗夜夜夜行路阅读 5,794评论 0 17
  • 一切应该都是最好的安排
    焕语阅读 264评论 0 1
  • 彼岸烟波流转,可有人寻我。对岸繁华三千,可有人渡我。 第一次听到“摆渡人”这个名字,是通过看克莱儿·麦克福尔的《摆...
    往生地狱阅读 1,388评论 0 0