-
Notifications
You must be signed in to change notification settings - Fork 214
/
Copy pathRotateWithLocationProvider.cs
164 lines (138 loc) · 4.77 KB
/
RotateWithLocationProvider.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
namespace Mapbox.Examples
{
using Mapbox.Unity.Location;
using UnityEngine;
public class RotateWithLocationProvider : MonoBehaviour
{
/// <summary>
/// Location property used for rotation: false=Heading (default), true=Orientation
/// </summary>
[SerializeField]
[Tooltip("Per default 'UserHeading' (direction the device is moving) is used for rotation. Check to use 'DeviceOrientation' (where the device is facing)")]
bool _useDeviceOrientation;
/// <summary>
///
/// </summary>
[SerializeField]
[Tooltip("Only evaluated when 'Use Device Orientation' is checked. Subtracts UserHeading from DeviceOrientation. Useful if map is rotated by UserHeading and DeviceOrientation should be displayed relative to the heading.")]
bool _subtractUserHeading;
/// <summary>
/// The rate at which the transform's rotation tries catch up to the provided heading.
/// </summary>
[SerializeField]
[Tooltip("The rate at which the transform's rotation tries catch up to the provided heading. ")]
float _rotationFollowFactor = 1;
/// <summary>
/// Set this to true if you'd like to adjust the rotation of a RectTransform (in a UI canvas) with the heading.
/// </summary>
[SerializeField]
bool _rotateZ;
/// <summary>
/// <para>Set this to true if you'd like to adjust the sign of the rotation angle.</para>
/// <para>eg angle passed in 63.5, angle that should be used for rotation: -63.5.</para>
/// <para>This might be needed when rotating the map and not objects on the map.</para>
/// </summary>
[SerializeField]
[Tooltip("Set this to true if you'd like to adjust the sign of the rotation angle. eg angle passed in 63.5, angle that should be used for rotation: -63.5.")]
bool _useNegativeAngle;
/// <summary>
/// Use a mock <see cref="T:Mapbox.Unity.Location.TransformLocationProvider"/>,
/// rather than a <see cref="T:Mapbox.Unity.Location.EditorLocationProvider"/>.
/// </summary>
[SerializeField]
bool _useTransformLocationProvider;
Quaternion _targetRotation;
/// <summary>
/// The location provider.
/// This is public so you change which concrete <see cref="ILocationProvider"/> to use at runtime.
/// </summary>
ILocationProvider _locationProvider;
public ILocationProvider LocationProvider
{
private get
{
if (_locationProvider == null)
{
_locationProvider = _useTransformLocationProvider ?
LocationProviderFactory.Instance.TransformLocationProvider : LocationProviderFactory.Instance.DefaultLocationProvider;
}
return _locationProvider;
}
set
{
if (_locationProvider != null)
{
_locationProvider.OnLocationUpdated -= LocationProvider_OnLocationUpdated;
}
_locationProvider = value;
_locationProvider.OnLocationUpdated += LocationProvider_OnLocationUpdated;
}
}
Vector3 _targetPosition;
void Start()
{
LocationProvider.OnLocationUpdated += LocationProvider_OnLocationUpdated;
}
void OnDestroy()
{
if (LocationProvider != null)
{
LocationProvider.OnLocationUpdated -= LocationProvider_OnLocationUpdated;
}
}
void LocationProvider_OnLocationUpdated(Location location)
{
float rotationAngle = _useDeviceOrientation ? location.DeviceOrientation : location.UserHeading;
if (_useNegativeAngle) { rotationAngle *= -1f; }
// 'Orientation' changes all the time, pass through immediately
if (_useDeviceOrientation)
{
if (_subtractUserHeading)
{
if (rotationAngle > location.UserHeading)
{
rotationAngle = 360 - (rotationAngle - location.UserHeading);
}
else
{
rotationAngle = location.UserHeading - rotationAngle + 360;
}
if (rotationAngle < 0) { rotationAngle += 360; }
if (rotationAngle >= 360) { rotationAngle -= 360; }
}
_targetRotation = Quaternion.Euler(getNewEulerAngles(rotationAngle));
}
else
{
// if rotating by 'Heading' only do it if heading has a new value
if (location.IsUserHeadingUpdated)
{
_targetRotation = Quaternion.Euler(getNewEulerAngles(rotationAngle));
}
}
}
private Vector3 getNewEulerAngles(float newAngle)
{
var localRotation = transform.localRotation;
var currentEuler = localRotation.eulerAngles;
var euler = Mapbox.Unity.Constants.Math.Vector3Zero;
if (_rotateZ)
{
euler.z = -newAngle;
euler.x = currentEuler.x;
euler.y = currentEuler.y;
}
else
{
euler.y = newAngle;
euler.x = currentEuler.x;
euler.z = currentEuler.z;
}
return euler;
}
void Update()
{
transform.localRotation = Quaternion.Lerp(transform.localRotation, _targetRotation, Time.deltaTime * _rotationFollowFactor);
}
}
}