这篇文章的应用场景是这样子的:
首先我们要做的是一个带有LBS定位服务(比如高德地图、百度地图等)AR功能,在这个场景中,会有一些地图上的”点“(如派出所、学校)是我们需要显示在我们的AR镜头上的,如下图:
图片摘自wuyt2008的博客:《unity3d 尝试 基于地理定位的 增强现实》
本文要解决的问题就是,如何判断这些Point的东南西北,即坐标。
博主采用的开发环境是
- Unity3D: 5.3.5;
- EazyAR: SKD v1.3.0;
- GyroDroid :这个Unity3D 的传感器插件包上网搜一下就有了。
要应用本博客的内容之前,必须先看过wuyt2008的博客《unity3d 尝试 基于地理定位的 增强现实》。
开坑:
实际上我们要实现功能的原理就是根据定位返回的自身(Location)坐标(XL,ZL),跟我们需要标注显示出来的Marker的坐标(XM,ZM)进行坐标系的运算,很简单,
X∆=XL-XM,
Z∆=ZL-ZM,
然后我们再把我们的主摄像机的Position的x、z轴(Y是控制高的,与物体平面定位没有关系)设置为(0,0),这样呢,Marker在Unity中得坐标,就是(X∆,Z∆),并且经过我的测试,Unity中在使用ARCamera的情况下,+X轴指的方向是东90°,+Z轴指的方向是北90°。我们根本不需要做任何其他判断,只需要根据这个特性,再根据经纬度的数据特性,如果X∆的值为负,则表示Marker位于东边(相对自身),如果Z∆的值为负,则表示Marker位于北边(相对自身),所以只需要按照(-X∆,-Z∆)设置Marker的(X,Z)坐标,即可准确的安放Marker的位置。
首先,我们使用EazyAR SDK 中提供的AR摄像机,为什么说它是AR摄像机呢,因为它本身已经根据AR应用的特性,进行了一些优化,还有添加了一些C# Script,所以直接拿来用,你会发现比你自己写一大堆乱七八糟的代码好多了。然后呢,再将GyroDroid插件包中的“MinimalSensorCamera”脚本,绑定到摄像机上,让AR摄像机镜头随着手机的旋转而旋转。
然后就是最重要的一个环节,这里要结合wuyt2008博主的博客中得代码来介绍,我把代码贴一下,并且我自己加了一些注释,助于观看:
1 using UnityEngine;
2 using System.Collections;
3 using System.Collections.Generic;
4 using UnityEngine.UI;
5
6 public class ARMange : MonoBehaviour {
7
8 public List<PlaceInfo> places = new List<PlaceInfo>();//Marker的集合(Marker是什么,请学习一下高德地图api的应用,简单的说,就是你想要标记在地图上的点,比如一些餐厅或景点之类,甚至可以是自定义的点)
9 public GameObject perfab;
10 public PlaceInfo location = new PlaceInfo ();//根据高德API定位后传递过来的自身坐标信息
11
12 public void ShowPlaces(){
13 ClearPlace ();
14
15 for (int i = 0; i < places.Count; i++) {
16
17 GameObject newPlace = Instantiate<GameObject> (perfab);
18 newPlace.transform.parent = this.transform;
19
20 double posZ = places [i].Latitude - location.Latitude;//计算相对距离,z轴(Unity中的坐标系)
21 double posX = places [i].Longitude - location.Longitude;//计算相对距离,x轴(Unity中的坐标系)
22
23 float z = 0;
24 float x = 0;
25 float y = 0;
26
27 if (posZ > 0) {
28 z = 500f;
29 } else {
30 z = -500f;
31 }
32
33 if (posX > 0) {
34 x = 500f;
35 } else {
36 x = -500f;
37 }
38
39 z = z + (float)(posZ * 1000);
40 x = x + (float)(posX * 1000);
41 y = y + i * 20;
42
43 newPlace.transform.position = new Vector3 (x, y, z);//设置Marker
44 newPlace.transform.LookAt (this.transform);
45 newPlace.transform.Rotate (new Vector3 (0f, 180f, 0f));
46
47 newPlace.gameObject.GetComponentInChildren<Text> ().text = places [i].Name;
48 }
49 }
50
51 private void ClearPlace(){
52 GameObject[] oldPlaces = GameObject.FindGameObjectsWithTag ("Place");
53 for (int i = 0; i < oldPlaces.Length; i++) {
54 Destroy (oldPlaces [i].gameObject);
55 }
56 }
57 }
然后我们把Unity3D中,ARCamera(AR摄像机)的Position,设置为(0,0,0)这么做呢,是方便等会儿设定Marker的坐标,只有相机坐标设置为这样子,X∆和Z∆的值才是Marker的直接坐标,否则还得进行一些运算。不过这里值得注意的是,坐标系的设定,应该在CameraDevice对象上,否则是无效的。
然后我们修改上述的代码,修改起来,并不会太难,只需要修改第43行,改成:
20 double posZ = places [i].Latitude - location.Latitude;//计算相对距离,z轴(Unity中的坐标系)
21 double posX = places [i].Longitude - location.Longitude;//计算相对距离,x轴(Unity中的坐标系)
43 newPlace.transform.position = new Vector3 (-x, y, -z);//设置Marker
这样就完事儿了。。。wuyt2008的博客《unity3d 尝试 基于地理定位的 增强现实》最后所提及的偏移问题,貌似没有这种情况,可能是跟EazyAR提供的ARCamera有关,以及镜头自动会有东南西北的方向感。最后感谢wuyt2008,最近在做AR的项目CityHunter,一直在研究如何将高德定位整合进Unity3D来,虽然应用方面与其所描述的不太相干(实际上我要运用的是高德API的电子围栏的功能),但是在操作中可举一反三,果断是学习到了。