Liberi
An exergame built for kids with CP!
Cameraman.cs
1 using UnityEngine;
2 using System.Linq;
3 using System.Collections;
4 using System.Collections.Generic;
5 using Janus;
6 
10 [ExecuteInEditMode]
11 [Script(ScriptRole.Client)]
12 public class Cameraman: MonoBehaviour
13 {
17  public const float ShakeShiftInterval = .01f;
18 
19  public static Cameraman Instance
20  {
21  get { return _instance; }
22  }
23 
27  public GameObject Target
28  {
29  get
30  {
31  if (_target != null)
32  return _target.gameObject;
33 
34  return null;
35  }
36  set { Follow(value); }
37  }
38 
42  public Vector3 DefaultTargetOffset = new Vector3(0, 0, -8f);
46  public float DefaultNoFollowRadius;
51  public float DefaultFollowFactor = 5f;
55  [System.NonSerialized]
56  public Vector3 AdditionalOffset = Vector3.zero;
61  public Color DesiredSceneColor;
65  public float ColorCorrectionFactor;
69  public Color DefaultSceneColor { get; private set;}
70 
71  Transform _target;
72  CameraTarget _camTarget;
73  Vector3 _targetOffset;
74  float _noFollowRadius;
75  float _followFactor;
76  Vector3 _shakeVector;
77  Camera _camera;
78 
79  static Cameraman _instance;
80 
81  void Awake ()
82  {
83  _instance = this;
84 
85  camera.transparencySortMode = TransparencySortMode.Orthographic;
86 
87  foreach (var cam in GetComponentsInChildren<Camera>())
88  {
89  cam.transparencySortMode = TransparencySortMode.Orthographic;
90  }
91 
92  _camera = GetComponent<Camera>();
93 
94  DefaultSceneColor = _camera.backgroundColor;
95  DesiredSceneColor = DefaultSceneColor;
96  }
97 
101  public void Follow (GameObject go)
102  {
103  if (go == null)
104  {
105  _target = null;
106  return;
107  }
108 
109  bool snap = _target == null;
110 
111  _target = go.transform;
112 
113  _camTarget = go.GetComponent<CameraTarget>();
114 
115  if (_camTarget != null)
116  {
117  _targetOffset = _camTarget.CameramanOffset;
118  _noFollowRadius = _camTarget.NoFollowRadius;
119  _followFactor = _camTarget.FollowFactor;
120  }
121  else
122  {
123  _targetOffset = DefaultTargetOffset;
124  _noFollowRadius = DefaultNoFollowRadius;
125  _followFactor = DefaultFollowFactor;
126  }
127 
128  AdditionalOffset = Vector3.zero;
129 
130  if (snap)
131  SnapToTarget();
132  }
133 
137  public void Follow (Component component)
138  {
139  Follow(component.gameObject);
140  }
141 
145  public void StopFollowing ()
146  {
147  _target = null;
148 
149  _targetOffset = DefaultTargetOffset;
150  _noFollowRadius = DefaultNoFollowRadius;
151  _followFactor = DefaultFollowFactor;
152 
153  AdditionalOffset = Vector3.zero;
154  }
155 
159  public void SnapToTarget ()
160  {
161  if (_target != null)
162  transform.position = _target.position + _targetOffset;
163  }
164 
170  public void Shake (float scale, float duration)
171  {
172  StartCoroutine(ShakeAsync(scale, duration));
173  }
174 
175  IEnumerator ShakeAsync (float scale, float duration)
176  {
177  float shakeTimeLeft = duration;
178 
179  while (shakeTimeLeft > 0f)
180  {
181  _shakeVector = new Vector3(Random.Range(-.5f, .5f), Random.Range(-.5f, .5f), 0f) * scale;
182  shakeTimeLeft -= ShakeShiftInterval;
183  yield return new WaitForSeconds(ShakeShiftInterval);
184  }
185 
186  _shakeVector = Vector3.zero;
187 
188  yield break;
189  }
190 
194  public void StopShaking ()
195  {
196  _shakeVector = Vector3.zero;
197  StopCoroutine("ShakeAsync");
198  }
199 
200  void Update ()
201  {
202  Vector3 newVelocity = Vector3.zero;
203 
204  if (_target != null)
205  {
206  if (_camTarget != null)
207  {
208  _targetOffset = _camTarget.CameramanOffset;
209  _noFollowRadius = _camTarget.NoFollowRadius;
210  _followFactor = _camTarget.FollowFactor;
211  }
212  else
213  {
214  _targetOffset = DefaultTargetOffset;
215  _noFollowRadius = DefaultNoFollowRadius;
216  _followFactor = DefaultFollowFactor;
217  }
218 
219  Vector3 toTarget = _target.position + _targetOffset + AdditionalOffset - transform.position;
220  float distToTarget = toTarget.magnitude;
221 
222  if (distToTarget != 0f)
223  {
224  Vector3 followVector = toTarget / distToTarget * Mathf.Max(0f, distToTarget - _noFollowRadius);
225  newVelocity += followVector * _followFactor;
226  }
227 
228  _camera.backgroundColor = Color.Lerp(_camera.backgroundColor, DesiredSceneColor, Time.deltaTime * ColorCorrectionFactor);
229  RenderSettings.fogColor = _camera.backgroundColor;
230  }
231 
232  if (_shakeVector != Vector3.zero)
233  newVelocity += _shakeVector;
234 
235  if (newVelocity != Vector3.zero)
236  transform.position += newVelocity * Time.deltaTime;
237  }
238 
239  void OnDestroy ()
240  {
241  if (_instance == this)
242  _instance = null;
243  }
244 }
float ColorCorrectionFactor
Effects the speed at which the camera's color corrects to the desired color.
Definition: Cameraman.cs:65
const float ShakeShiftInterval
Frequency of camera shakes.
Definition: Cameraman.cs:17
Vector3 CameramanOffset
The desired offset from this object to the cameraman.
Definition: CameraTarget.cs:17
Color DefaultSceneColor
The default scene color, set at startup to the background color of the camera.
Definition: Cameraman.cs:69
float NoFollowRadius
The allowed distance between this object and the camera before following behaviour is triggered...
Definition: CameraTarget.cs:21
Vector3 AdditionalOffset
An optional additional offset.
Definition: Cameraman.cs:56
void SnapToTarget()
Bring the camera to the desired position immediately without smooth following.
Definition: Cameraman.cs:159
float FollowFactor
The "strength" of the following behaviour. Use higher values for quicker response. Use lower values for smoother movement.
Definition: CameraTarget.cs:26
GameObject Target
Gets or sets the target for this cameraman to follow.
Definition: Cameraman.cs:28
Color DesiredSceneColor
The desired fog and background color of the scene. The camera will always try to correct to this desi...
Definition: Cameraman.cs:61
Vector3 DefaultTargetOffset
The default desired offset from the target object to follow.
Definition: Cameraman.cs:42
float DefaultFollowFactor
The default "strength" of the following behaviour. Use higher values for quicker response. Use lower values for smoother movement.
Definition: Cameraman.cs:51
void StopFollowing()
Stop following the target.
Definition: Cameraman.cs:145
Camera target class which marks this object to be followed by the cameraman.
Definition: CameraTarget.cs:12
void Follow(GameObject go)
Make this cameraman follow the given object.
Definition: Cameraman.cs:101
void StopShaking()
Stop shaking the camera.
Definition: Cameraman.cs:194
void Follow(Component component)
Make this cameraman follow the given object.
Definition: Cameraman.cs:137
void Shake(float scale, float duration)
Shake the camera.
Definition: Cameraman.cs:170
float DefaultNoFollowRadius
The default allowed distance between the target and the camera before following behaviour is triggere...
Definition: Cameraman.cs:46
Standard cameraman for the game.
Definition: Cameraman.cs:12