【ARFoundation学习笔记】平面检测


写在前面的话

本系列笔记旨在记录作者在学习Unity中的AR开发过程中需要记录的问题和知识点。难免出现纰漏,更多详细内容请阅读原文


文章目录


平面检测属性

AR中检测平面的原理:AR Foundation对摄像机获取的图像进行分析处理,分离图像中的特征点(这些特征点往往是图像中明暗、强弱、颜色变化较大的点);利用VIO和IMU跟踪这些特征点的三维空间信息;在跟踪过程中,对特征点信息进行处理,并尝试用空间中位置相近或者符合一定规律的特征点构建平面,如果成功就是检测出了平面。平面有位置、方向和边界信息,AR Plane Manager负责检测平面以及管理这些检测出来的平面,但它并不负责渲染平面。

在AR Plane Manager中,我们可以设置平面检测的方式,如水平平面(Horizontal)、垂直平面(Vertical)、水平平面&垂直平面(Everything)或者不检测平面(Nothing),检测平面也是一个消耗性能的工作,而根据应用需要选择合适的检测方式可以优化应用性能。

平面本身是一个Trackable对象,因此在AR Session Origin上检测到的时候,AR Plane Manager会实例化一个平面Prefab并挂载AR Plane组件。


可视化平面

AR Plane Manager只负责平面的检测,并不负责平面的渲染。平面渲染通常在检测构建的Prefab上执行,预制体上的脚本如上所示:

红框中的顶点偏差阈值表示只有偏差值在阈值范围内的特征点才被归为同一平面,因此阈值越小检测越精确。AR Plane Mesh Visualizer组件主要是从边界特征点与其他特征点三角化生成一个平面网格,而这个网格由Mesh Renderer进行渲染。默认平面预制体还有一个Line Renderer用于渲染边缘。

书中示例了自定义Shader和渲染脚本以实现定制化的平面渲染。


平面检测的开关控制

csharp 复制代码
15.    public void TogglePlaneDetection()
16.    {
17.       m_ARPlaneManager.enabled = !m_ARPlaneManager.enabled;
18.       string planeDetectionMessage = "";
19.       if (m_ARPlaneManager.enabled)
20.       {
21.          planeDetectionMessage = "禁用平面检测";
22.          SetAllPlanesActive(true);
23.       }
24.       else
25.       {
26.          planeDetectionMessage = "启用平面检测";
27.          SetAllPlanesActive(false);
28.       }

34.    void SetAllPlanesActive(bool value)
35.    {
36.       foreach (var plane in m_ARPlaneManager.trackables)
37.          plane.gameObject.SetActive(value);
38.    }

对书内的代码进行了小小的裁剪。对于平面而言,我们可以通过设置平面物体的Active状态来控制平面的显示。还记得我们说平面是受Manager自动管理的,因此如果我们手动销毁平面可能会引发异常。

显示与隐藏已检测平面

直接关闭平面检测的话,那么程序后续也不会再检测新的平面。有时我们想要隐藏已检测平面的同时保留平面检测功能,以便在显示平面检测时直接显示那些新检测的平面,而不是重新开始检测。

csharp 复制代码
1.  using System.Collections;
2.  using System.Collections.Generic;
3.  using UnityEngine;
4.  using UnityEngine.XR.ARFoundation;
5.  using UnityEngine.UI;
6. 
7.  public class PlaneDisplay : MonoBehaviour
8.  {
9.     public Text m_TogglePlaneDetectionText;
10.    private ARPlaneManager m_ARPlaneManager;
11.    private bool isShow = true;
12.    private List<ARPlane> mPlanes;
13.    void Start()
14.    {
15.       m_ARPlaneManager = GetComponent<ARPlaneManager>();
16.       mPlanes = new List<ARPlane>();
17.       m_ARPlaneManager.planesChanged += OnPlaneChanged;
18.    }
19.    void OnDisable()
20.    {
21.       m_ARPlaneManager.planesChanged -= OnPlaneChanged;
22.    }
23.    #region 显示与隐藏检测的平面
24.    public void TogglePlaneDisplay()
25.    {
26.       string planeDisplayMessage = "";
27.       if (isShow)
28.       {
29.          planeDisplayMessage = "隐藏平面";
30.       }
31.       else
32.       {
33.          planeDisplayMessage = "显示平面";
34.       }
35.       for (int i = mPlanes.Count - 1; i >= 0; i--)
36.       {
37.          if (mPlanes[i] == null || mPlanes[i].gameObject == null)
38.             mPlanes.Remove(mPlanes[i]);
39.          else
40.             mPlanes[i].gameObject.SetActive(isShow);
41.       }
42.       if (m_TogglePlaneDetectionText != null)
43.          m_TogglePlaneDetectionText.text = planeDisplayMessage;
44. 
45.       isShow = !isShow;
46.    }
47. 
48.    private void OnPlaneChanged(ARPlanesChangedEventArgs arg)
49.    {
50.       for (int i = 0; i < arg.added.Count; i++)
51.       {
52.          mPlanes.Add(arg.added[i]);
53.          arg.added[i].gameObject.SetActive(isShow);
54.       }
55.    }
56.    #endregion
57. }

上述代码实现了在不关闭平面检测时隐藏已检测平面的功能。原理就是对平面变化的委托添加一个OnPlaneChanged的处理事件,并从附带的事件参数中获取检测到的平面信息,保存在一个私有的List<ARPlane>中。由于PanelManager中对平面的检测由Manager进行自动管理,因此附带参数Args会产生变化,例如增加新的面,更新已有的面,删除过期的面。

所以我们切换平面检测状态的时候,还需要检测参数Args回传的面是否依旧存在,若不存在,则应当移除。否则切换已经过期的面的状态会引发异常。

csharp 复制代码
37.          if (mPlanes[i] == null || mPlanes[i].gameObject == null)
38.             mPlanes.Remove(mPlanes[i]);

事件注册与撤销一定是成双成对的,上述代码在Start()方法中进行了注册,在OnDisable()方法中撤消了注册,如果事件没有在适当的时机撤销会引发难已排查的错误。

相关推荐
安冬的码畜日常21 分钟前
【Vim Masterclass 笔记24】S10L43 + L44:同步练习10 —— 基于 Vim 缓冲区的各类基础操作练习(含点评课)
笔记·vim·自学笔记·vim同步练习·vim缓冲区·vim buffer·vim缓冲区练习
lozhyf25 分钟前
Go语言-学习一
开发语言·学习·golang
一只码代码的章鱼30 分钟前
粒子群算法 笔记 数学建模
笔记·算法·数学建模·逻辑回归
圆圆滚滚小企鹅。36 分钟前
刷题笔记 贪心算法-1 贪心算法理论基础
笔记·算法·leetcode·贪心算法
mascon43 分钟前
U3D的.Net学习
学习
加德霍克1 小时前
【机器学习】使用scikit-learn中的KNN包实现对鸢尾花数据集或者自定义数据集的的预测
人工智能·python·学习·机器学习·作业
漂亮_大男孩1 小时前
深度学习|表示学习|卷积神经网络|局部链接是什么?|06
深度学习·学习·cnn
杨过姑父1 小时前
ES6 简单练习笔记--变量申明
前端·笔记·es6
l1x1n01 小时前
No.37 笔记 | Python面向对象编程学习笔记:探索代码世界的奇妙之旅
笔记·python·学习
骇客野人1 小时前
【人工智能】循环神经网络学习
人工智能·rnn·学习