AR + GPS Location  3.0.0
All Classes Namespaces Functions Variables Properties Events Pages
GroundHeight.cs
1 using System;
2 using UnityEngine;
3 
4 
5 #if !ARGPS_USE_VUFORIA
6 using UnityEngine.XR.ARFoundation;
7 using UnityEngine.XR.ARSubsystems;
8 #endif
9 // ReSharper disable once RedundantUsingDirective
10 using Logger = ARLocation.Utils.Logger;
11 
12 #if ARGPS_USE_VUFORIA
13 using Vuforia;
14 #endif
15 
16 namespace ARLocation
17 {
22  [DisallowMultipleComponent]
23  public class GroundHeight : MonoBehaviour
24  {
25  [Serializable]
26  public class SettingsData
27  {
28  [Range(0, 10)]
29  public float InitialGroundHeightGuess = 1.4f;
30  [Range(0, 10)]
31  public float MinGroundHeight = 0.4f;
32  [Range(0, 10)]
33  public float MaxGroundHeight = 3.0f;
34  [Range(0, 1)]
35  public float Smoothing = 0.05f;
36 
37  public float Altitude;
38 
39  public bool DisableUpdate;
40 
41  [Range(0, 0.1f)]
42  public float Precision = 0.005f;
43  public bool UseArLocationConfigSettings = true;
44 
45 #if ARGPS_USE_VUFORIA
46  public float MinHitDistance = 0.5f;
47 #endif
48  }
49 
50  [Serializable]
51  public class StateData
52  {
53  public float CurrentGroundY;
54  public float CurrentPlaneDistance = -1.0f;
55  public Vector3 CurrentPlaneCenter;
56  public bool NeedsUpdate = true;
57  }
58 
59  public float CurrentGroundY => state.CurrentGroundY;
60 
61  public SettingsData Settings = new SettingsData();
62  private readonly StateData state = new StateData();
63  private Camera mainCamera;
64 
65 #if !ARGPS_USE_VUFORIA
66  private ARPlaneManager arPlaneManager;
67  private float targetY;
68 
69  void Start()
70  {
71  arPlaneManager = FindObjectOfType<ARPlaneManager>();
72  var arSessionOrigin = FindObjectOfType<ARSessionOrigin>();
73  mainCamera = ARLocationManager.Instance.MainCamera;
74 
75  if (arPlaneManager == null)
76  {
77  if (arSessionOrigin == null)
78  {
79  Debug.LogWarning("[AR+GPS][GroundHeight#Start]: ARSessionOrigin not present in the scene!");
80  return;
81  }
82 
83  arPlaneManager = arSessionOrigin.gameObject.AddComponent<ARPlaneManager>();
84  arPlaneManager.detectionMode = PlaneDetectionMode.Horizontal;
85  }
86 
87  if (Settings.UseArLocationConfigSettings)
88  {
89  Settings.MaxGroundHeight = ARLocation.Config.MaxGroundHeight;
90  Settings.MinGroundHeight = ARLocation.Config.MinGroundHeight;
91  Settings.InitialGroundHeightGuess = ARLocation.Config.InitialGroundHeightGuess;
92  Settings.Smoothing = ARLocation.Config.GroundHeightSmoothingFactor;
93  }
94 
95  state.CurrentGroundY = -Settings.InitialGroundHeightGuess;
96 
97  arPlaneManager.planesChanged += ArPlaneManagerOnPlanesChanged;
98 
99  UpdateObjectHeight();
100  }
101 
102  void OnEnable() {
103  if (arPlaneManager)
104  {
105  arPlaneManager.planesChanged += ArPlaneManagerOnPlanesChanged;
106  }
107  }
108 
109  void OnDisable() {
110  if (arPlaneManager)
111  {
112  arPlaneManager.planesChanged -= ArPlaneManagerOnPlanesChanged;
113  }
114  }
115 
116  private void ArPlaneManagerOnPlanesChanged(ARPlanesChangedEventArgs eventArgs)
117  {
118  var addedPlanes = eventArgs.added;
119  var updatedPlanes = eventArgs.updated;
120 
121  if (addedPlanes.Count <= 0 && updatedPlanes.Count <= 0)
122  {
123  // Debug.Log("[AR+GPS][GroundHeight#ArPlaneManagerOnPlanesChanged]: No added or modified planes!");
124  return;
125  }
126 
127  foreach (ARPlane plane in addedPlanes)
128  {
129  ProcessPlane(plane);
130  }
131 
132  foreach (ARPlane plane in updatedPlanes)
133  {
134  ProcessPlane(plane);
135  }
136 
137  UpdateObjectHeight();
138  }
139 
140  private void ProcessPlane(ARPlane plane)
141  {
142  // Debug.Log("[AR+GPS][GroundHeight#ProcessPlane]: Processing plane " + plane.trackableId.subId1 + ", " + plane.trackableId.subId2);
143 
144  if (plane.alignment != PlaneAlignment.HorizontalDown && plane.alignment != PlaneAlignment.HorizontalUp)
145  {
146  // Debug.LogWarning("[AR+GPS][GroundHeight#ProcessPlane]: Wrong plane alignment!");
147  return;
148  }
149 
150  if (!IsValidHeightForGround(plane.center.y))
151  {
152  // Debug.LogWarning("[AR+GPS][GroundHeight#ProcessPlane]: Invalid plane height!");
153  return;
154  }
155 
156  var distance = MathUtils.HorizontalDistance(transform.position, plane.center);
157 
158  if (!(state.CurrentPlaneDistance < 0) && (distance >= state.CurrentPlaneDistance))
159  {
160  // Debug.LogWarning("[AR+GPS][GroundHeight#ProcessPlane]: Plane too far!");
161  return;
162  }
163 
164  // Debug.Log("[AR+GPS][GroundHeight#ProcessPlane]: New plane Y: " + plane.center.y);
165 
166  state.CurrentPlaneDistance = distance;
167  state.CurrentGroundY = plane.center.y;
168  state.CurrentPlaneCenter = plane.center;
169 
170  state.NeedsUpdate = true;
171  }
172 #else
173  private PlaneFinderBehaviour planeFinderBehaviour;
174 
175  private void Start()
176  {
177  planeFinderBehaviour = FindObjectOfType<PlaneFinderBehaviour>();
178  mainCamera = ARLocationManager.Instance.MainCamera;
179 
180  if (planeFinderBehaviour == null)
181  {
182  Logger.WarnFromMethod("VuforiaGroundHeight", "Start", "No planeFinderBehaviour!");
183  }
184 
185  if (Settings.UseArLocationConfigSettings)
186  {
187  Settings.MaxGroundHeight = ARLocation.Config.MaxGroundHeight;
188  Settings.MinGroundHeight = ARLocation.Config.MinGroundHeight;
189  Settings.InitialGroundHeightGuess = ARLocation.Config.InitialGroundHeightGuess;
190  Settings.MinHitDistance = ARLocation.Config.VuforiaGroundHitTestDistance;
191  Settings.Smoothing = ARLocation.Config.GroundHeightSmoothingFactor;
192  }
193 
194  state.CurrentGroundY = -Settings.InitialGroundHeightGuess;
195 
196  planeFinderBehaviour.Height = Settings.InitialGroundHeightGuess;
197  planeFinderBehaviour.HitTestMode = HitTestMode.AUTOMATIC;
198  planeFinderBehaviour.OnAutomaticHitTest.AddListener(HitTestHandler);
199  planeFinderBehaviour.OnInteractiveHitTest.AddListener(HitTestHandler);
200 
201  UpdateObjectHeight();
202  }
203 
204  private void HitTestHandler(HitTestResult result)
205  {
206  //Logger.LogFromMethod("VuforiaGroundHeight", "HitTestHandler", $"result.Position = {result.Position}");
207 
208  // If the ground height is not in range, reject
209  // var height = -1.0f * result.Position.y;
210  if (!IsValidHeightForGround(result.Position.y))
211  {
212  //Logger.LogFromMethod("VuforiaGroundHeight", "HitTestHandler", $"Not in range: {result.Position.y} {height}) > {Settings.MinGroundHeight}");
213  return;
214  }
215 
216  var distanceToObject = MathUtils.HorizontalDistance(transform.position, result.Position);
217 
218  // If hit to close to previous hit do nothing
219  if (state.CurrentPlaneDistance >= 0 && distanceToObject <= Settings.MinHitDistance)
220  {
221  //Logger.LogFromMethod("VuforiaGroundHeight", "HitTestHandler", $"Too close :{distanceToObject}");
222  return;
223  }
224 
225  // If there is no previous hit, or if the new hit is closes to the object, apply new
226  // hit point.
227  if (state.CurrentPlaneDistance < 0 || distanceToObject < state.CurrentPlaneDistance)
228  {
229  state.CurrentPlaneDistance = distanceToObject;
230  state.CurrentGroundY = result.Position.y;
231  state.NeedsUpdate = true;
232 
233  UpdateObjectHeight();
234 
235  // Logger.LogFromMethod("VuforiaGroundHeight", "HitTestHandler", $"New ground Y = {state.CurrentGroundY}");
236  }
237  }
238 
239 #endif
240  private void UpdateObjectHeight()
241  {
242  if (!state.NeedsUpdate) return;
243 
244  // Debug.Log("[AR+GPS][GroundHeight#UpdateObjectHeight]: Setting Y to " + state.CurrentGroundY);
245 
246  if (Settings.Smoothing <= 0)
247  {
248  transform.position = MathUtils.SetY(transform.position, state.CurrentGroundY + Settings.Altitude);
249  }
250 
251  state.NeedsUpdate = false;
252  }
253 
254  private bool IsValidHeightForGround(float y)
255  {
256  var diff = (mainCamera.transform.position.y - y);
257 
258  return (diff >= Settings.MinGroundHeight) && (diff <= Settings.MaxGroundHeight);
259  }
260 
261  public void Update()
262  {
263  if (Settings.Smoothing <= 0 || Settings.DisableUpdate) return;
264 
265  if (Mathf.Abs(transform.position.y - (state.CurrentGroundY + Settings.Altitude)) <= Settings.Precision)
266  {
267  transform.position = MathUtils.SetY(transform.position, (state.CurrentGroundY + Settings.Altitude));
268  return;
269  }
270 
271  var t = 1.0f - Mathf.Pow(Settings.Smoothing, Time.deltaTime);
272  var position = transform.position;
273  var value = Mathf.Lerp(position.y, (state.CurrentGroundY + Settings.Altitude), t);
274 
275  position = MathUtils.SetY(position, value);
276  transform.position = position;
277  }
278  }
279 }
ARLocation.GroundHeight.SettingsData
Definition: GroundHeight.cs:27
ARLocation.ARLocationManager
This Component manages all positioned GameObjects, synchronizing their world position in the scene wi...
Definition: ARLocationManager.cs:30
ARLocation.GroundHeight.StateData
Definition: GroundHeight.cs:52
ARLocation.Utils.Logger
Definition: Logger.cs:8
ARLocation.Utils.Singleton.Instance
static T Instance
Access singleton instance through this propriety.
Definition: Singleton.cs:18
ARLocation.Utils
Definition: CreatePointOfInterestTextMeshes.cs:9
ARLocation
Definition: ARLocationConfigInspector.cs:7
ARLocation.GroundHeight
This component will change the Y component of the GameObject's position, so that it is set to the lev...
Definition: GroundHeight.cs:24