SpriteManager 2
 All Classes Functions Variables Enumerations Enumerator Properties
SpriteBase.cs
1 //-----------------------------------------------------------------
2 // Copyright 2010 Brady Wright and Above and Beyond Software
3 // All rights reserved
4 //-----------------------------------------------------------------
5 
6 
7 using UnityEngine;
8 using System.Collections;
9 
10 
15 [System.Serializable]
16 public class UVAnimation
17 {
21  public enum ANIM_END_ACTION
22  {
26  Do_Nothing,
27 
32 
37 
41  Hide,
42 
47  Deactivate,
48 
52  Destroy
53  };
54 
55 
56  protected SPRITE_FRAME[] frames; // Array of values defining the frames of an animation
57 
58  // Animation state vars:
59  protected int curFrame = -1; // The current frame
60  protected int stepDir = 1; // The direction we're currently playing the animation (1=forwards (default), -1=backwards)
61  protected int numLoops = 0; // Number of times we've looped since last animation
62  protected bool playInReverse = false; // Indicates that we've been instructed to play in reverse, as opposed to reversing merely as a result of loopReverse.
63 
64  protected float length; // Length of the clip, in seconds (not taking into account looping, loop reversing, etc)
65 
69  public string name; // The name of the
70 
78  public int loopCycles = 0; // How many times to loop the animation (-1 loop infinitely)
79 
86  public bool loopReverse = false; // Reverse the play direction when the end of the animation is reached? (if true, a loop iteration isn't counted until we return to the beginning)
87 
92  [HideInInspector]
93  public float framerate = 15f; // The rate in frames per second at which to play the animation
94 
102  [HideInInspector]
103  public int index = -1;
104 
110  [HideInInspector]
111  public ANIM_END_ACTION onAnimEnd = ANIM_END_ACTION.Do_Nothing;
112 
117  public int StepDirection
118  {
119  get { return stepDir; }
120  set { SetStepDir(value); }
121  }
122 
123  // Copy constructor:
124  public UVAnimation(UVAnimation anim)
125  {
126  frames = new SPRITE_FRAME[anim.frames.Length];
127  anim.frames.CopyTo(frames, 0);
128 
129  name = anim.name;
130 
131  loopCycles = anim.loopCycles;
132  loopReverse = anim.loopReverse;
133  framerate = anim.framerate;
134  onAnimEnd = anim.onAnimEnd;
135 
136  curFrame = anim.curFrame;
137  stepDir = anim.stepDir;
138  numLoops = anim.numLoops;
139  playInReverse = anim.playInReverse;
140  length = anim.length;
141 
142  CalcLength();
143  }
144 
145  public UVAnimation Clone()
146  {
147  return new UVAnimation(this);
148  }
149 
150  public UVAnimation()
151  {
152  frames = new SPRITE_FRAME[0];
153  }
154 
159  public void Reset()
160  {
161  curFrame = -1;
162  stepDir = 1;
163  numLoops = 0;
164  playInReverse = false;
165  }
166 
167  // Sets the stepDir to -1 and sets the current frame to the end
168  // so that the animation plays in reverse
169  public void PlayInReverse()
170  {
171  stepDir = -1;
172  curFrame = frames.Length;
173  numLoops = 0;
174  playInReverse = true;
175  }
176 
177  public void SetStepDir(int dir)
178  {
179  if (dir < 0)
180  {
181  stepDir = -1;
182  playInReverse = true;
183  }
184  else
185  stepDir = 1;
186  }
187 
188  // Stores the UV and other info of the next frame in 'nextFrame',
189  // returns false if we've reached the end of the animation (this
190  // will never happen if it is set to loop infinitely)
191  public bool GetNextFrame(ref SPRITE_FRAME nextFrame)
192  {
193  if (frames.Length < 1)
194  return false;
195 
196  // See if we can advance to the next frame:
197  if ((curFrame + stepDir) >= frames.Length || (curFrame + stepDir) < 0)
198  {
199  // See if we need to loop (if we're reversing, we don't loop until we get back to the beginning):
200  if (stepDir > 0 && loopReverse)
201  {
202  stepDir = -1; // Reverse playback direction
203  curFrame += stepDir;
204 
205  curFrame = Mathf.Clamp(curFrame, 0, frames.Length - 1);
206 
207  nextFrame = frames[curFrame];
208  }
209  else
210  {
211  // See if we can loop:
212  if (numLoops + 1 > loopCycles && loopCycles != -1)
213  return false;
214  else
215  { // Loop the animation:
216  ++numLoops;
217 
218  if (loopReverse)
219  {
220  stepDir *= -1;
221  curFrame += stepDir;
222  curFrame = Mathf.Clamp(curFrame, 0, frames.Length - 1);
223  }
224  else
225  {
226  if (playInReverse)
227  curFrame = frames.Length - 1;
228  else
229  curFrame = 0;
230  }
231 
232  nextFrame = frames[curFrame];
233  }
234  }
235  }
236  else
237  {
238  curFrame += stepDir;
239  nextFrame = frames[curFrame];
240  }
241 
242  return true;
243  }
244 
245 
252  {
253  return frames[Mathf.Clamp(curFrame, 0, curFrame)];
254  }
255 
256 
262  public SPRITE_FRAME GetFrame(int frame)
263  {
264  return frames[frame];
265  }
266 
267 
281  public SPRITE_FRAME[] BuildUVAnim(Vector2 start, Vector2 cellSize, int cols, int rows, int totalCells)
282  {
283  int cellCount = 0;
284 
285  frames = new SPRITE_FRAME[totalCells];
286 
287  frames[0] = new SPRITE_FRAME(0);
288  frames[0].uvs.x = start.x;
289  frames[0].uvs.y = start.y;
290  frames[0].uvs.xMax = start.x + cellSize.x;
291  frames[0].uvs.yMax = start.y + cellSize.y;
292 
293  for (int row = 0; row < rows; ++row)
294  {
295  for (int col = 0; col < cols && cellCount < totalCells; ++col)
296  {
297  frames[cellCount] = new SPRITE_FRAME(0);
298  frames[cellCount].uvs.x = start.x + cellSize.x * ((float)col);
299  frames[cellCount].uvs.y = start.y - cellSize.y * ((float)row);
300  frames[cellCount].uvs.xMax = frames[cellCount].uvs.x + cellSize.x;
301  frames[cellCount].uvs.yMax = frames[cellCount].uvs.y + cellSize.y;
302 
303  ++cellCount;
304  }
305  }
306 
307  CalcLength();
308 
309  return frames;
310  }
311 
312 
325  public SPRITE_FRAME[] BuildWrappedUVAnim(Vector2 start, Vector2 cellSize, int cols, int rows, int totalCells)
326  {
327  return BuildWrappedUVAnim(start, cellSize, totalCells);
328  }
329 
330 
341  public SPRITE_FRAME[] BuildWrappedUVAnim(Vector2 start, Vector2 cellSize, int totalCells)
342  {
343  int cellCount = 0;
344  Vector2 curPos;
345 
346  frames = new SPRITE_FRAME[totalCells];
347 
348  frames[0] = new SPRITE_FRAME(0);
349  frames[0].uvs.x = start.x;
350  frames[0].uvs.y = start.y;
351  frames[0].uvs.xMax = start.x + cellSize.x;
352  frames[0].uvs.yMax = start.y + cellSize.y;
353 
354  curPos = start;
355 
356  for (cellCount = 1; cellCount < totalCells; ++cellCount)
357  {
358  curPos.x += cellSize.x;
359  if (curPos.x + cellSize.x > 1.01f)
360  {
361  curPos.x = 0;
362  curPos.y -= cellSize.y;
363  }
364 
365  frames[cellCount] = new SPRITE_FRAME(0);
366  frames[cellCount].uvs.x = curPos.x;
367  frames[cellCount].uvs.y = curPos.y;
368  frames[cellCount].uvs.xMax = curPos.x + cellSize.x;
369  frames[cellCount].uvs.yMax = curPos.y + cellSize.y;
370  }
371 
372  return frames;
373  }
374 
375 
382  public void SetAnim(SPRITE_FRAME[] anim)
383  {
384  frames = anim;
385  CalcLength();
386  }
387 
395  public void SetAnim(TextureAnim anim, int idx)
396  {
397  if (anim == null)
398  return;
399  if (anim.spriteFrames == null)
400  return;
401 
402  frames = new SPRITE_FRAME[anim.spriteFrames.Length];
403 
404  index = idx;
405  name = anim.name;
406  loopCycles = anim.loopCycles;
407  loopReverse = anim.loopReverse;
408  framerate = anim.framerate;
409  onAnimEnd = anim.onAnimEnd;
410 
411  try
412  {
413  for (int i = 0; i < frames.Length; ++i)
414  {
415  frames[i] = anim.spriteFrames[i].ToStruct();
416  }
417  }
418  catch (System.Exception err)
419  {
420  Debug.LogError("Exception caught in UVAnimation.SetAnim(). Make sure you have re-built your atlases!\nException: " + err.Message);
421  }
422 
423  CalcLength();
424  }
425 
432  public void AppendAnim(SPRITE_FRAME[] anim)
433  {
434  SPRITE_FRAME[] tempFrames = frames;
435 
436  frames = new SPRITE_FRAME[frames.Length + anim.Length];
437  tempFrames.CopyTo(frames, 0);
438  anim.CopyTo(frames, tempFrames.Length);
439 
440  CalcLength();
441  }
442 
447  public void SetCurrentFrame(int f)
448  {
449  // Allow to go 1 out of bounds since this is
450  // how we prepare to "advance" to the first or
451  // last frame:
452  f = Mathf.Clamp(f, -1, frames.Length + 1);
453 
454  curFrame = f;
455  }
456 
468  public void SetPosition(float pos)
469  {
470  pos = Mathf.Clamp01(pos);
471 
472  // If this is an infinitely looping animation,
473  // or a single-play animation, just set the
474  // position within the clip:
475  if (loopCycles < 1)
476  {
477  SetClipPosition(pos);
478  return;
479  }
480 
481  // The percentage of total animation that is
482  // accounted for by a single loop iteration:
483  float iterationPct = 1f / (loopCycles + 1f);
484 
485  // The loop iteration containing the desired
486  // frame:
487  numLoops = Mathf.FloorToInt(pos / iterationPct);
488 
489  // Portion of our "pos" that is unaccounted for
490  // merely by counting loop iterations:
491  float remainder = pos - (((float)numLoops) * iterationPct);
492 
493  // Position within the "clip" (the "clip" being
494  // the series of frames between the first frame
495  // and last frame, without regard to the loop cycles)
496  float clipPos = remainder / iterationPct;
497 
498  if(loopReverse)
499  {
500  if (clipPos < 0.5f)
501  {
502  curFrame = (int) ( ((float)frames.Length-1) * (clipPos/0.5f) );
503  // We're stepping forward from here:
504  stepDir = 1;
505  }
506  else
507  {
508  curFrame = (frames.Length-1) - (int)(((float)frames.Length-1) * ((clipPos-0.5f) / 0.5f));
509  // We're stepping backwards from here:
510  stepDir = -1;
511  }
512  }
513  else
514  {
515  curFrame = (int) ( ((float)frames.Length-1) * clipPos );
516  }
517  }
518 
533  public void SetClipPosition(float pos)
534  {
535  curFrame = (int)(((float)frames.Length - 1) * pos);
536  }
537 
538  // Calculates the length of the animation clip (not accounting for looping, loop reversing, etc)
539  protected void CalcLength()
540  {
541  length = (1f / framerate) * frames.Length;
542  }
543 
553  public float GetLength()
554  {
555  return length;
556  }
557 
567  public float GetDuration()
568  {
569  // If this loops infinitely, return -1:
570  if(loopCycles < 0)
571  return -1f;
572 
573  float length = GetLength();
574 
575  if (loopReverse)
576  length *= 2f;
577 
578  return length + (loopCycles * length);
579  }
580 
585  public int GetFrameCount()
586  {
587  return frames.Length;
588  }
589 
596  public int GetFramesDisplayed()
597  {
598  if (loopCycles == -1)
599  return -1;
600 
601  int count = frames.Length + (frames.Length * loopCycles);
602 
603  if (loopReverse)
604  count *= 2;
605 
606  return count;
607  }
608 
613  public int GetCurPosition()
614  {
615  return curFrame;
616  }
617 }
618 
619 
624 [System.Serializable]
626 {
631  public Vector2 start;
632 
640  public Vector2 pixelsToNextColumnAndRow;
641 
645  public int cols;
646 
650  public int rows;
651 
655  public int totalCells;
656 
657  // No-arg constructor:
658  public UVAnimation_Auto() {}
659 
660  // Copy constructor:
661  public UVAnimation_Auto(UVAnimation_Auto anim) : base(anim)
662  {
663  start = anim.start;
664  pixelsToNextColumnAndRow = anim.pixelsToNextColumnAndRow;
665  cols = anim.cols;
666  rows = anim.rows;
667  totalCells = anim.totalCells;
668  }
669 
670  public new UVAnimation_Auto Clone()
671  {
672  return new UVAnimation_Auto(this);
673  }
674 
682  {
683  if (totalCells < 1)
684  return null;
685 
686  return this.BuildUVAnim(s.PixelCoordToUVCoord(start), s.PixelSpaceToUVSpace(pixelsToNextColumnAndRow), cols, rows, totalCells);
687  }
688 }
689 
690 
696 [System.Serializable]
697 public class UVAnimation_Multi
698 {
702  public string name; // The name of the animation sequence
703 
707  public int loopCycles = 0; // How many times to loop the animation (-1 loop infinitely)
708 
712  public bool loopReverse = false; // Reverse the play direction when the end of the animation is reached? (if true, a loop iteration isn't counted until we return to the beginning)
713 
717  public float framerate = 15f; // The rate in frames per second at which to play the animation
718 
724  public UVAnimation.ANIM_END_ACTION onAnimEnd = UVAnimation.ANIM_END_ACTION.Do_Nothing;
725 
729  public UVAnimation_Auto[] clips; // The actual sprite animation clips that make up this animation sequence
730 
731  [HideInInspector]
732  public int index;
733 
734  protected int curClip; // Index of the currently-playing clip
735  protected int stepDir = 1; // The direction to step through our clips (1 == forwards, -1 == backwards)
736  protected int numLoops = 0; // Number of times we've looped since last animation
737 
738  protected float duration; // The duration of the animation, accounting for looping and loop reversing.
739 
740 
741  // Working vars:
742  protected bool ret;
743  protected int i;
744  protected int framePos = -1; // Keeps track of which frame (over all) we are on in the animation multi.
745  // In other words, if the multi contains a total of 100 frames across all
746  // sub-animations, then this value will count from 0-99 (and back again if
747  // if we're reversing).
748 
749 
750  public UVAnimation_Multi()
751  {
752  if (clips == null)
753  clips = new UVAnimation_Auto[0];
754  }
755 
756  // Copy constructor:
758  {
759  name = anim.name;
760  loopCycles = anim.loopCycles;
761  loopReverse = anim.loopReverse;
762  framerate = anim.framerate;
763  onAnimEnd = anim.onAnimEnd;
764  curClip = anim.curClip;
765  stepDir = anim.stepDir;
766  numLoops = anim.numLoops;
767  duration = anim.duration;
768 
769  clips = new UVAnimation_Auto[anim.clips.Length];
770  for (int i = 0; i < clips.Length; ++i)
771  clips[i] = anim.clips[i].Clone();
772 
773  CalcDuration();
774  }
775 
776  public UVAnimation_Multi Clone()
777  {
778  return new UVAnimation_Multi(this);
779  }
780 
786  {
787  return clips[curClip];
788  }
789 
794  public int StepDirection
795  {
796  get { return stepDir; }
797  set { stepDir = value; }
798  }
799 
807  {
808  for (i = 0; i < clips.Length; ++i)
809  {
810  clips[i].BuildUVAnim(s);
811  }
812 
813  CalcDuration();
814 
815  return clips;
816  }
817 
818  public bool GetNextFrame(ref SPRITE_FRAME nextFrame)
819  {
820  if (clips.Length < 1)
821  return false;
822 
823  ret = clips[curClip].GetNextFrame(ref nextFrame);
824 
825  if (!ret)
826  {
827  // See if we have another clip in the queue:
828  if ((curClip + stepDir) >= clips.Length || (curClip + stepDir) < 0)
829  {
830  // See if we need to loop (if we're reversing, we don't loop until we get back to the beginning):
831  if (stepDir > 0 && loopReverse)
832  {
833  stepDir = -1; // Reverse playback direction
834  curClip += stepDir;
835 
836  curClip = Mathf.Clamp(curClip, 0, clips.Length - 1);
837 
838  // Make the newly selected clip ready for playing:
839  clips[curClip].Reset();
840  clips[curClip].PlayInReverse();
841  // Go ahead and step one since it will start
842  // at the end, and that's where we are already:
843  clips[curClip].GetNextFrame(ref nextFrame);
844  }
845  else
846  {
847  // See if we can loop:
848  if (numLoops + 1 > loopCycles && loopCycles != -1)
849  return false; // We've reached the end of the last clip
850  else
851  { // Loop the animation:
852  ++numLoops;
853 
854  if (loopReverse)
855  {
856  stepDir *= -1;
857  curClip += stepDir;
858 
859  curClip = Mathf.Clamp(curClip, 0, clips.Length - 1);
860 
861  // Make the newly selected clip ready for playing:
862  clips[curClip].Reset();
863 
864  if (stepDir < 0)
865  clips[curClip].PlayInReverse();
866 
867  // Go ahead and step one since it will start
868  // where we are already:
869  clips[curClip].GetNextFrame(ref nextFrame);
870  }
871  else
872  {
873  curClip = 0;
874  framePos = -1;
875 
876  // Make the newly selected clip ready for playing:
877  clips[curClip].Reset();
878  }
879  }
880  }
881  }
882  else
883  {
884  curClip += stepDir;
885 
886  // Make the newly selected clip ready for playing:
887  clips[curClip].Reset();
888 
889  if (stepDir < 0)
890  {
891  clips[curClip].PlayInReverse();
892  // Go ahead and step one since it will start
893  // at the end, and that's where we are already:
894  clips[curClip].GetNextFrame(ref nextFrame);
895  }
896  }
897 
898  framePos += stepDir;
899 
900  // Get the next frame since our previous
901  // attempt failed, and all that we just
902  // did was to be able to get the next one:
903  clips[curClip].GetNextFrame(ref nextFrame);
904 
905  return true; // Keep playing
906  }
907 
908  framePos += stepDir;
909 
910  /*
911  // Simpler, non-looping logic:
912  if (curClip < clips.Length - 1)
913  {
914  // Go to the next clip:
915  ++curClip;
916  return true; // Keep playing
917  }
918  else
919  return false; // We've reached the end of the last clip
920  */
921 
922  return true;
923  }
924 
925 
932  {
933  return clips[Mathf.Clamp(curClip, 0, curClip)].GetCurrentFrame();
934  }
935 
936 
942  public void AppendAnim(int index, SPRITE_FRAME[] anim)
943  {
944  if (index >= clips.Length)
945  return;
946 
947  clips[index].AppendAnim(anim);
948 
949  CalcDuration();
950  }
951 
956  public void AppendClip(UVAnimation clip)
957  {
958  UVAnimation[] temp;
959  temp = clips;
960 
961  clips = new UVAnimation_Auto[clips.Length + 1];
962  temp.CopyTo(clips, 0);
963 
964  clips[clips.Length - 1] = (UVAnimation_Auto)clip;
965 
966  CalcDuration();
967  }
968 
969  public void PlayInReverse()
970  {
971  for (i = 0; i < clips.Length; ++i)
972  {
973  clips[i].PlayInReverse();
974  }
975 
976  stepDir = -1;
977  framePos = GetFrameCount()-1;
978  curClip = clips.Length - 1;
979  }
980 
986  public void SetAnim(int index, SPRITE_FRAME[] frames)
987  {
988  if (index >= clips.Length)
989  return;
990 
991  clips[index].SetAnim(frames);
992 
993  CalcDuration();
994  }
995 
999  public void Reset()
1000  {
1001  curClip = 0;
1002  stepDir = 1;
1003  numLoops = 0;
1004  framePos = -1;
1005 
1006  for (i = 0; i < clips.Length; ++i)
1007  {
1008  clips[i].Reset();
1009  }
1010  }
1011 
1019  public void SetPosition(float pos)
1020  {
1021  pos = Mathf.Clamp01(pos);
1022 
1023  // If this is an infinitely looping animation,
1024  // or if it is a single-play animation, just
1025  // set the position:
1026  if(loopCycles < 1)
1027  {
1028  SetAnimPosition(pos);
1029  return;
1030  }
1031 
1032  // The percentage of total animation that is
1033  // accounted for by a single loop iteration:
1034  float iterationPct = 1f / (loopCycles + 1f);
1035 
1036  // Find the loop iteration of the desired position:
1037  numLoops = Mathf.FloorToInt(pos / iterationPct);
1038 
1039  // Portion of our "pos" that is unaccounted for
1040  // merely by counting loop iterations:
1041  float remainder = pos - (((float)numLoops) * iterationPct);
1042 
1043  SetAnimPosition(remainder / iterationPct);
1044  }
1045 
1054  public void SetAnimPosition(float pos)
1055  {
1056  int totalFrames = 0;
1057  float pct;
1058  float remaining = pos;
1059 
1060  // Get the total number of frames:
1061  for (int n = 0; n < clips.Length; ++n)
1062  {
1063  totalFrames += clips[n].GetFramesDisplayed();
1064  }
1065 
1066  // Find which clip our desired position is in:
1067  if(loopReverse)
1068  {
1069  if(pos < 0.5f)
1070  {
1071  // We will step forward from here:
1072  stepDir = 1;
1073 
1074  // Adjust to account for the fact that a value
1075  // of .5 in this context means 100% of the way
1076  // from the first frame to the last frame:
1077  remaining *= 2f;
1078 
1079  for (int n = 0; n < clips.Length; ++n)
1080  {
1081  // Get the percentage of our animation
1082  // that is accounted for by this clip:
1083  pct = clips[n].GetFramesDisplayed() / totalFrames;
1084 
1085  // If the distance we have left to go into
1086  // our animation is less than the distance
1087  // accounted for by this clip, this is the
1088  // clip we're looking for!:
1089  if (remaining <= pct)
1090  {
1091  curClip = n;
1092  clips[curClip].SetPosition(remaining / pct);
1093 
1094  framePos = ((int)pct * (totalFrames-1));
1095 
1096  return;
1097  }
1098  else
1099  remaining -= pct;
1100  }
1101  }
1102  else
1103  {
1104  // We will step backward from here:
1105  stepDir = -1;
1106 
1107  // Adjust for the fact that in this context,
1108  // a value of .5 means 0% of the way from the
1109  // last frame to the first frame:
1110  remaining = (remaining-0.5f) / 0.5f;
1111 
1112  for (int n = clips.Length-1; n >= 0; --n)
1113  {
1114  // Get the percentage of our animation
1115  // that is accounted for by this clip:
1116  pct = clips[n].GetFramesDisplayed() / totalFrames;
1117 
1118  // If the distance we have left to go into
1119  // our animation is less than the distance
1120  // accounted for by this clip, this is the
1121  // clip we're looking for!:
1122  if (remaining <= pct)
1123  {
1124  curClip = n;
1125  clips[curClip].SetPosition(1f - (remaining / pct));
1126  clips[curClip].SetStepDir(-1);
1127 
1128  framePos = ((int)pct * (totalFrames - 1));
1129 
1130  return;
1131  }
1132  else
1133  remaining -= pct;
1134  }
1135  }
1136  }
1137  else
1138  {
1139  for (int n = 0; n < clips.Length; ++n)
1140  {
1141  // Get the percentage of our animation
1142  // that is accounted for by this clip:
1143  pct = clips[n].GetFramesDisplayed() / totalFrames;
1144 
1145  // If the distance we have left to go into
1146  // our animation is less than the distance
1147  // accounted for by this clip, this is the
1148  // clip we're looking for!:
1149  if (remaining <= pct)
1150  {
1151  curClip = n;
1152  clips[curClip].SetPosition(remaining / pct);
1153 
1154  framePos = ((int)pct * (totalFrames - 1));
1155 
1156  return;
1157  }
1158  else
1159  remaining -= pct;
1160  }
1161  }
1162  }
1163 
1164  // Calculates the duration of the animation:
1165  protected void CalcDuration()
1166  {
1167  // If this loops infinitely, set duration to -1:
1168  if (loopCycles < 0)
1169  {
1170  duration = -1f;
1171  return;
1172  }
1173 
1174  duration = 0;
1175 
1176  for (int n = 0; n < clips.Length; ++n)
1177  {
1178  duration += clips[n].GetDuration();
1179  }
1180 
1181  if (loopReverse)
1182  duration *= 2f;
1183 
1184  duration += (loopCycles * duration);
1185  }
1186 
1194  public float GetDuration()
1195  {
1196  return duration;
1197  }
1198 
1204  public int GetFrameCount()
1205  {
1206  int totalFrames = 0;
1207 
1208  // Get the total number of frames:
1209  for (int n = 0; n < clips.Length; ++n)
1210  {
1211  totalFrames += clips[n].GetFramesDisplayed();
1212  }
1213 
1214  return totalFrames;
1215  }
1216 
1224  public int GetCurPosition()
1225  {
1226  return framePos;
1227  }
1228 
1233  public int GetCurClipNum()
1234  {
1235  return curClip;
1236  }
1237 
1242  public void SetCurClipNum(int index)
1243  {
1244  curClip = index;
1245  }
1246 }
1247 
1248 
1249 
1256 [ExecuteInEditMode]
1257 public abstract class SpriteBase : SpriteRoot, ISpriteAnimatable
1258 {
1259  // Animation-related vars and types:
1260 
1267  public delegate void AnimCompleteDelegate(SpriteBase sprite); // Definition of delegate to be called upon animation completion
1268 
1277  public delegate void AnimFrameDelegate(SpriteBase sprite, int frame);
1278 
1284  public bool playAnimOnStart = false; // When set to true, will start playing the default animation on start
1285 
1286  // <summary>
1287  // When set to true, the sprite will crossfade between the current frame
1288  // and the next frame in the sprite animation. NOTE: requires the use of the
1289  // "Crossfade Sprite" shader, which replaces sprite coloring functionality.
1290  // </summary>
1291  [HideInInspector]
1292  public bool crossfadeFrames = false;
1293 
1297  public int defaultAnim = 0; // Index of the default animation
1298 
1299 
1300  protected int curAnimIndex = 0; // Index of the "current" animation, if any
1301 
1302  protected AnimCompleteDelegate animCompleteDelegate = null; // Delegate to be called upon animation completion
1303  protected AnimFrameDelegate animFrameDelegate = null; // Delegate to be called each frame
1304  protected float timeSinceLastFrame = 0; // The total time since our last animation frame change
1305  protected float timeBetweenAnimFrames; // The amount of time we want to pass before moving to the next frame of animation
1306  protected float framesToAdvance; // (working) The number of animation frames to advance given the time elapsed
1307  protected bool animating = false; // True when an animation is intended to be playing (remains true when object is deactivated while playing an animation)
1308  protected bool currentlyAnimating = false; // True when an animation is actually playing (is false when the object is inactive)
1309 
1310  protected SPRITE_FRAME nextFrameInfo = new SPRITE_FRAME(0);
1311 
1312 
1313  protected override void Awake()
1314  {
1315  base.Awake();
1316 
1317  // Start the shared animation coroutine if it is not running already:
1318 // if (!SpriteAnimationPump.pumpIsRunning && Application.isPlaying)
1319 // SpriteAnimationPump.Instance.StartAnimationPump();
1320  }
1321 
1322  public override void Start()
1323  {
1324  base.Start();
1325 
1326  if(m_spriteMesh != null)
1327  m_spriteMesh.UseUV2 = crossfadeFrames;
1328  }
1329 
1330 
1334  public override void Clear()
1335  {
1336  base.Clear();
1337 
1338  //animations.Clear();
1339  animCompleteDelegate = null;
1340  }
1341 
1342  public override void Delete()
1343  {
1344  if (currentlyAnimating)
1345  {
1346  // Remove ourselves from the animating list.
1347  RemoveFromAnimatedList();
1348  }
1349 
1350  base.Delete();
1351  }
1352 
1353  // Called when the GO is disabled or destroyed
1354  protected override void OnDisable()
1355  {
1356  base.OnDisable();
1357 
1358  if(animating)
1359  {
1360  // Remove ourselves from the animating list.
1361  RemoveFromAnimatedList();
1362 
1363  // Leave "animating" set to true so that when
1364  // we re-enable, we can pick up animating again:
1365  animating = true;
1366  }
1367  }
1368 
1369  // Called when the GO is enabled or created:
1370  protected override void OnEnable()
1371  {
1372  base.OnEnable();
1373 
1374  // If this is being called in edit mode,
1375  // disregard:
1376  if (!Application.isPlaying)
1377  return;
1378 
1379  // If we were previously animating,
1380  // resume animation:
1381  if (animating)
1382  {
1383  // Set to false so AddToAnimatingList()
1384  // won't bail out:
1385  animating = false;
1386  AddToAnimatedList();
1387  }
1388  }
1389 
1390  public override void Copy(SpriteRoot s)
1391  {
1392  base.Copy(s);
1393 
1394  if (!(s is SpriteBase))
1395  return;
1396 
1397  SpriteBase sb = (SpriteBase)s;
1398 
1399  defaultAnim = sb.defaultAnim;
1400  playAnimOnStart = sb.playAnimOnStart;
1401  }
1402 
1403  public override void Hide(bool tf)
1404  {
1405  base.Hide(tf);
1406 
1407  // If we're hideAtStart, pause animation:
1408  if (tf)
1409  PauseAnim();
1410  }
1411 
1412 
1413 /*
1421  public virtual void Setup(float width, float height, Vector2 lowerleftPixel, Vector2 pixeldimensions)
1422  {
1423  SetSize(width, height);
1424  lowerLeftPixel = lowerleftPixel;
1425  pixelDimensions = pixeldimensions;
1426 
1427  tempUV = PixelCoordToUVCoord(lowerLeftPixel);
1428  uvRect.x = tempUV.x;
1429  uvRect.y = tempUV.y;
1430 
1431  tempUV = PixelSpaceToUVSpace(pixelDimensions);
1432  uvRect.xMax = uvRect.x + tempUV.x;
1433  uvRect.yMax = uvRect.y + tempUV.y;
1434 
1435  SetBleedCompensation(bleedCompensation);
1436  }
1437 */
1438 
1439 
1440  //-----------------------------------------------------------------
1441  // Animation-related routines:
1442  //-----------------------------------------------------------------
1443 
1448  public void SetAnimCompleteDelegate(AnimCompleteDelegate del)
1449  {
1450  animCompleteDelegate = del;
1451  }
1452 
1457  public void SetAnimFrameDelegate(AnimFrameDelegate del)
1458  {
1459  animFrameDelegate = del;
1460  }
1461 
1467  {
1468  resizedDelegate = del;
1469  }
1470 
1476  {
1477  resizedDelegate += del;
1478  }
1479 
1485  {
1486  resizedDelegate -= del;
1487  }
1488 
1489  public virtual bool StepAnim(float time) { return false; }
1490 
1491  public virtual void PlayAnim(int index) { }
1492  public virtual void PlayAnim(string name) { }
1493  public virtual void PlayAnimInReverse(int index) { }
1494  public virtual void PlayAnimInReverse(string name) { }
1495 /* public virtual void PlayAnim(int index, int frame) {}
1496  public virtual void PlayAnim(string name, int frame) {}
1497 */
1498 
1506  public void SetFramerate(float fps)
1507  {
1508  timeBetweenAnimFrames = 1f / fps;
1509  }
1510 
1511 
1515  public void PauseAnim()
1516  {
1517  if(animating)
1518  RemoveFromAnimatedList();
1519  // Stop coroutine
1520  //animating = false;
1521  //StopCoroutine("AnimationPump");
1522  //StopAllCoroutines();
1523  }
1524 
1525 
1532  public virtual void StopAnim(){}
1533 
1534 
1538  public void RevertToStatic()
1539  {
1540  if (animating)
1541  StopAnim();
1542 
1543  InitUVs();
1544  SetBleedCompensation();
1545 
1546  if (autoResize || pixelPerfect)
1547  CalcSize();
1548  }
1549 
1550  // Adds the sprite to the list of currently
1551  // animating sprites:
1552  protected abstract void AddToAnimatedList();
1553 
1554  // Removes the sprite from the list of currently
1555  // animating sprites:
1556  protected abstract void RemoveFromAnimatedList();
1557 
1558 
1559  //--------------------------------------------------------------
1560  // Accessors:
1561  //--------------------------------------------------------------
1566  public bool IsAnimating() { return animating; }
1567 
1574  public bool Animating
1575  {
1576  get { return animating; }
1577 
1578  set
1579  {
1580  if(value)
1581  PlayAnim(curAnimIndex);
1582  }
1583  }
1584 
1591  public int CurAnimIndex
1592  {
1593  get { return curAnimIndex; }
1594  set { curAnimIndex = value; }
1595  }
1596 }
1597 
1598 
1599 
SPRITE_FRAME GetCurrentFrame()
Returns the current frame info without advancing to the next frame.
Definition: SpriteBase.cs:251
int GetCurPosition()
Returns the (zero-based) index of the current frame.
Definition: SpriteBase.cs:613
void RevertToStatic()
Reverts the sprite to its static (non-animating) default appearance.
Definition: SpriteBase.cs:1538
virtual void StopAnim()
Stops the current animation from playing and resets it to the beginning for playing again...
Definition: SpriteBase.cs:1532
bool pixelPerfect
Automatically sizes the sprite so that it will display pixel-perfect on-screen. NOTE: If you change t...
Definition: SpriteRoot.cs:762
SPRITE_FRAME[] BuildUVAnim(Vector2 start, Vector2 cellSize, int cols, int rows, int totalCells)
Constructs an array of frames based upon the info supplied. NOTE: When the edge of the texture is rea...
Definition: SpriteBase.cs:281
string name
The name of the animation sequence
Definition: SpriteBase.cs:702
float framerate
The rate in frames per second at which to play the animation
Definition: SpriteBase.cs:93
SPRITE_FRAME GetCurrentFrame()
Returns the SPRITE_FRAME of the current frame without advancing to the next frame.
Definition: SpriteBase.cs:931
void SetFramerate(float fps)
Changes the framerate at which the current animation plays. NOTE: This only has effect if called AFTE...
Definition: SpriteBase.cs:1506
int index
The index of this animation in the sprite's animation list. This value is only populated by and used ...
Definition: SpriteBase.cs:103
void SetPosition(float pos)
Sets the current playing position of the animation. NOTE: This method takes loop cycles and loop reve...
Definition: SpriteBase.cs:1019
delegate void SpriteResizedDelegate(float newWidth, float newHeight, SpriteRoot sprite)
bool autoResize
Automatically resizes the sprite based on its new UV dimensions compared to its previous dimensions...
Definition: SpriteRoot.cs:771
void SetAnim(SPRITE_FRAME[] anim)
Assigns the specified array of frames to the animation, replacing its current contents.
Definition: SpriteBase.cs:382
void SetCurrentFrame(int f)
Sets the current frame of animation.
Definition: SpriteBase.cs:447
Hide the sprite when the animation ends.
int loopCycles
How many times to loop the animation IN ADDITION to the initial play-through. -1 indicates to loop in...
Definition: SpriteBase.cs:78
override void Copy(SpriteRoot s)
Copies all the vital attributes of another sprite.
Definition: SpriteBase.cs:1390
int rows
The number of rows in the animation.
Definition: SpriteBase.cs:650
void SetAnim(TextureAnim anim, int idx)
Assigns all the various animation info to this animation from the specified TextureAnim, replacing any existing animation data.
Definition: SpriteBase.cs:395
int GetCurPosition()
Returns the (zero-based) frame number of the current position in the over all animation. Example: If the multi contains a total of 100 frames and 25 frames have played so far, then 24 will be returned (because it is zero-based). If the multi is playing backwards, this number will count down from 100 as well.
Definition: SpriteBase.cs:1224
UVAnimation_Auto[] BuildUVAnim(SpriteRoot s)
Builds the UV animations for all animation clips that are a part of this animation sequence...
Definition: SpriteBase.cs:806
SPRITE_FRAME[] BuildWrappedUVAnim(Vector2 start, Vector2 cellSize, int totalCells)
Constructs an array of frames based upon the info supplied. NOTE: When the edge of the texture is rea...
Definition: SpriteBase.cs:341
int loopCycles
How many times to loop the animation IN ADDITION TO the initial play-through. (-1 to loop infinitely...
Definition: SpriteBase.cs:707
void SetClipPosition(float pos)
Sets the current frame based on a 0-1 value indicating the desired position in the animation...
Definition: SpriteBase.cs:533
void CalcSize()
Recalculates the width and height of the sprite based upon the change in its UV dimensions (autoResiz...
Definition: SpriteRoot.cs:1350
int defaultAnim
Index of the animation to play by default.
Definition: SpriteBase.cs:1297
int CurAnimIndex
Property useful for use with EZ Game Saver. Include this as a saved property, along with Animating...
Definition: SpriteBase.cs:1592
void SetSpriteResizedDelegate(SpriteResizedDelegate del)
Sets the delegate to be called when the sprite is resized.
Definition: SpriteBase.cs:1466
int StepDirection
The direction in which the animation is currently set to advance. 1 for forwards, -1 for backwards...
Definition: SpriteBase.cs:795
ANIM_END_ACTION onAnimEnd
What the sprite should do when the animation is done playing. The options are to: 1) Do nothing...
Definition: SpriteBase.cs:111
bool playAnimOnStart
When set to true, the sprite will play the default animation (see defaultAnim) when the sprite is ins...
Definition: SpriteBase.cs:1284
Play the default animation when the animation ends.
Vector2 PixelSpaceToUVSpace(Vector2 xy)
Converts pixel-space values to UV-space scalar values according to the currently assigned material...
Definition: SpriteRoot.cs:2425
int GetCurClipNum()
Returns the (zero-based) index of the current clip.
Definition: SpriteBase.cs:1233
void PauseAnim()
Pauses the currently-playing animation.
Definition: SpriteBase.cs:1515
void SetCurClipNum(int index)
Sets the current clip by index.
Definition: SpriteBase.cs:1242
UVAnimation_Auto[] clips
The actual sprite animation clips that make up the animation sequence.
Definition: SpriteBase.cs:729
bool IsAnimating()
Returns whether the sprite is currently animating.
Definition: SpriteBase.cs:1566
void SetPosition(float pos)
Sets the current frame based on a 0-1 value indicating the desired position in the animation...
Definition: SpriteBase.cs:468
Vector2 pixelsToNextColumnAndRow
The number of pixels from the left edge of one sprite frame to the left edge of the next one...
Definition: SpriteBase.cs:640
SPRITE_FRAME[] BuildUVAnim(SpriteRoot s)
Uses the information stored in this class to build a UV animation for the specified sprite...
Definition: SpriteBase.cs:681
SPRITE_FRAME GetFrame(int frame)
Returns the specified frame.
Definition: SpriteBase.cs:262
float GetLength()
Returns the length, in seconds, of the animation. NOTE: This does not take into account looping or re...
Definition: SpriteBase.cs:553
int totalCells
The total number of frames (cells) of animation.
Definition: SpriteBase.cs:655
void AppendAnim(int index, SPRITE_FRAME[] anim)
Appends UV animation to the clip specified by index.
Definition: SpriteBase.cs:942
void AppendAnim(SPRITE_FRAME[] anim)
Appends the specified array of frames to the existing animation.
Definition: SpriteBase.cs:432
bool loopReverse
Reverse the play direction when the end of the animation is reached? (Ping-pong) If true...
Definition: SpriteBase.cs:86
void AppendClip(UVAnimation clip)
Appends UV animation clip to the end of the animation sequence.
Definition: SpriteBase.cs:956
void SetAnimFrameDelegate(AnimFrameDelegate del)
Sets the delegate to be called each frame of animation.
Definition: SpriteBase.cs:1457
string name
The name of the animation.
Definition: SpriteBase.cs:69
override void Hide(bool tf)
Hides or displays the sprite by disabling/enabling the sprite's mesh renderer component, or if managed, sets the mesh size to 0.
Definition: SpriteBase.cs:1403
override void Delete()
If non-managed, call Delete() before destroying this component or the GameObject to which it is attac...
Definition: SpriteBase.cs:1342
int StepDirection
The direction in which the animation is currently set to advance. 1 for forwards, -1 for backwards...
Definition: SpriteBase.cs:118
float framerate
The rate in frames per second at which to play the animation.
Definition: SpriteBase.cs:717
bool Animating
Property useful for use with EZ Game Saver. When it gets set, the current animation is started...
Definition: SpriteBase.cs:1575
The root class of all sprites. Does not assume any animation capabilities or atlas packing...
Definition: SpriteRoot.cs:628
int GetFramesDisplayed()
Returns the number of frames displayed by the animation according to its current loop cycles and loop...
Definition: SpriteBase.cs:596
int GetFrameCount()
Returns the number of frames in the animation.
Definition: SpriteBase.cs:585
Deactivate the sprite when the animation ends (sets the GameObject's .active property to false)...
int cols
The number of columns in the animation.
Definition: SpriteBase.cs:645
SPRITE_FRAME[] BuildWrappedUVAnim(Vector2 start, Vector2 cellSize, int cols, int rows, int totalCells)
Constructs an array of frames based upon the info supplied. NOTE: When the edge of the texture is rea...
Definition: SpriteBase.cs:325
UVAnimation_Auto GetCurrentClip()
Gets a reference to the currently-playing clip.
Definition: SpriteBase.cs:785
float GetDuration()
Returns the duration, in seconds, of the animation. NOTE: This takes into account loop cycles and loo...
Definition: SpriteBase.cs:1194
Revert to the static image when the animation ends.
delegate void AnimFrameDelegate(SpriteBase sprite, int frame)
void SetAnimPosition(float pos)
Sets the current playing position of the animation. NOTE: This method does NOT take loop cycles and l...
Definition: SpriteBase.cs:1054
void RemoveSpriteresizedDelegate(SpriteResizedDelegate del)
Removes the specified delegate from the list of those to be called when the sprite is resized...
Definition: SpriteBase.cs:1484
void Reset()
Resets the animation sequence for playing anew.
Definition: SpriteBase.cs:999
void Reset()
Resets all the animation state vars to ready the object for playing anew.
Definition: SpriteBase.cs:159
bool loopReverse
Reverse the play direction when the end of the animation is reached? If true, a loop iteration isn't ...
Definition: SpriteBase.cs:712
delegate void AnimCompleteDelegate(SpriteBase sprite)
float GetDuration()
Returns the duration, in seconds, of the animation. NOTE: This takes into account loop cycles and loo...
Definition: SpriteBase.cs:567
void SetAnimCompleteDelegate(AnimCompleteDelegate del)
Sets the delegate to be called upon animation completion.
Definition: SpriteBase.cs:1448
void AddSpriteResizedDelegate(SpriteResizedDelegate del)
Adds the delegate to be called when the sprite is resized.
Definition: SpriteBase.cs:1475
Vector2 start
The pixel coordinates of the lower-left corner of the first frame in the animation sequence...
Definition: SpriteBase.cs:631
override void Clear()
Resets important sprite values to defaults for reuse.
Definition: SpriteBase.cs:1334
int GetFrameCount()
Returns the total number of frames displayed by this animation.
Definition: SpriteBase.cs:1204
void SetAnim(int index, SPRITE_FRAME[] frames)
Replaces the contents of the specified clip.
Definition: SpriteBase.cs:986
Destroys the sprite when the animation ends.
UVAnimation.ANIM_END_ACTION onAnimEnd
What the sprite should do when the animation is done playing. The options are to: 1) Do nothing...
Definition: SpriteBase.cs:724
Do nothing when the animation ends.