7 #define SPRITE_WANT_NORMALS
12 using System.Collections;
15 using System.Collections.Generic;
24 [RequireComponent(typeof(SkinnedMeshRenderer))]
55 public bool drawBoundingBox =
false;
63 protected bool initialized =
false;
65 protected EZLinkedList<SpriteMesh_Managed> availableBlocks =
new EZLinkedList<SpriteMesh_Managed>();
66 protected bool vertsChanged =
false;
67 protected bool uvsChanged =
false;
68 protected bool colorsChanged =
false;
69 protected bool vertCountChanged =
false;
70 protected bool updateBounds =
false;
71 protected SpriteMesh_Managed[] sprites;
72 protected EZLinkedList<SpriteMesh_Managed> activeBlocks =
new EZLinkedList<SpriteMesh_Managed>();
75 protected List<SpriteMesh_Managed> spriteDrawOrder =
new List<SpriteMesh_Managed>();
77 protected ArrayList spriteDrawOrder =
new ArrayList();
79 protected SpriteDrawLayerComparer drawOrderComparer =
new SpriteDrawLayerComparer();
80 protected float boundUpdateInterval;
82 protected List<SpriteRoot> spriteAddQueue;
84 protected SkinnedMeshRenderer meshRenderer;
86 protected Texture texture;
88 protected Transform[] bones;
89 protected BoneWeight[] boneWeights;
90 protected Matrix4x4[] bindPoses;
91 protected Vector3[] vertices;
92 protected int[] triIndices;
93 protected Vector2[] UVs;
94 protected Vector2[] UVs2;
95 protected Color[] colors;
98 protected SpriteMesh_Managed tempSprite = null;
118 return new Vector2(xy.x / ((
float)texture.width), xy.y / ((
float)texture.height));
150 return new Vector2(xy.x / ((
float)texture.width - 1), 1.0f - (xy.y / ((
float)texture.height - 1)));
169 protected void SetupBoneWeights(SpriteMesh_Managed s)
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;
188 if (spriteAddQueue == null)
189 spriteAddQueue =
new List<SpriteRoot>();
194 meshRenderer = (SkinnedMeshRenderer)GetComponent(typeof(SkinnedMeshRenderer));
196 if(meshRenderer != null)
197 if(meshRenderer.sharedMaterial != null)
198 texture = meshRenderer.sharedMaterial.GetTexture(
"_MainTex");
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
209 DontDestroyOnLoad(
this);
210 DontDestroyOnLoad(mesh);
219 transform.rotation = Quaternion.identity;
224 for(
int i=0; i<spriteAddQueue.Count; ++i)
229 protected void InitArrays()
231 bones =
new Transform[1];
232 bones[0] = transform;
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];
242 sprites[0].index = 0;
247 SetupBoneWeights(sprites[0]);
257 protected int EnlargeArrays(
int count)
268 firstNewElement = sprites.Length;
271 SpriteMesh_Managed[] tempSprites = sprites;
272 sprites =
new SpriteMesh_Managed[sprites.Length + count];
273 tempSprites.CopyTo(sprites, 0);
276 Transform[] tempBones = bones;
277 bones =
new Transform[bones.Length + count];
278 tempBones.CopyTo(bones, 0);
281 Matrix4x4[] tempPoses = bindPoses;
282 bindPoses =
new Matrix4x4[bindPoses.Length + count];
283 tempPoses.CopyTo(bindPoses, 0);
286 Vector3[] tempVerts = vertices;
287 vertices =
new Vector3[vertices.Length + count * 4];
288 tempVerts.CopyTo(vertices, 0);
291 BoneWeight[] tempWeights = boneWeights;
292 boneWeights =
new BoneWeight[boneWeights.Length + count * 4];
293 tempWeights.CopyTo(boneWeights, 0);
296 Vector2[] tempUVs = UVs;
297 UVs =
new Vector2[UVs.Length + count * 4];
298 tempUVs.CopyTo(UVs, 0);
301 Color[] tempColors = colors;
302 colors =
new Color[colors.Length + count * 4];
303 tempColors.CopyTo(colors, 0);
306 int[] tempTris = triIndices;
307 triIndices =
new int[triIndices.Length + count * 6];
308 tempTris.CopyTo(triIndices, 0);
311 for (
int i = 0; i < firstNewElement; ++i)
313 sprites[i].SetBuffers(vertices, UVs, UVs2, colors);
318 for (
int i = firstNewElement; i < sprites.Length; ++i)
322 sprites[i] =
new SpriteMesh_Managed();
323 sprites[i].index = i;
324 sprites[i].manager =
this;
326 sprites[i].SetBuffers(vertices, UVs, UVs2, colors);
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;
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;
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;
347 availableBlocks.Add(sprites[i]);
362 triIndices[i * 6 + 0] = i * 4 + 0;
363 triIndices[i * 6 + 1] = i * 4 + 3;
364 triIndices[i * 6 + 2] = i * 4 + 1;
366 triIndices[i * 6 + 3] = i * 4 + 3;
367 triIndices[i * 6 + 4] = i * 4 + 2;
368 triIndices[i * 6 + 5] = i * 4 + 1;
372 spriteDrawOrder.Add(sprites[i]);
375 bones[i] = transform;
378 bindPoses[i] = bones[i].worldToLocalMatrix * transform.localToWorldMatrix;
381 SetupBoneWeights(sprites[i]);
386 colorsChanged =
true;
387 vertCountChanged =
true;
389 return firstNewElement;
400 if(activeBlocks.Rewind())
404 if (activeBlocks.Current.sprite == sprite)
406 }
while (activeBlocks.MoveNext());
437 if(sprite.
manager ==
this && sprite.AddedToManager)
438 return (SpriteMesh_Managed)sprite.spriteMesh;
450 if (spriteAddQueue == null)
451 spriteAddQueue =
new List<SpriteRoot>();
453 spriteAddQueue.Add(sprite);
458 if (availableBlocks.Empty)
462 spriteIndex = availableBlocks.Head.index;
463 availableBlocks.Remove(availableBlocks.Head);
466 SpriteMesh_Managed newSpritesMesh = sprites[spriteIndex];
467 sprite.spriteMesh = newSpritesMesh;
468 sprite.manager =
this;
469 sprite.AddedToManager =
true;
470 newSpritesMesh.drawLayer = sprite.drawLayer;
473 bones[spriteIndex] = sprite.gameObject.transform;
475 bindPoses[spriteIndex] = bones[spriteIndex].worldToLocalMatrix * sprite.transform.localToWorldMatrix;
486 activeBlocks.Add(newSpritesMesh);
488 newSpritesMesh.Init();
494 vertCountChanged =
true;
498 return newSpritesMesh;
510 return CreateSprite(prefab, Vector3.zero, Quaternion.identity);
524 GameObject go = (GameObject) Instantiate(prefab, position, rotation);
549 if (sprite.spriteMesh is SpriteMesh_Managed && sprite.spriteMesh != null)
554 sprite.manager = null;
555 sprite.AddedToManager =
false;
566 vertices[sprite.mv1] = Vector3.zero;
567 vertices[sprite.mv2] = Vector3.zero;
568 vertices[sprite.mv3] = Vector3.zero;
569 vertices[sprite.mv4] = Vector3.zero;
579 activeBlocks.Remove(sprite);
582 if (gameObject != null)
583 bones[sprite.index] = transform;
587 sprite.sprite.spriteMesh = null;
588 sprite.sprite = null;
590 availableBlocks.Add(sprite);
603 int[] indices =
new int[6];
604 int offset = spriteDrawOrder.IndexOf(s) * 6;
609 s.drawLayer = spriteDrawOrder[spriteDrawOrder.Count - 1].drawLayer + 1;
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];
620 for (
int i = offset; i < triIndices.Length - 6; i += 6)
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];
629 spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 + 1];
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];
641 spriteDrawOrder[spriteDrawOrder.Count - 1] = s;
643 vertCountChanged =
true;
653 int[] indices =
new int[6];
654 int offset = spriteDrawOrder.IndexOf(s) * 6;
659 s.drawLayer = spriteDrawOrder[0].drawLayer - 1;
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];
670 for (
int i = offset; i > 5; i -= 6)
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];
679 spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 - 1];
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];
691 spriteDrawOrder[0] = s;
693 vertCountChanged =
true;
703 public void MoveInfrontOf(SpriteMesh_Managed toMove, SpriteMesh_Managed reference)
705 int[] indices =
new int[6];
706 int offset = spriteDrawOrder.IndexOf(toMove) * 6;
707 int refOffset = spriteDrawOrder.IndexOf(reference) * 6;
713 if (offset > refOffset)
716 toMove.drawLayer = reference.drawLayer + 1;
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];
727 for (
int i = offset; i < refOffset; i += 6)
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];
736 spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 + 1];
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];
748 spriteDrawOrder[refOffset / 6] = toMove;
750 vertCountChanged =
true;
760 public void MoveBehind(SpriteMesh_Managed toMove, SpriteMesh_Managed reference)
762 int[] indices =
new int[6];
763 int offset = spriteDrawOrder.IndexOf(toMove) * 6;
764 int refOffset = spriteDrawOrder.IndexOf(reference) * 6;
770 if (offset < refOffset)
773 toMove.drawLayer = reference.drawLayer - 1;
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];
784 for (
int i = offset; i > refOffset; i -= 6)
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];
793 spriteDrawOrder[i / 6] = spriteDrawOrder[i / 6 - 1];
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];
805 spriteDrawOrder[refOffset / 6] = toMove;
807 vertCountChanged =
true;
812 public void SortDrawingOrder()
814 SpriteMesh_Managed s;
816 spriteDrawOrder.Sort(drawOrderComparer);
821 for (
int i = 0; i < spriteDrawOrder.Count; ++i)
823 s = (SpriteMesh_Managed)spriteDrawOrder[i];
826 triIndices[i * 6 + 0] = s.mv1;
827 triIndices[i * 6 + 1] = s.mv2;
828 triIndices[i * 6 + 2] = s.mv4;
830 triIndices[i * 6 + 3] = s.mv4;
831 triIndices[i * 6 + 4] = s.mv2;
832 triIndices[i * 6 + 5] = s.mv3;
837 for (
int i = 0; i < spriteDrawOrder.Count; ++i)
839 s = (SpriteMesh_Managed)spriteDrawOrder[i];
842 triIndices[i * 6 + 0] = s.mv1;
843 triIndices[i * 6 + 1] = s.mv4;
844 triIndices[i * 6 + 2] = s.mv2;
846 triIndices[i * 6 + 3] = s.mv4;
847 triIndices[i * 6 + 4] = s.mv3;
848 triIndices[i * 6 + 5] = s.mv2;
852 vertCountChanged =
true;
856 public SpriteMesh_Managed GetSprite(
int i)
858 if (i < sprites.Length)
883 public void UpdatePositions()
889 public void UpdateUVs()
895 public void UpdateColors()
897 colorsChanged =
true;
915 boundUpdateInterval = seconds;
916 InvokeRepeating(
"UpdateBounds", seconds, seconds);
924 CancelInvoke(
"UpdateBounds");
928 public bool IsInitialized
930 get {
return initialized; }
934 virtual public void LateUpdate()
937 if (vertCountChanged)
939 vertCountChanged =
false;
940 colorsChanged =
false;
941 vertsChanged =
false;
943 updateBounds =
false;
945 meshRenderer.bones = bones;
948 mesh.vertices = vertices;
949 mesh.bindposes = bindPoses;
950 mesh.boneWeights = boneWeights;
952 mesh.colors = colors;
953 mesh.triangles = triIndices;
955 #if SPRITE_WANT_NORMALS
956 mesh.RecalculateNormals();
959 mesh.RecalculateBounds();
965 vertsChanged =
false;
970 mesh.vertices = vertices;
975 mesh.RecalculateBounds();
976 updateBounds =
false;
981 colorsChanged =
false;
983 mesh.colors = colors;
1042 public virtual void DoMirror()
1045 if (Application.isPlaying)
1049 if (vertCountChanged)
1051 vertCountChanged =
false;
1052 colorsChanged =
false;
1053 vertsChanged =
false;
1055 updateBounds =
false;
1057 meshRenderer.bones = bones;
1060 mesh.vertices = vertices;
1061 mesh.bindposes = bindPoses;
1062 mesh.boneWeights = boneWeights;
1064 mesh.colors = colors;
1065 mesh.triangles = triIndices;
1071 vertsChanged =
false;
1073 updateBounds =
true;
1075 mesh.vertices = vertices;
1080 mesh.RecalculateBounds();
1081 updateBounds =
false;
1086 colorsChanged =
false;
1088 mesh.colors = colors;
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
1107 public virtual void OnDrawGizmos()
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
1112 if (drawBoundingBox)
1114 Gizmos.color = Color.yellow;
1117 Gizmos.DrawWireCube(meshRenderer.bounds.center, meshRenderer.bounds.size);
1122 public void OnDrawGizmosSelected()
1124 Gizmos.color = Color.yellow;
1127 Gizmos.DrawWireCube(meshRenderer.bounds.center, meshRenderer.bounds.size);
1130 protected void DrawCenter()
1135 longestSide = Mathf.Max(meshRenderer.bounds.size.x, meshRenderer.bounds.size.y);
1136 longestSide = Mathf.Max(longestSide, meshRenderer.bounds.size.z);
1138 starSize = longestSide * 0.015f;
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);
1150 public class SpriteDrawLayerComparer : IComparer<SpriteMesh_Managed>
1152 public int Compare(SpriteMesh_Managed a, SpriteMesh_Managed b)
1154 if (a.drawLayer > b.drawLayer)
1156 else if (a.drawLayer < b.drawLayer)
1166 public class SpriteDrawLayerComparer : IComparer
1168 static SpriteMesh_Managed s1;
1169 static SpriteMesh_Managed s2;
1171 int IComparer.Compare(
object a,
object b)
1173 s1 = (SpriteMesh_Managed)a;
1174 s2 = (SpriteMesh_Managed)b;
1176 if (s1.drawLayer > s2.drawLayer)
1178 else if (s1.drawLayer < s2.drawLayer)
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...
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...
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.