AR + GPS Location  3.0.0
All Classes Namespaces Functions Variables Properties Events Pages
Hotspot.cs
1 using System;
2 using UnityEngine;
3 using UnityEngine.Events;
4 using Logger = ARLocation.Utils.Logger;
5 
6 namespace ARLocation {
7 
8  [AddComponentMenu("AR+GPS/Hotspot")]
9  [HelpURL("https://http://docs.unity-ar-gps-location.com/guide/#hotspot")]
10  [DisallowMultipleComponent]
11  public class Hotspot : MonoBehaviour
12  {
13  [Serializable]
14  public class OnHotspotActivatedUnityEvent : UnityEvent<GameObject> {}
15 
16  [Serializable]
17  public class OnHotspotLeaveUnityEvent : UnityEvent<GameObject> {}
18 
19  [Serializable]
20  public enum PositionModes {
21  HotspotCenter,
22  CameraPosition
23  };
24 
25  [Serializable]
26  public class HotspotSettingsData
27  {
28  [Tooltip("The prefab/GameObject that will be instantiated by the Hotspot.")]
29  public GameObject Prefab;
30 
31  [Tooltip("The positioning mode. 'HotspotCenter' means the object will be instantiated at the Hotpot's center geo-location. 'CameraPosition' means it will be positioned at the front of the camera.")]
32  public PositionModes PositionMode;
33 
34  [Tooltip("The distance from the center that the user must be located to activate the Hotspot.")]
35  public float ActivationRadius = 4.0f;
36 
37  [Tooltip("If true, align the instantiated object to face the camera (horizontally).")]
38  public bool AlignToCamera = true;
39 
40  [Tooltip("If 'PositionMode' is set to 'CameraPosition', how far from the camera the instantiated object should be placed.")]
41  public float DistanceFromCamera = 3.0f;
42  }
43 
44  [Serializable]
45  public class StateData
46  {
47  public bool Activated;
48  public GameObject Instance;
49  public Location Location;
50  public bool EmittedLeaveHotspotEvent;
51  }
52 
54  public HotspotSettingsData HotspotSettings = new HotspotSettingsData();
55 
56  [Space(4.0f)]
57 
58  [Header("Debug")]
59  [Tooltip("When debug mode is enabled, this component will print relevant messages to the console. Filter by 'Hotspot' in the log output to see the messages.")]
60  public bool DebugMode;
61 
62  [Space(4.0f)]
63 
64 
65  [Header("Events")]
66  [Tooltip("If true, monitor distance to emit the 'OnLeaveHotspot' event. If you don't need it, keep this disabled for better performance.")]
67  public bool UseOnLeaveHotspotEvent;
68 
69  [Tooltip("Event for the Hotspot is activated.")]
70  public OnHotspotActivatedUnityEvent OnHotspotActivated = new OnHotspotActivatedUnityEvent();
71 
72  [Tooltip("This event will be emited only once, when the user leaves the hotspot area after it is activated.")]
73  public OnHotspotLeaveUnityEvent OnHotspotLeave = new OnHotspotLeaveUnityEvent();
74 
75  private readonly StateData state = new StateData();
76 
77  private Transform root;
78  private Camera arCamera;
79  private double currentDistance;
80 
81  // ReSharper disable once UnusedMember.Global
82  public GameObject Instance => state.Instance;
83 
84  // ReSharper disable once MemberCanBePrivate.Global
85  public Location Location {
86  // ReSharper disable once UnusedMember.Global
87  get => state.Location;
88  set => state.Location = value;
89  }
90 
94  public float CurrentDistance => (float) currentDistance;
95 
96  void Start()
97  {
98  var arLocationProvider = ARLocationProvider.Instance;
99  arLocationProvider.Provider.LocationUpdatedRaw += Provider_LocationUpdatedRaw;
100 
101  root = ARLocationManager.Instance.gameObject.transform;
102 
103  if (state.Location == null)
104  {
105  state.Location = LocationSettings.GetLocation();
106  }
107 
108  arCamera = ARLocationManager.Instance.MainCamera;
109 
110  if (arLocationProvider.IsEnabled)
111  {
112  Provider_LocationUpdatedRaw(arLocationProvider.CurrentLocation, arLocationProvider.LastLocation);
113  }
114  }
115 
116  public void Restart()
117  {
118  state.Activated = false;
119  state.EmittedLeaveHotspotEvent = false;
120  state.Instance = null;
121 
122  Destroy(state.Instance);
123  }
124 
125 
126  private void Provider_LocationUpdatedRaw(LocationReading currentLocation, LocationReading lastLocation)
127  {
128  if (state.Activated) return;
129 
130  Logger.LogFromMethod("Hotspot", "LocationUpdatedRaw", $"({gameObject.name}): New device location {currentLocation}", DebugMode);
131 
132  currentDistance = Location.HorizontalDistance(currentLocation.ToLocation(), state.Location);
133 
134  var delta = Location.HorizontalVectorFromTo(currentLocation.ToLocation(), state.Location);
135 
136  if (currentDistance <= HotspotSettings.ActivationRadius)
137  {
138  ActivateHotspot(new Vector3((float) delta.x, 0, (float) delta.y));
139  }
140  else
141  {
142  Logger.LogFromMethod("Hotspot", "LocationUpdatedRaw", $"({gameObject.name}): No activation - distance = {currentDistance}", DebugMode);
143  }
144  }
145 
146  private void ActivateHotspot(Vector3 delta)
147  {
148  Logger.LogFromMethod("Hotspot", "ActivateHotspot", $"({gameObject.name}): Activating hotspot...", DebugMode);
149 
150  if (HotspotSettings.Prefab == null) return;
151 
152  state.Instance = Instantiate(HotspotSettings.Prefab, HotspotSettings.AlignToCamera ? gameObject.transform : root);
153 
154  switch (HotspotSettings.PositionMode)
155  {
156  case PositionModes.HotspotCenter:
157  state.Instance.transform.position = arCamera.transform.position + delta;
158  break;
159  case PositionModes.CameraPosition:
160  var transform1 = arCamera.transform;
161  var forward = transform1.forward;
162  forward.y = 0;
163  state.Instance.transform.position = transform1.position + forward * HotspotSettings.DistanceFromCamera;
164  break;
165  }
166 
167  if (HotspotSettings.AlignToCamera)
168  {
169  state.Instance.transform.LookAt(arCamera.transform);
170  }
171 
172  state.Instance.AddComponent<GroundHeight>();
173  state.Instance.name = name + " (Hotspot)";
174 
175  state.Activated = true;
176 
177  Logger.LogFromMethod("Hotspot", "ActivateHotspot", $"({name}): Hotspot activated", DebugMode);
178 
179  OnHotspotActivated?.Invoke(state.Instance);
180  }
181 
182  // ReSharper disable once UnusedMethodReturnValue.Global
183  // ReSharper disable once MemberCanBePrivate.Global
184  public static Hotspot AddHotspotComponent(GameObject go, Location location, HotspotSettingsData settings)
185  {
186  var hotspot = go.AddComponent<Hotspot>();
187  hotspot.Location = location.Clone();
188  hotspot.HotspotSettings = settings;
189 
190  return hotspot;
191  }
192 
193  // ReSharper disable once UnusedMember.Global
194  public static GameObject CreateHotspotGameObject(Location location, HotspotSettingsData settings,
195  string name = "GPS Hotspot")
196  {
197  var go = new GameObject(name);
198 
199  AddHotspotComponent(go, location, settings);
200 
201  return go;
202  }
203 
204  void Update()
205  {
206  if (UseOnLeaveHotspotEvent && state.Activated && !state.EmittedLeaveHotspotEvent)
207  {
208  var distance = Vector3.Distance(arCamera.transform.position, state.Instance.transform.position);
209  if (distance >= HotspotSettings.ActivationRadius)
210  {
211  OnHotspotLeave?.Invoke(state.Instance);
212  state.EmittedLeaveHotspotEvent = true;
213  }
214  }
215  }
216  }
217 }
ARLocation.Hotspot.OnHotspotActivatedUnityEvent
Definition: Hotspot.cs:14
ARLocation.ARLocationProvider
Definition: ARLocationProvider.cs:17
ARLocation.ARLocationManager
This Component manages all positioned GameObjects, synchronizing their world position in the scene wi...
Definition: ARLocationManager.cs:30
ARLocation.Location.HorizontalVectorFromTo
static DVector2 HorizontalVectorFromTo(Location l1, Location l2)
Calculates the horizontal vector pointing from l1 to l2, in meters.
Definition: Location.cs:243
ARLocation.Hotspot
Definition: Hotspot.cs:12
ARLocation.Location
Represents a geographical location.
Definition: Location.cs:19
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.Hotspot.HotspotSettingsData
Definition: Hotspot.cs:27
ARLocation.Utils
Definition: CreatePointOfInterestTextMeshes.cs:9
ARLocation.Location.HorizontalDistance
static double HorizontalDistance(Location l1, Location l2)
Calculates the horizontal distance according to the current function set in the configuration.
Definition: Location.cs:214
ARLocation.Hotspot.OnHotspotLeaveUnityEvent
Definition: Hotspot.cs:17
ARLocation.Hotspot.CurrentDistance
float CurrentDistance
Returns the current user distance to the Hotspot center.
Definition: Hotspot.cs:94
ARLocation.Hotspot.StateData
Definition: Hotspot.cs:46
ARLocation.PlaceAtLocation
Apply to a GameObject to place it at a specified geographic location.
Definition: PlaceAtLocation.cs:54
ARLocation
Definition: ARLocationConfigInspector.cs:7
ARLocation.PlaceAtLocation.LocationSettingsData
Definition: PlaceAtLocation.cs:89