SpriteManager 2
 All Classes Functions Variables Enumerations Enumerator Properties
SpriteAnimationPump.cs
1 //-----------------------------------------------------------------
2 // Copyright 2009 Brady Wright and Above and Beyond Software
3 // All rights reserved
4 //-----------------------------------------------------------------
5 
6 
7 #define COROUTINE_PUMP
8 #define PUMP_ALWAYS_RUNNING
9 #define PUMP_EVERY_FRAME
10 #define STOP_ON_LEVEL_LOAD
11 // #define USE_DELTA_TIME // Base animation on Time.deltaTime instead of Time.realtimeSinceStartup
12 
13 
14 using UnityEngine;
15 using System.Collections;
16 
17 
23 public class SpriteAnimationPump : MonoBehaviour
24 {
25  static SpriteAnimationPump instance = null;
26 
27  // The first element in our sprite list:
28  protected static ISpriteAnimatable head;
29 
30  // Used to iterate through the sprite list:
31  protected static ISpriteAnimatable cur;
32 
33 
34  // Pump state vars:
35  static float _timeScale = 1f;
36  static float startTime;
37  static float time;
38  static float elapsed;
39  static float timePaused;
40  static bool isPaused = false;
41  static ISpriteAnimatable next;
42 
43  // Indicates whether the animation pump is
44  // currently running. The pump can also be
45  // turned off by setting this value to false.
46  protected static bool pumpIsRunning = false;
47 
48  // Tells us when the coroutine has run its course.
49  protected static bool pumpIsDone = true;
50 
54  public bool IsRunning
55  {
56  get { return pumpIsRunning; }
57  }
58 
66  public static float timeScale
67  {
68  get
69  {
70 #if USE_DELTA_TIME
71  return Time.timeScale;
72 #else
73  return _timeScale;
74 #endif
75  }
76 
77  set
78  {
79 #if USE_DELTA_TIME
80  Time.timeScale = value;
81  _timeScale = Time.timeScale;
82 #else
83  _timeScale = Mathf.Max(0, value);
84 #endif
85  }
86  }
87 
92  public static float animationPumpInterval = 0.03333f;
93 
94 
95  void Awake()
96  {
97 #if STOP_ON_LEVEL_LOAD
98  _timeScale = 1f;
99  isPaused = false;
100  pumpIsRunning = false;
101  pumpIsDone = true;
102 #else
103  DontDestroyOnLoad(this);
104 #endif
105  instance = this;
106  }
107 
108  // Makes sure we account for realtime passage while paused
109  // due to loss of focus, etc:
110  void OnApplicationPause(bool paused)
111  {
112 #if !USE_DELTA_TIME
113  if (paused && !isPaused)
114  {
115  timePaused = Time.realtimeSinceStartup;
116  }
117  else if (!paused && isPaused)
118  {
119  // See how long we were paused:
120  float pauseDuration = Time.realtimeSinceStartup - timePaused;
121  startTime += pauseDuration;
122  }
123 
124  isPaused = paused;
125 #endif
126  }
127 
135  public void StartAnimationPump()
136  {
137  if (!pumpIsRunning)
138  {
139  pumpIsRunning = true;
140 #if COROUTINE_PUMP
141  StartCoroutine(PumpStarter());
142 #endif
143  }
144  }
145 
146 
147 #if COROUTINE_PUMP
148  // Coroutine that gets the pump started:
149  protected IEnumerator PumpStarter()
150  {
151  while (!pumpIsDone)
152  yield return null;
153 
154  StartCoroutine(AnimationPump());
155  }
156 #endif
157 
164  public static void StopAnimationPump()
165  {
166 #if !PUMP_ALWAYS_RUNNING
167  pumpIsRunning = false;
168  //Instance.StopAllCoroutines();
169 #endif
170  }
171 
172 
173 #if !COROUTINE_PUMP
174 
175  void Update()
176  {
177  if(!pumpIsRunning)
178  return; // Abort
179 
180  // Check for pause:
181  if ((!isPaused && Time.timeScale == 0) ||
182  (isPaused && Time.timeScale != 0))
183  instance.OnApplicationPause(Time.timeScale == 0);
184 
185 #if !USE_DELTA_TIME
186  time = Time.realtimeSinceStartup;
187  elapsed = (time - startTime) * _timeScale;
188  startTime = time;
189 #endif
190 
191  cur = head;
192 
193  while (cur != null)
194  {
195  next = (ISpriteAnimatable)cur.next;
196 #if !USE_DELTA_TIME
197  cur.StepAnim(elapsed);
198 #else
199  cur.StepAnim(Time.deltaTime);
200 #endif
201  cur = next;
202  }
203  }
204 
205 #else
206 
207  // The coroutine that drives animation:
208  protected static IEnumerator AnimationPump()
209  {
210 #if !USE_DELTA_TIME
211  startTime = Time.realtimeSinceStartup;
212 #else
213  startTime = Time.time;
214 #endif
215 
216  pumpIsDone = false;
217 
218  while (pumpIsRunning)
219  {
220  // Check for pause:
221  if ((!isPaused && Time.timeScale == 0) ||
222  (isPaused && Time.timeScale != 0))
223  instance.OnApplicationPause(Time.timeScale == 0);
224 
225 #if !PUMP_EVERY_FRAME
226  yield return new WaitForSeconds(animationPumpInterval);
227 #else
228  yield return null;
229 #endif
230 
231 
232 #if USE_DELTA_TIME
233  time = Time.time;
234  elapsed = time - startTime;
235  startTime = time;
236 #else
237  time = Time.realtimeSinceStartup;
238  elapsed = (time - startTime) * _timeScale;
239  startTime = time;
240 #endif
241 
242  // Start at the beginning:
243  cur = head;
244 
245  while( cur != null )
246  {
247  next = (ISpriteAnimatable)cur.next;
248  cur.StepAnim(elapsed);
249  cur = next;
250  }
251  }
252 
253  pumpIsDone = true;
254  }
255 
256 #endif // end #if !COROUTINE_PUMP
257 
258  public static SpriteAnimationPump Instance
259  {
260  get
261  {
262  if(instance == null)
263  {
264  GameObject go = new GameObject("SpriteAnimationPump");
265  instance = (SpriteAnimationPump)go.AddComponent(typeof(SpriteAnimationPump));
266  }
267 
268  return instance;
269  }
270  }
271 
272  public void OnDestroy()
273  {
274  head = null;
275  cur = null;
276  next = null;
277  instance = null;
278  }
279 
280 
281  // Adds the specified sprite to the animation list.
282  // s: A reference to the sprite to be animated.
283  public static void Add(ISpriteAnimatable s)
284  {
285  if(head != null)
286  {
287  s.next = head;
288  head.prev = s;
289  head = s;
290  }
291  else
292  {
293  head = s;
294 
295  // We've got our first item, so
296  // we need to start the pump:
297  Instance.StartAnimationPump();
298  }
299  }
300 
301  // Removes the specified sprite from the animation list,
302  // thereby not receiving animation updates.
303  // s: A reference to the sprite to be removed.
304  public static void Remove(ISpriteAnimatable s)
305  {
306  if(head == s)
307  {
308  head = (ISpriteAnimatable)s.next;
309 
310  // See if we need to stop the pump:
311  if(head == null)
312  {
314  }
315  }
316  else
317  {
318  if(s.next != null)
319  { // Connect either side:
320  if (s.prev != null)
321  s.prev.next = s.next;
322  s.next.prev = s.prev;
323  }
324  else if(s.prev != null)
325  {
326  // Removing the tail item:
327  s.prev.next = null;
328  }
329  }
330  s.next = null;
331  s.prev = null;
332  }
333 }
bool IsRunning
Returns whether or not the pump is currently running.
static float timeScale
Works like Time.timeScale, only it still works when using realtime tracking. When USE_DELTA_TIME is d...
static float animationPumpInterval
The interval between animation coroutine updates. Defaults to 0.03333f (30 frames per second)...
static void StopAnimationPump()
Stops the animation pump from running. Normally, there is no need to call this directly. Only use this if you want to pause all animations or something.
void StartAnimationPump()
Starts the animation pump coroutine. Normally, there is no need to call this directly. Only use this if you have manually stopped the pump to pause all animations or something.