SpriteManager 2
 All Classes Functions Variables Enumerations Enumerator Properties
SpriteManager.cs
1 //-----------------------------------------------------------------
2 // Copyright 2010 Brady Wright and Above and Beyond Software
3 // All rights reserved
4 //-----------------------------------------------------------------
5 
6 
7 #define SPRITE_WANT_NORMALS
8 #define USE_GENERICS
9 
10 
11 using UnityEngine;
12 using System.Collections;
13 
14 #if USE_GENERICS
15 using System.Collections.Generic;
16 #endif
17 
18 
24 [RequireComponent(typeof(SkinnedMeshRenderer))]
25 [ExecuteInEditMode]
26 public class SpriteManager : MonoBehaviour
27 {
34  public SpriteRoot.WINDING_ORDER winding = SpriteRoot.WINDING_ORDER.CW;
35 
41  public int allocBlockSize = 10; // How many sprites to allocate space for at a time. ex: if set to 10, 10 new sprite blocks will be allocated at a time. Once all of these are used, 10 more will be allocated, and so on...
42 
52  public bool autoUpdateBounds = true; // Automatically recalculate the bounds of the mesh when vertices change?
53  //public bool ensureUnique = true; // Ensures that all sprite additions are unique by checking added sprites against the current list
54 
55  public bool drawBoundingBox = false;
56 
61  public bool persistent = false;
62 
63  protected bool initialized = false;
64 
65  protected EZLinkedList<SpriteMesh_Managed> availableBlocks = new EZLinkedList<SpriteMesh_Managed>(); // Array of references to sprites which are currently not in use
66  protected bool vertsChanged = false; // Have changes been made to the vertices of the mesh since the last frame?
67  protected bool uvsChanged = false; // Have changes been made to the UVs of the mesh since the last frame?
68  protected bool colorsChanged = false; // Have the colors changed?
69  protected bool vertCountChanged = false;// Has the number of vertices changed?
70  protected bool updateBounds = false; // Update the mesh bounds?
71  protected SpriteMesh_Managed[] sprites; // Array of all sprites (the offset of the vertices corresponding to each sprite should be found simply by taking the sprite's index * 4 (4 verts per sprite).
72  protected EZLinkedList<SpriteMesh_Managed> activeBlocks = new EZLinkedList<SpriteMesh_Managed>(); // Array of references to all the currently active (non-empty) sprites
73  //protected ArrayList activeBillboards = new ArrayList(); // Array of references to all the *active* sprites which are to be rendered as billboards
74 #if USE_GENERICS
75  protected List<SpriteMesh_Managed> spriteDrawOrder = new List<SpriteMesh_Managed>(); // Array of indices of sprite objects stored in the order they are to be drawn (corresponding to the position of their vertex indices in the triIndices list) Allows us to keep track of where a given Sprite is in the drawing order (triIndices)
76 #else
77  protected ArrayList spriteDrawOrder = new ArrayList();
78 #endif
79  protected SpriteDrawLayerComparer drawOrderComparer = new SpriteDrawLayerComparer(); // Used to sort our draw order array
80  protected float boundUpdateInterval; // Interval, in seconds, to update the mesh bounds
81 
82  protected List<SpriteRoot> spriteAddQueue; // List of sprites to be added that we have accumilated before initialization
83 
84  protected SkinnedMeshRenderer meshRenderer;
85  protected Mesh mesh; // Reference to our mesh (contained in the SkinnedMeshRenderer)
86  protected Texture texture;
87 
88  protected Transform[] bones; // The bones by which we'll transform our sprites' vertices
89  protected BoneWeight[] boneWeights; // Bone weights - Each element corresponds to a vertex in the vertices array, and indicates which bones, and at what weight, will influence the vertex.
90  protected Matrix4x4[] bindPoses; // Refers the bind pose of each bone
91  protected Vector3[] vertices; // The vertices of our mesh
92  protected int[] triIndices; // Indices into the vertex array
93  protected Vector2[] UVs; // UV coordinates
94  protected Vector2[] UVs2; // UV coordinates
95  protected Color[] colors; // Color values
96 
97  // Working vars:
98  protected SpriteMesh_Managed tempSprite = null;
99 
100  //--------------------------------------------------------------
101  // Utility functions:
102  //--------------------------------------------------------------
103 
113  public Vector2 PixelSpaceToUVSpace(Vector2 xy)
114  {
115  if (texture == null)
116  return Vector2.zero;
117 
118  return new Vector2(xy.x / ((float)texture.width), xy.y / ((float)texture.height));
119  }
120 
131  public Vector2 PixelSpaceToUVSpace(int x, int y)
132  {
133  return PixelSpaceToUVSpace(new Vector2((float)x, (float)y));
134  }
135 
145  public Vector2 PixelCoordToUVCoord(Vector2 xy)
146  {
147  if (texture == null)
148  return Vector2.zero;
149 
150  return new Vector2(xy.x / ((float)texture.width - 1), 1.0f - (xy.y / ((float)texture.height - 1)));
151  }
152 
163  public Vector2 PixelCoordToUVCoord(int x, int y)
164  {
165  return PixelCoordToUVCoord(new Vector2((float)x, (float)y));
166  }
167 
168  // Setups up bone weights for a sprite
169  protected void SetupBoneWeights(SpriteMesh_Managed s)
170  {
171  // Each bone element corresponds to each sprite element:
172  boneWeights[s.mv1].boneIndex0 = s.index;
173  boneWeights[s.mv1].weight0 = 1f;
174  boneWeights[s.mv2].boneIndex0 = s.index;
175  boneWeights[s.mv2].weight0 = 1f;
176  boneWeights[s.mv3].boneIndex0 = s.index;
177  boneWeights[s.mv3].weight0 = 1f;
178  boneWeights[s.mv4].boneIndex0 = s.index;
179  boneWeights[s.mv4].weight0 = 1f;
180  }
181 
182  //--------------------------------------------------------------
183  // End utility functions
184  //--------------------------------------------------------------
185 
186  void Awake()
187  {
188  if (spriteAddQueue == null)
189  spriteAddQueue = new List<SpriteRoot>();
190 
191  // Make sure the manager is centered:
192  //transform.position = Vector3.zero;
193 
194  meshRenderer = (SkinnedMeshRenderer)GetComponent(typeof(SkinnedMeshRenderer));
195 
196  if(meshRenderer != null)
197  if(meshRenderer.sharedMaterial != null)
198  texture = meshRenderer.sharedMaterial.GetTexture("_MainTex");
199 
200  if (meshRenderer.sharedMesh == null)
201  meshRenderer.sharedMesh = new Mesh();
202  mesh = meshRenderer.sharedMesh;
203 #if UNITY_4 || UNITY_4_1 || UNITY_4_2 || UNITY_4_3 || UNITY_4_4 || UNITY_4_5 || UNITY_4_6 || UNITY_4_7 || UNITY_4_8 || UNITY_4_9
204  mesh.MarkDynamic();
205 #endif
206 
207  if (persistent)
208  {
209  DontDestroyOnLoad(this);
210  DontDestroyOnLoad(mesh);
211  }
212 
213  // Create our first batch of sprites 'n' such:
214  EnlargeArrays(allocBlockSize);
215 
216  // Move the object to the origin so the objects drawn will not
217  // be offset from the objects they are intended to represent.
218  //transform.position = Vector3.zero;
219  transform.rotation = Quaternion.identity;
220 
221  initialized = true;
222 
223  // Now process any outstanding sprite additions:
224  for(int i=0; i<spriteAddQueue.Count; ++i)
225  AddSprite(spriteAddQueue[i]);
226  }
227 
228  // Allocates initial arrays
229  protected void InitArrays()
230  {
231  bones = new Transform[1];
232  bones[0] = transform; // Just give it something for now
233  bindPoses = new Matrix4x4[1];
234  sprites = new SpriteMesh_Managed[1];
235  sprites[0] = new SpriteMesh_Managed();
236  vertices = new Vector3[4];
237  UVs = new Vector2[4];
238  colors = new Color[4];
239  triIndices = new int[6];
240  boneWeights = new BoneWeight[4];
241 
242  sprites[0].index = 0;
243  sprites[0].mv1 = 0;
244  sprites[0].mv2 = 1;
245  sprites[0].mv3 = 2;
246  sprites[0].mv4 = 3;
247  SetupBoneWeights(sprites[0]);
248  }
249 
250  // Enlarges the sprite array by the specified count and also resizes
251  // the UV and vertex arrays by the necessary corresponding amount.
252  // Returns the index of the first newly allocated element
253  // (ex: if the sprite array was already 10 elements long and is
254  // enlarged by 10 elements resulting in a total length of 20,
255  // EnlargeArrays() will return 10, indicating that element 10 is the
256  // first of the newly allocated elements.)
257  protected int EnlargeArrays(int count)
258  {
259  int firstNewElement;
260 
261  if (sprites == null)
262  {
263  InitArrays();
264  firstNewElement = 0;
265  count = count - 1; // Allocate one less since InitArrays already allocated one sprite for us
266  }
267  else
268  firstNewElement = sprites.Length;
269 
270  // Resize sprite array:
271  SpriteMesh_Managed[] tempSprites = sprites;
272  sprites = new SpriteMesh_Managed[sprites.Length + count];
273  tempSprites.CopyTo(sprites, 0);
274 
275  // Resize the bone array:
276  Transform[] tempBones = bones;
277  bones = new Transform[bones.Length + count];
278  tempBones.CopyTo(bones, 0);
279 
280  // Resize the bind poses array:
281  Matrix4x4[] tempPoses = bindPoses;
282  bindPoses = new Matrix4x4[bindPoses.Length + count];
283  tempPoses.CopyTo(bindPoses, 0);
284 
285  // Vertices:
286  Vector3[] tempVerts = vertices;
287  vertices = new Vector3[vertices.Length + count * 4];
288  tempVerts.CopyTo(vertices, 0);
289 
290  // BoneWeights:
291  BoneWeight[] tempWeights = boneWeights;
292  boneWeights = new BoneWeight[boneWeights.Length + count * 4];
293  tempWeights.CopyTo(boneWeights, 0);
294 
295  // UVs:
296  Vector2[] tempUVs = UVs;
297  UVs = new Vector2[UVs.Length + count * 4];
298  tempUVs.CopyTo(UVs, 0);
299 
300  // Colors:
301  Color[] tempColors = colors;
302  colors = new Color[colors.Length + count * 4];
303  tempColors.CopyTo(colors, 0);
304 
305  // Triangle indices:
306  int[] tempTris = triIndices;
307  triIndices = new int[triIndices.Length + count * 6];
308  tempTris.CopyTo(triIndices, 0);
309 
310  // Inform existing sprites of the new vertex and UV buffers:
311  for (int i = 0; i < firstNewElement; ++i)
312  {
313  sprites[i].SetBuffers(vertices, UVs, UVs2, colors);
314  }
315 
316  // Setup the newly-added sprites and Add them to the list of available
317  // sprite blocks. Also initialize the triangle indices while we're at it:
318  for (int i = firstNewElement; i < sprites.Length; ++i)
319  {
320  // Create and setup sprite:
321 
322  sprites[i] = new SpriteMesh_Managed();
323  sprites[i].index = i;
324  sprites[i].manager = this;
325 
326  sprites[i].SetBuffers(vertices, UVs, UVs2, colors);
327 
328  // Setup indices of the sprite's vertices in the vertex buffer:
329  sprites[i].mv1 = i * 4 + 0;
330  sprites[i].mv2 = i * 4 + 1;
331  sprites[i].mv3 = i * 4 + 2;
332  sprites[i].mv4 = i * 4 + 3;
333 
334  // Setup the indices of the sprite's UV entries in the UV buffer:
335  sprites[i].uv1 = i * 4 + 0;
336  sprites[i].uv2 = i * 4 + 1;
337  sprites[i].uv3 = i * 4 + 2;
338  sprites[i].uv4 = i * 4 + 3;
339 
340  // Setup the indices to the color values:
341  sprites[i].cv1 = i * 4 + 0;
342  sprites[i].cv2 = i * 4 + 1;
343  sprites[i].cv3 = i * 4 + 2;
344  sprites[i].cv4 = i * 4 + 3;
345 
346  // Add as an available sprite:
347  availableBlocks.Add(sprites[i]);
348 
349  // Init triangle indices:
350 /* if (winding == SpriteRoot.WINDING_ORDER.CCW)
351  { // Counter-clockwise winding
352  triIndices[i * 6 + 0] = i * 4 + 0; // 0_ 2 0 ___ 3
353  triIndices[i * 6 + 1] = i * 4 + 1; // | / Verts: | /|
354  triIndices[i * 6 + 2] = i * 4 + 3; // 1|/ 1|/__|2
355 
356  triIndices[i * 6 + 3] = i * 4 + 3; // 3
357  triIndices[i * 6 + 4] = i * 4 + 1; // /|
358  triIndices[i * 6 + 5] = i * 4 + 2; // 4/_|5
359  }
360  else
361 */ { // Clockwise winding
362  triIndices[i * 6 + 0] = i * 4 + 0; // 0_ 1 0 ___ 3
363  triIndices[i * 6 + 1] = i * 4 + 3; // | / Verts: | /|
364  triIndices[i * 6 + 2] = i * 4 + 1; // 2|/ 1|/__|2
365 
366  triIndices[i * 6 + 3] = i * 4 + 3; // 3
367  triIndices[i * 6 + 4] = i * 4 + 2; // /|
368  triIndices[i * 6 + 5] = i * 4 + 1; // 5/_|4
369  }
370 
371  // Add the index of this sprite to the draw order list
372  spriteDrawOrder.Add(sprites[i]);
373 
374  // Give the bones something to point to:
375  bones[i] = transform;
376 
377  // Setup a default bindpose:
378  bindPoses[i] = bones[i].worldToLocalMatrix * transform.localToWorldMatrix;
379 
380  // Setup the weights:
381  SetupBoneWeights(sprites[i]);
382  }
383 
384  vertsChanged = true;
385  uvsChanged = true;
386  colorsChanged = true;
387  vertCountChanged = true;
388 
389  return firstNewElement;
390  }
391 
398  public bool AlreadyAdded(SpriteRoot sprite)
399  {
400  if(activeBlocks.Rewind())
401  {
402  do
403  {
404  if (activeBlocks.Current.sprite == sprite)
405  return true;
406  } while (activeBlocks.MoveNext());
407  }
408 
409  return false;
410  }
411 
418  public SpriteMesh_Managed AddSprite(GameObject go)
419  {
420  SpriteRoot sprite = (SpriteRoot)go.GetComponent(typeof(SpriteRoot));
421  if (sprite == null)
422  return null;
423 
424  return AddSprite(sprite);
425  }
426 
432  public SpriteMesh_Managed AddSprite(SpriteRoot sprite)
433  {
434  int spriteIndex;
435 
436  // See if the sprite is already added:
437  if(sprite.manager == this && sprite.AddedToManager)
438  return (SpriteMesh_Managed)sprite.spriteMesh;
439 /*
440  if (ensureUnique)
441  if (AlreadyAdded(sprite))
442  return (SpriteMesh_Managed) sprite.spriteMesh;
443 */
444 
445  // See if we're ready to add sprites yet,
446  // and if not, add the sprite to our deferred
447  // add queue:
448  if(!initialized)
449  {
450  if (spriteAddQueue == null)
451  spriteAddQueue = new List<SpriteRoot>();
452 
453  spriteAddQueue.Add(sprite);
454  return null;
455  }
456 
457  // Get an available sprite:
458  if (availableBlocks.Empty)
459  EnlargeArrays(allocBlockSize); // If we're out of available sprites, allocate some more:
460 
461  // Use a sprite from the list of available blocks:
462  spriteIndex = availableBlocks.Head.index;
463  availableBlocks.Remove(availableBlocks.Head); // Now that we're using this one, remove it from the available list
464 
465  // Assign the new sprite:
466  SpriteMesh_Managed newSpritesMesh = sprites[spriteIndex];
467  sprite.spriteMesh = newSpritesMesh;
468  sprite.manager = this;
469  sprite.AddedToManager = true;
470  newSpritesMesh.drawLayer = sprite.drawLayer;
471 
472  // Associate the sprite's bone:
473  bones[spriteIndex] = sprite.gameObject.transform;
474 // meshRenderer.bones = bones;
475  bindPoses[spriteIndex] = bones[spriteIndex].worldToLocalMatrix * sprite.transform.localToWorldMatrix;
476 
477 /*
478  // Save this to an active list now that it is in-use:
479  if (billboarded)
480  {
481  newSprite.billboarded = true;
482  activeBillboards.Add(newSprite);
483  }
484  else
485 */
486  activeBlocks.Add(newSpritesMesh);
487 
488  newSpritesMesh.Init();
489 
490  // Sort the draw layers:
491  SortDrawingOrder();
492 
493  // Set our flags:
494  vertCountChanged = true;
495  vertsChanged = true;
496  uvsChanged = true;
497 
498  return newSpritesMesh;
499  }
500 
501 
508  public SpriteRoot CreateSprite(GameObject prefab)
509  {
510  return CreateSprite(prefab, Vector3.zero, Quaternion.identity);
511  }
512 
513 
522  public SpriteRoot CreateSprite(GameObject prefab, Vector3 position, Quaternion rotation)
523  {
524  GameObject go = (GameObject) Instantiate(prefab, position, rotation);
525  SpriteRoot ret = (SpriteRoot) go.GetComponent(typeof(SpriteRoot));
526  AddSprite(go);
527 
528  // if (sm == null)
529  // return null;
530 
531  return ret;
532  }
533 
534 /*
535  public void SetBillboarded(Sprite sprite)
536  {
537  // Make sure the sprite isn't in the active list
538  // or else it'll get handled twice:
539  activeBlocks.Remove(sprite);
540  activeBillboards.Add(sprite);
541  }
542 */
547  public void RemoveSprite(SpriteRoot sprite)
548  {
549  if (sprite.spriteMesh is SpriteMesh_Managed && sprite.spriteMesh != null)
550  {
551  // Disown the sprite:
552  if (sprite.manager == this)
553  {
554  sprite.manager = null;
555  sprite.AddedToManager = false;
556  }
557 
558  RemoveSprite((SpriteMesh_Managed)sprite.spriteMesh);
559  }
560  }
561 
562 
563  // Removes the sprite with the specified managed mesh from the manager.
564  public void RemoveSprite(SpriteMesh_Managed sprite)
565  {
566  vertices[sprite.mv1] = Vector3.zero;
567  vertices[sprite.mv2] = Vector3.zero;
568  vertices[sprite.mv3] = Vector3.zero;
569  vertices[sprite.mv4] = Vector3.zero;
570 
571  // Remove the sprite from the billboarded list
572  // since that list should only contain active
573  // sprites:
574 /*
575  if (sprite.billboarded)
576  activeBillboards.Remove(sprite);
577  else
578 */
579  activeBlocks.Remove(sprite);
580 
581  // Reset the bone:
582  if (gameObject != null)
583  bones[sprite.index] = transform;
584 
585  // Clean the sprite's settings:
586  sprite.Clear();
587  sprite.sprite.spriteMesh = null;
588  sprite.sprite = null;
589 
590  availableBlocks.Add(sprite);
591 
592  vertsChanged = true;
593  }
594 
595 
601  public void MoveToFront(SpriteMesh_Managed s)
602  {
603  int[] indices = new int[6];
604  int offset = spriteDrawOrder.IndexOf(s) * 6;
605 
606  if (offset < 0)
607  return;
608 
609  s.drawLayer = spriteDrawOrder[spriteDrawOrder.Count - 1].drawLayer + 1;
610 
611  // Save our indices:
612  indices[0] = triIndices[offset];
613  indices[1] = triIndices[offset + 1];
614  indices[2] = triIndices[offset + 2];
615  indices[3] = triIndices[offset + 3];
616  indices[4] = triIndices[offset + 4];
617  indices[5] = triIndices[offset + 5];
618 
619  // Shift all indices from here forward down 6 slots (each sprite occupies 6 index slots):
620  for (int i = offset; i < triIndices.Length - 6; i += 6)
621  {
622  triIndices[i] = triIndices[i + 6];
623  triIndices[i + 1] = triIndices[i + 7];
624  triIndices[i + 2] = triIndices[i + 8];
625  triIndices[i + 3] = triIndices[i + 9];
626  triIndices[i + 4] = triIndices[i + 10];
627  triIndices[i + 5] = triIndices[i + 11];
628 
629  spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 + 1];
630  }
631 
632  // Place our desired index value at the end:
633  triIndices[triIndices.Length - 6] = indices[0];
634  triIndices[triIndices.Length - 5] = indices[1];
635  triIndices[triIndices.Length - 4] = indices[2];
636  triIndices[triIndices.Length - 3] = indices[3];
637  triIndices[triIndices.Length - 2] = indices[4];
638  triIndices[triIndices.Length - 1] = indices[5];
639 
640  // Update the sprite's index offset:
641  spriteDrawOrder[spriteDrawOrder.Count - 1] = s;
642 
643  vertCountChanged = true;
644  }
645 
651  public void MoveToBack(SpriteMesh_Managed s)
652  {
653  int[] indices = new int[6];
654  int offset = spriteDrawOrder.IndexOf(s) * 6;
655 
656  if (offset < 0)
657  return;
658 
659  s.drawLayer = spriteDrawOrder[0].drawLayer - 1;
660 
661  // Save our indices:
662  indices[0] = triIndices[offset];
663  indices[1] = triIndices[offset + 1];
664  indices[2] = triIndices[offset + 2];
665  indices[3] = triIndices[offset + 3];
666  indices[4] = triIndices[offset + 4];
667  indices[5] = triIndices[offset + 5];
668 
669  // Shift all indices from here back up 6 slots (each sprite occupies 6 index slots):
670  for (int i = offset; i > 5; i -= 6)
671  {
672  triIndices[i] = triIndices[i - 6];
673  triIndices[i + 1] = triIndices[i - 5];
674  triIndices[i + 2] = triIndices[i - 4];
675  triIndices[i + 3] = triIndices[i - 3];
676  triIndices[i + 4] = triIndices[i - 2];
677  triIndices[i + 5] = triIndices[i - 1];
678 
679  spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 - 1];
680  }
681 
682  // Place our desired index value at the beginning:
683  triIndices[0] = indices[0];
684  triIndices[1] = indices[1];
685  triIndices[2] = indices[2];
686  triIndices[3] = indices[3];
687  triIndices[4] = indices[4];
688  triIndices[5] = indices[5];
689 
690  // Update the sprite's index offset:
691  spriteDrawOrder[0] = s;
692 
693  vertCountChanged = true;
694  }
695 
703  public void MoveInfrontOf(SpriteMesh_Managed toMove, SpriteMesh_Managed reference)
704  {
705  int[] indices = new int[6];
706  int offset = spriteDrawOrder.IndexOf(toMove) * 6;
707  int refOffset = spriteDrawOrder.IndexOf(reference) * 6;
708 
709  if (offset < 0)
710  return;
711 
712  // Check to see if the sprite is already in front:
713  if (offset > refOffset)
714  return;
715 
716  toMove.drawLayer = reference.drawLayer + 1;
717 
718  // Save our indices:
719  indices[0] = triIndices[offset];
720  indices[1] = triIndices[offset + 1];
721  indices[2] = triIndices[offset + 2];
722  indices[3] = triIndices[offset + 3];
723  indices[4] = triIndices[offset + 4];
724  indices[5] = triIndices[offset + 5];
725 
726  // Shift all indices from here to the reference sprite down 6 slots (each sprite occupies 6 index slots):
727  for (int i = offset; i < refOffset; i += 6)
728  {
729  triIndices[i] = triIndices[i + 6];
730  triIndices[i + 1] = triIndices[i + 7];
731  triIndices[i + 2] = triIndices[i + 8];
732  triIndices[i + 3] = triIndices[i + 9];
733  triIndices[i + 4] = triIndices[i + 10];
734  triIndices[i + 5] = triIndices[i + 11];
735 
736  spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 + 1];
737  }
738 
739  // Place our desired index value at the destination:
740  triIndices[refOffset] = indices[0];
741  triIndices[refOffset + 1] = indices[1];
742  triIndices[refOffset + 2] = indices[2];
743  triIndices[refOffset + 3] = indices[3];
744  triIndices[refOffset + 4] = indices[4];
745  triIndices[refOffset + 5] = indices[5];
746 
747  // Update the sprite's index offset:
748  spriteDrawOrder[refOffset / 6] = toMove;
749 
750  vertCountChanged = true;
751  }
752 
760  public void MoveBehind(SpriteMesh_Managed toMove, SpriteMesh_Managed reference)
761  {
762  int[] indices = new int[6];
763  int offset = spriteDrawOrder.IndexOf(toMove) * 6;
764  int refOffset = spriteDrawOrder.IndexOf(reference) * 6;
765 
766  if (offset < 0)
767  return;
768 
769  // Check to see if the sprite is already behind:
770  if (offset < refOffset)
771  return;
772 
773  toMove.drawLayer = reference.drawLayer - 1;
774 
775  // Save our indices:
776  indices[0] = triIndices[offset];
777  indices[1] = triIndices[offset + 1];
778  indices[2] = triIndices[offset + 2];
779  indices[3] = triIndices[offset + 3];
780  indices[4] = triIndices[offset + 4];
781  indices[5] = triIndices[offset + 5];
782 
783  // Shift all indices from here to the reference sprite up 6 slots (each sprite occupies 6 index slots):
784  for (int i = offset; i > refOffset; i -= 6)
785  {
786  triIndices[i] = triIndices[i - 6];
787  triIndices[i + 1] = triIndices[i - 5];
788  triIndices[i + 2] = triIndices[i - 4];
789  triIndices[i + 3] = triIndices[i - 3];
790  triIndices[i + 4] = triIndices[i - 2];
791  triIndices[i + 5] = triIndices[i - 1];
792 
793  spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 - 1];
794  }
795 
796  // Place our desired index value at the destination:
797  triIndices[refOffset] = indices[0];
798  triIndices[refOffset + 1] = indices[1];
799  triIndices[refOffset + 2] = indices[2];
800  triIndices[refOffset + 3] = indices[3];
801  triIndices[refOffset + 4] = indices[4];
802  triIndices[refOffset + 5] = indices[5];
803 
804  // Update the sprite's index offset:
805  spriteDrawOrder[refOffset / 6] = toMove;
806 
807  vertCountChanged = true;
808  }
809 
810 
811  // Rebuilds the drawing order based upon the drawing order buffer
812  public void SortDrawingOrder()
813  {
814  SpriteMesh_Managed s;
815 
816  spriteDrawOrder.Sort(drawOrderComparer);
817 
818  // Now reconstitute the triIndices in the order we want:
819  if (winding == SpriteRoot.WINDING_ORDER.CCW)
820  {
821  for (int i = 0; i < spriteDrawOrder.Count; ++i)
822  {
823  s = (SpriteMesh_Managed)spriteDrawOrder[i];
824 
825  // Counter-clockwise winding
826  triIndices[i * 6 + 0] = s.mv1; // 0_ 2 1 ___ 4
827  triIndices[i * 6 + 1] = s.mv2; // | / Verts: | /|
828  triIndices[i * 6 + 2] = s.mv4; // 1|/ 2|/__|3
829 
830  triIndices[i * 6 + 3] = s.mv4; // 3
831  triIndices[i * 6 + 4] = s.mv2; // /|
832  triIndices[i * 6 + 5] = s.mv3; // 4/_|5
833  }
834  }
835  else
836  {
837  for (int i = 0; i < spriteDrawOrder.Count; ++i)
838  {
839  s = (SpriteMesh_Managed)spriteDrawOrder[i];
840 
841  // Clockwise winding
842  triIndices[i * 6 + 0] = s.mv1; // 0_ 1 1 ___ 4
843  triIndices[i * 6 + 1] = s.mv4; // | / Verts: | /|
844  triIndices[i * 6 + 2] = s.mv2; // 2|/ 2|/__|3
845 
846  triIndices[i * 6 + 3] = s.mv4; // 3
847  triIndices[i * 6 + 4] = s.mv3; // /|
848  triIndices[i * 6 + 5] = s.mv2; // 5/_|4
849  }
850  }
851 
852  vertCountChanged = true;
853  }
854 
855  // Returns the mesh of the sprite at the specified index.
856  public SpriteMesh_Managed GetSprite(int i)
857  {
858  if (i < sprites.Length)
859  return sprites[i];
860  else
861  return null;
862  }
863 
864  // Updates the vertices of a sprite such that it is oriented
865  // more or less toward the camera
866 /*
867  public void TransformBillboarded(Sprite sprite)
868  {
869  Vector3 pos = sprite.clientTransform.position;
870  Transform t = Camera.main.transform;
871 
872  vertices[sprite.mv1] = pos + t.TransformDirection(sprite.v1);
873  vertices[sprite.mv2] = pos + t.TransformDirection(sprite.v2);
874  vertices[sprite.mv3] = pos + t.TransformDirection(sprite.v3);
875  vertices[sprite.mv4] = pos + t.TransformDirection(sprite.v4);
876 
877  vertsChanged = true;
878  }
879 */
880 
881  // Informs the SpriteManager that some vertices have changed position
882  // and the mesh needs to be reconstructed accordingly
883  public void UpdatePositions()
884  {
885  vertsChanged = true;
886  }
887 
888  // Informs the SpriteManager that some UVs have changed
889  public void UpdateUVs()
890  {
891  uvsChanged = true;
892  }
893 
894  // Informs the SpriteManager that some colors have changed
895  public void UpdateColors()
896  {
897  colorsChanged = true;
898  }
899 
903  public void UpdateBounds()
904  {
905  updateBounds = true;
906  }
907 
913  public void ScheduleBoundsUpdate(float seconds)
914  {
915  boundUpdateInterval = seconds;
916  InvokeRepeating("UpdateBounds", seconds, seconds);
917  }
918 
922  public void CancelBoundsUpdate()
923  {
924  CancelInvoke("UpdateBounds");
925  }
926 
927  // Returns whether the SpriteManager is done initializing.
928  public bool IsInitialized
929  {
930  get { return initialized; }
931  }
932 
933  // LateUpdate is called once per frame
934  virtual public void LateUpdate()
935  {
936  // Were changes made to the mesh since last time?
937  if (vertCountChanged)
938  {
939  vertCountChanged = false;
940  colorsChanged = false;
941  vertsChanged = false;
942  uvsChanged = false;
943  updateBounds = false;
944 
945  meshRenderer.bones = bones;
946 
947  mesh.Clear();
948  mesh.vertices = vertices;
949  mesh.bindposes = bindPoses;
950  mesh.boneWeights = boneWeights;
951  mesh.uv = UVs;
952  mesh.colors = colors;
953  mesh.triangles = triIndices;
954 
955 #if SPRITE_WANT_NORMALS
956  mesh.RecalculateNormals();
957 #endif
958  if (autoUpdateBounds)
959  mesh.RecalculateBounds();
960  }
961  else
962  {
963  if (vertsChanged)
964  {
965  vertsChanged = false;
966 
967  if (autoUpdateBounds)
968  updateBounds = true;
969 
970  mesh.vertices = vertices;
971  }
972 
973  if (updateBounds)
974  {
975  mesh.RecalculateBounds();
976  updateBounds = false;
977  }
978 
979  if (colorsChanged)
980  {
981  colorsChanged = false;
982 
983  mesh.colors = colors;
984  }
985 
986  if (uvsChanged)
987  {
988  uvsChanged = false;
989  mesh.uv = UVs;
990  }
991  }
992  }
993 
994 
995 
996  // Empties the manager of all sprites.
997  // This is used internally in edit mode
998  // to keep the manager up-to-date despite
999  // Unity's mucking with us.
1000 /*
1001  protected void Clear()
1002  {
1003  SpriteRoot spr;
1004 
1005  if(activeBlocks.Rewind())
1006  {
1007  do
1008  {
1009  spr = activeBlocks.Current.sprite;
1010  RemoveSprite(activeBlocks.Current);
1011  spr.manager = this; // Don't forget us!
1012  }while(activeBlocks.MoveNext());
1013  }
1014  }
1015 
1016  // Scans the scene for sprites managed
1017  // by this manager.
1018  // This is used internally in edit mode
1019  // to keep the manager up-to-date despite
1020  // Unity's mucking with us.
1021  protected void ScanScene()
1022  {
1023  Object[] c = FindObjectsOfType(typeof(SpriteRoot));
1024  SpriteRoot spr;
1025 
1026  if(c == null)
1027  return;
1028 
1029  for(int i=0; i<c.Length; ++i)
1030  {
1031  spr = (SpriteRoot) c[i];
1032 
1033  if (spr.managed)
1034  if (spr.manager == this)
1035  AddSprite(spr);
1036  }
1037  }
1038 */
1039 
1040  // Ensures that the sprites are updated in the scene view
1041  // while editing:
1042  public virtual void DoMirror()
1043  {
1044  // Only run if we're not playing:
1045  if (Application.isPlaying)
1046  return;
1047 
1048  // Were changes made to the mesh since last time?
1049  if (vertCountChanged)
1050  {
1051  vertCountChanged = false;
1052  colorsChanged = false;
1053  vertsChanged = false;
1054  uvsChanged = false;
1055  updateBounds = false;
1056 
1057  meshRenderer.bones = bones;
1058 
1059  mesh.Clear();
1060  mesh.vertices = vertices;
1061  mesh.bindposes = bindPoses;
1062  mesh.boneWeights = boneWeights;
1063  mesh.uv = UVs;
1064  mesh.colors = colors;
1065  mesh.triangles = triIndices;
1066  }
1067  else
1068  {
1069  if (vertsChanged)
1070  {
1071  vertsChanged = false;
1072 
1073  updateBounds = true;
1074 
1075  mesh.vertices = vertices;
1076  }
1077 
1078  if(updateBounds)
1079  {
1080  mesh.RecalculateBounds();
1081  updateBounds = false;
1082  }
1083 
1084  if (colorsChanged)
1085  {
1086  colorsChanged = false;
1087 
1088  mesh.colors = colors;
1089  }
1090 
1091  if (uvsChanged)
1092  {
1093  uvsChanged = false;
1094  mesh.uv = UVs;
1095  }
1096  }
1097  }
1098 
1099 #if (UNITY_3_0 || UNITY_3_1 || UNITY_3_2 || UNITY_3_3 || UNITY_3_4 || UNITY_3_5 || UNITY_3_6 || UNITY_3_7 || UNITY_3_8 || UNITY_3_9 || UNITY_4_0 || UNITY_4_1 || UNITY_4_2 || UNITY_4_3 || UNITY_4_4 || UNITY_4_5 || UNITY_4_6 || UNITY_4_7 || UNITY_4_8 || UNITY_4_9) && UNITY_EDITOR
1100  void Update()
1101  {
1102  DoMirror();
1103  }
1104 #endif
1105  // Ensures that the text is updated in the scene view
1106  // while editing:
1107  public virtual void OnDrawGizmos()
1108  {
1109 #if !(UNITY_3_0 || UNITY_3_1 || UNITY_3_2 || UNITY_3_3 || UNITY_3_4 || UNITY_3_5 || UNITY_3_6 || UNITY_3_7 || UNITY_3_8 || UNITY_3_9 || UNITY_4_0 || UNITY_4_1 || UNITY_4_2 || UNITY_4_3 || UNITY_4_4 || UNITY_4_5 || UNITY_4_6 || UNITY_4_7 || UNITY_4_8 || UNITY_4_9) && UNITY_EDITOR
1110  DoMirror();
1111 #endif
1112  if (drawBoundingBox)
1113  {
1114  Gizmos.color = Color.yellow;
1115  DrawCenter();
1116  // Draw bounding box:
1117  Gizmos.DrawWireCube(meshRenderer.bounds.center, meshRenderer.bounds.size);
1118  }
1119  }
1120 
1121 
1122  public void OnDrawGizmosSelected()
1123  {
1124  Gizmos.color = Color.yellow;
1125  DrawCenter();
1126  // Draw bounding box:
1127  Gizmos.DrawWireCube(meshRenderer.bounds.center, meshRenderer.bounds.size);
1128  }
1129 
1130  protected void DrawCenter()
1131  {
1132  float longestSide;
1133  float starSize;
1134 
1135  longestSide = Mathf.Max(meshRenderer.bounds.size.x, meshRenderer.bounds.size.y);
1136  longestSide = Mathf.Max(longestSide, meshRenderer.bounds.size.z);
1137 
1138  starSize = longestSide * 0.015f;
1139 
1140  // Draw the center:
1141  Gizmos.DrawLine(meshRenderer.bounds.center - Vector3.up * starSize, meshRenderer.bounds.center + Vector3.up * starSize);
1142  Gizmos.DrawLine(meshRenderer.bounds.center - Vector3.right * starSize, meshRenderer.bounds.center + Vector3.right * starSize);
1143  Gizmos.DrawLine(meshRenderer.bounds.center - Vector3.forward * starSize, meshRenderer.bounds.center + Vector3.forward * starSize);
1144  }
1145 }
1146 
1147 
1148 #if USE_GENERICS
1149 // Compares drawing layers of sprites
1150 public class SpriteDrawLayerComparer : IComparer<SpriteMesh_Managed>
1151 {
1152  public int Compare(SpriteMesh_Managed a, SpriteMesh_Managed b)
1153  {
1154  if (a.drawLayer > b.drawLayer)
1155  return 1;
1156  else if (a.drawLayer < b.drawLayer)
1157  return -1;
1158  else
1159  return 0;
1160  }
1161 }
1162 
1163 #else
1164 
1165 // Compares drawing layers of sprites
1166 public class SpriteDrawLayerComparer : IComparer
1167 {
1168  static SpriteMesh_Managed s1;
1169  static SpriteMesh_Managed s2;
1170 
1171  int IComparer.Compare(object a, object b)
1172  {
1173  s1 = (SpriteMesh_Managed)a;
1174  s2 = (SpriteMesh_Managed)b;
1175 
1176  if (s1.drawLayer > s2.drawLayer)
1177  return 1;
1178  else if (s1.drawLayer < s2.drawLayer)
1179  return -1;
1180  else
1181  return 0;
1182  }
1183 }
1184 
1185 #endif
int allocBlockSize
How many sprites to allocate at a time if the sprite pool is used up. This is also the starting pool ...
SpriteRoot CreateSprite(GameObject prefab)
Instantiates the specified prefab, which should contain a sprite, and immediately adds it to the mana...
void MoveBehind(SpriteMesh_Managed toMove, SpriteMesh_Managed reference)
Moves the first sprite behind the second sprite by placing it earlier in the draw order...
SpriteMesh_Managed AddSprite(GameObject go)
Adds the sprite attached to the specified GameObject to the SpriteManager.
SpriteRoot.WINDING_ORDER winding
Whether sprite polygons should be wound clock-wise or counter clock-wise. Determines which side of th...
void MoveToBack(SpriteMesh_Managed s)
Moves the specified sprite to the start of the drawing order. (Causes it to appear behind other sprit...
bool autoUpdateBounds
When true, will automatically recalculate the bounding box for the mesh whenever sprite vertices chan...
void RemoveSprite(SpriteRoot sprite)
Removes the specified sprite from the manager.
bool AlreadyAdded(SpriteRoot sprite)
Returns whether the sprite is already managed by this manager.
void MoveInfrontOf(SpriteMesh_Managed toMove, SpriteMesh_Managed reference)
Moves the first sprite in front of the second sprite by placing it later in the draw order...
Vector2 PixelCoordToUVCoord(Vector2 xy)
Converts pixel coordinates to UV coordinates according to the currently assigned material. NOTE: This is for converting coordinates and will reverse the Y component accordingly. For converting widths and heights, use PixelSpaceToUVSpace()!
void MoveToFront(SpriteMesh_Managed s)
Moves the specified sprite to the end of the drawing order. (Causes it to appear in front of other sp...
void CancelBoundsUpdate()
Cancels any previously scheduled bounds recalculations.
The root class of all sprites. Does not assume any animation capabilities or atlas packing...
Definition: SpriteRoot.cs:628
void UpdateBounds()
Instructs the manager to recalculate the bounds of the mesh.
SpriteManager manager
Reference to the manager which will manage this sprite, provided managed is set to true...
Definition: SpriteRoot.cs:702
void ScheduleBoundsUpdate(float seconds)
Schedules a recalculation of the mesh bounds to occur at a regular interval (given in seconds)...
Vector2 PixelSpaceToUVSpace(int x, int y)
Converts pixel-space values to UV-space scalar values according to the currently assigned material...
SpriteRoot CreateSprite(GameObject prefab, Vector3 position, Quaternion rotation)
Instantiates the specified prefab, which should contain a sprite, and immediately adds it to the mana...
bool persistent
If true, the SpriteManager object and associated mesh will survive a level load.
Vector2 PixelCoordToUVCoord(int x, int y)
Converts pixel coordinates to UV coordinates according to the currently assigned material. NOTE: This is for converting coordinates and will reverse the Y component accordingly. For converting widths and heights, use PixelSpaceToUVSpace()!
Allows multiple sprites to be combined into a single mesh. Sprites are transformed using bones which ...
Vector2 PixelSpaceToUVSpace(Vector2 xy)
Converts pixel-space values to UV-space scalar values according to the currently assigned material...
SpriteMesh_Managed AddSprite(SpriteRoot sprite)
Adds the specified sprite to the SpriteManager.