using System; using System.Linq; using Unity.Jobs; using Unity.Collections; using Unity.Mathematics; using Unity.Collections.LowLevel.Unsafe; using Unity.Profiling; using Unity.SpriteShape.External.LibTessDotNet; using UnityEngine.U2D.Common.UTess; // We will enable this once Burst gets a verified final version as this attribute keeps changing. #if ENABLE_SPRITESHAPE_BURST using Unity.Burst; #endif namespace UnityEngine.U2D { internal enum SpriteShapeGeneratorResult { ErrorDefaultQuadCreated = -5, ErrorNativeDataOverflow = -4, ErrorSpritesWrongBorder = -3, ErrorSpritesTightPacked = -2, ErrorVertexLimitReached = -1, Success = 0, } // Generator Stats. internal struct SpriteShapeGeneratorStats { public SpriteShapeGeneratorResult status; // Adds any error Status } #if ENABLE_SPRITESHAPE_BURST [BurstCompile] #endif internal struct SpriteShapeGenerator : IJob { public ProfilerMarker generateGeometry; public ProfilerMarker generateCollider; struct JobParameters { public int4 shapeData; // x : ClosedShape (bool) y : AdaptiveUV (bool) z : SpriteBorders (bool) w : Enable Fill Texture. public int4 splineData; // x : StrtechUV. y : splineDetail z : _unused_ w: Collider On/Off public float4 curveData; // x : ColliderPivot y : BorderPivot z : AngleThreshold w : _unused_. public float4 fillData; // x : fillScale y : fillScale.x W z : fillScale.y H w: 0. } struct JobSpriteInfo { public float4 texRect; // TextureRect. public float4 texData; // x : GPUWidth y : GPUHeight z : TexelWidth w : TexelHeight public float4 uvInfo; // x : x, y : y, z : width, w : height public float4 metaInfo; // x : PPU, y : Pivot Y z : Original Rect Width w : Original Rect Height. public float4 border; // Sprite Border. } struct JobAngleRange { public float4 spriteAngles; // x, y | First Angle & z,w | Second Angle. public int4 spriteData; // Additional Data. x : sorting Order. y : variant Count. z : render Order Max. }; struct JobControlPoint { public int4 cpData; // x : Sprite Index y : Corner Type z : Mode w : Internal Sprite Index. public int4 exData; // x : Corner Type y: Corner Sprite z : Corner 0(disabled), 1 (stretch), (2, 3)(corner start/end) public float2 cpInfo; // x : Height y : Render Order. public float2 position; public float2 tangentLt; public float2 tangentRt; }; struct JobContourPoint { public float2 position; // x, y Position. public float2 ptData; // x : height. y :source cp. } struct JobIntersectPoint { public float2 top; public float2 bottom; } // Tessellation Structures. struct JobSegmentInfo { public int4 sgInfo; // x : Begin y : End. z : Sprite w : First Sprite for that Angle Range. public float4 spriteInfo; // x : width y : height z : Render Order. w : 0 (no) 1 (left stretchy) 2(right) }; struct JobCornerInfo { public float2 bottom; public float2 top; public float2 left; public float2 right; public int2 cornerData; } struct JobShapeVertex { public float2 pos; public float2 uv; public float4 tan; public float2 meta; // x : height y : - public int4 sprite; // x : sprite y : is main Point z : is edgeCaps. } /// /// Native Arrays : Scope : Initialized before and ReadOnly During Job /// [ReadOnly] private JobParameters m_ShapeParams; [ReadOnly] [DeallocateOnJobCompletion] private NativeArray m_SpriteInfos; [ReadOnly] [DeallocateOnJobCompletion] private NativeArray m_CornerSpriteInfos; [ReadOnly] [DeallocateOnJobCompletion] private NativeArray m_AngleRanges; /// /// Native Arrays : Scope : Job /// [DeallocateOnJobCompletion] private NativeArray m_Segments; private int m_SegmentCount; [DeallocateOnJobCompletion] private NativeArray m_ContourPoints; private int m_ContourPointCount; [DeallocateOnJobCompletion] private NativeArray m_Corners; private int m_CornerCount; [DeallocateOnJobCompletion] private NativeArray m_TessPoints; private int m_TessPointCount; [DeallocateOnJobCompletion] private NativeArray m_ControlPoints; private int m_ControlPointCount; [DeallocateOnJobCompletion] private NativeArray m_CornerCoordinates; [DeallocateOnJobCompletion] private NativeArray m_GeneratedControlPoints; [DeallocateOnJobCompletion] private NativeArray m_SpriteIndices; [DeallocateOnJobCompletion] private NativeArray m_Intersectors; /// /// Output Native Arrays : Scope : SpriteShapeRenderer / SpriteShapeController. /// private int m_IndexArrayCount; public NativeArray m_IndexArray; private int m_VertexArrayCount; public NativeSlice m_PosArray; public NativeSlice m_Uv0Array; public NativeSlice m_TanArray; private int m_GeomArrayCount; public NativeArray m_GeomArray; private int m_ColliderPointCount; public NativeArray m_ColliderPoints; public NativeArray m_Bounds; public NativeArray m_Stats; int m_IndexDataCount; int m_VertexDataCount; int m_ColliderDataCount; int m_ActiveIndexCount; int m_ActiveVertexCount; float2 m_FirstLT; float2 m_FirstLB; float4x4 m_Transform; int kModeLinear; int kModeContinous; int kModeBroken; int kModeUTess; int kCornerTypeOuterTopLeft; int kCornerTypeOuterTopRight; int kCornerTypeOuterBottomLeft; int kCornerTypeOuterBottomRight; int kCornerTypeInnerTopLeft; int kCornerTypeInnerTopRight; int kCornerTypeInnerBottomLeft; int kCornerTypeInnerBottomRight; int kControlPointCount; int kMaxArrayCount; float kEpsilon; float kEpsilonOrder; float kEpsilonRelaxed; float kExtendSegment; float kRenderQuality; float kOptimizeRender; float kColliderQuality; float kOptimizeCollider; float kLowestQualityTolerance; float kHighestQualityTolerance; #region Getters. // Return Vertex Data Count private int vertexDataCount { get { return m_VertexDataCount; } } // Return final Vertex Array Count private int vertexArrayCount { get { return m_VertexArrayCount; } } // Return Index Data Count private int indexDataCount { get { return m_IndexDataCount; } } // Return Sprite Count private int spriteCount { get { return m_SpriteInfos.Length; } } private int cornerSpriteCount { get { return m_CornerSpriteInfos.Length; } } // Return Angle Range Count private int angleRangeCount { get { return m_AngleRanges.Length; } } // Return the Input Control Point Count. private int controlPointCount { get { return m_ControlPointCount; } } // Return the Contour Point Count. private int contourPointCount { get { return m_ContourPointCount; } } // Return Segment Count private int segmentCount { get { return m_SegmentCount; } } // Needs Collider Generaie. private bool hasCollider { get { return m_ShapeParams.splineData.w == 1; } } // Collider Pivot private float colliderPivot { get { return m_ShapeParams.curveData.x; } } // Border Pivot private float borderPivot { get { return m_ShapeParams.curveData.y; } } // Spline Detail private int splineDetail { get { return m_ShapeParams.splineData.y; } } // Is this Closed-Loop. private bool isCarpet { get { return m_ShapeParams.shapeData.x == 1; } } // Is Adaptive UV private bool isAdaptive { get { return m_ShapeParams.shapeData.y == 1; } } // Has Sprite Border. private bool hasSpriteBorder { get { return m_ShapeParams.shapeData.z == 1; } } #endregion #region Safe Getters. JobSpriteInfo GetSpriteInfo(int index) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if (index >= m_SpriteInfos.Length) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException( $"GetSpriteInfo accessed with invalid Index {index} / {m_SpriteInfos.Length}"); } #endif return m_SpriteInfos[index]; } JobSpriteInfo GetCornerSpriteInfo(int index) { int ai = index - 1; #if ENABLE_UNITY_COLLECTIONS_CHECKS if (ai >= m_CornerSpriteInfos.Length || index == 0) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException( $"GetCornerSpriteInfo accessed with invalid Index {index} / {m_CornerSpriteInfos.Length}"); } #endif return m_CornerSpriteInfos[ai]; } JobAngleRange GetAngleRange(int index) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if (index >= m_AngleRanges.Length) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException( $"GetAngleRange accessed with invalid Index {index} / {m_AngleRanges.Length}"); } #endif return m_AngleRanges[index]; } JobControlPoint GetControlPoint(int index) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if (index >= m_ControlPoints.Length) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException( $"GetControlPoint accessed with invalid Index {index} / {m_ControlPoints.Length}"); } #endif return m_ControlPoints[index]; } JobContourPoint GetContourPoint(int index) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if (index >= m_ContourPointCount) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException( $"GetContourPoint accessed with invalid Index {index} / {m_ContourPointCount}"); } #endif return m_ContourPoints[index]; } JobSegmentInfo GetSegmentInfo(int index) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if (index >= m_SegmentCount) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException($"GetSegmentInfo accessed with invalid Index {index} / {m_SegmentCount}"); } #endif return m_Segments[index]; } int GetContourIndex(int index) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if (index >= m_ControlPoints.Length) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException( $"GetContourIndex accessed with invalid Index {index} / {m_ControlPoints.Length}"); } #endif return index * m_ShapeParams.splineData.y; } int GetEndContourIndexOfSegment(JobSegmentInfo isi) { int contourIndex = GetContourIndex(isi.sgInfo.y) - 1; #if ENABLE_UNITY_COLLECTIONS_CHECKS if (isi.sgInfo.y >= m_ControlPoints.Length || isi.sgInfo.y == 0) { SetResult(SpriteShapeGeneratorResult.ErrorNativeDataOverflow); throw new ArgumentException($"GetEndContourIndexOfSegment accessed with invalid Index"); } #endif return contourIndex; } void SetResult(SpriteShapeGeneratorResult result) { if (m_Stats.IsCreated) { var stats = m_Stats[0]; stats.status = result; m_Stats[0] = stats; } } #endregion #region Utility static void CopyToNativeArray(NativeArray from, int length, ref NativeArray to) where T : struct { to = new NativeArray(length, Allocator.TempJob); for (int i = 0; i < length; ++i) to[i] = from[i]; } static void SafeDispose(NativeArray na) where T : struct { if (na.Length > 0) na.Dispose(); } static bool IsPointOnLine(float epsilon, float2 a, float2 b, float2 c) { float cp = (c.y - a.y) * (b.x - a.x) - (c.x - a.x) * (b.y - a.y); if (math.abs(cp) > epsilon) return false; float dp = (c.x - a.x) * (b.x - a.x) + (c.y - a.y) * (b.y - a.y); if (dp < 0) return false; float ba = (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y); if (dp > ba) return false; return true; } static bool IsPointOnLines(float epsilon, float2 p1, float2 p2, float2 p3, float2 p4, float2 r) { return IsPointOnLine(epsilon, p1, p2, r) && IsPointOnLine(epsilon, p3, p4, r); } static bool Colinear(float2 p, float2 q, float2 r) { return (q.x <= math.max(p.x, r.x) && q.x >= math.min(p.x, r.x) && q.y <= math.max(p.y, r.y) && q.y >= math.min(p.y, r.y)); } static int Det(float epsilon, float2 p, float2 q, float2 r) { float val = (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y); if (val > -epsilon && val < epsilon) return 0; return (val > 0) ? 1 : 2; } static bool LineIntersectionTest(float epsilon, float2 p1, float2 q1, float2 p2, float2 q2) { int o1 = Det(epsilon, p1, q1, p2); int o2 = Det(epsilon, p1, q1, q2); int o3 = Det(epsilon, p2, q2, p1); int o4 = Det(epsilon, p2, q2, q1); if (o1 != o2 && o3 != o4) return true; // Special Cases if (o1 == 0 && Colinear(p1, p2, q1)) return true; if (o2 == 0 && Colinear(p1, q2, q1)) return true; if (o3 == 0 && Colinear(p2, p1, q2)) return true; if (o4 == 0 && Colinear(p2, q1, q2)) return true; return false; } static bool LineIntersection(float epsilon, float2 p1, float2 p2, float2 p3, float2 p4, ref float2 result) { if (!LineIntersectionTest(epsilon, p1, p2, p3, p4)) { return false; } float bx = p2.x - p1.x; float by = p2.y - p1.y; float dx = p4.x - p3.x; float dy = p4.y - p3.y; float bDotDPerp = bx * dy - by * dx; if (math.abs(bDotDPerp) < epsilon) { return false; } float cx = p3.x - p1.x; float cy = p3.y - p1.y; float t = (cx * dy - cy * dx) / bDotDPerp; if ((t >= -epsilon) && (t <= 1.0f + epsilon)) { result.x = p1.x + t * bx; result.y = p1.y + t * by; return true; } return false; } static float AngleBetweenVector(float2 a, float2 b) { float dot = math.dot(a, b); float det = (a.x * b.y) - (b.x * a.y); return math.atan2(det, dot) * Mathf.Rad2Deg; } static bool GenerateColumnsBi(float2 a, float2 b, float2 whsize, bool flip, ref float2 rt, ref float2 rb, float cph, float pivot) { float2 v1 = flip ? (a - b) : (b - a); if (math.length(v1) < 1e-30f) return false; float2 rvxy = new float2(-1f, 1f); float2 v2 = v1.yx * rvxy; float2 whsizey = new float2(whsize.y * cph); v2 = math.normalize(v2); float2 v3 = v2 * whsizey; rt = a - v3; rb = a + v3; float2 pivotSet = (rb - rt) * pivot; rt = rt + pivotSet; rb = rb + pivotSet; return true; } static bool GenerateColumnsTri(float2 a, float2 b, float2 c, float2 whsize, bool flip, ref float2 rt, ref float2 rb, float cph, float pivot) { float2 rvxy = new float2(-1f, 1f); float2 v0 = b - a; float2 v1 = c - b; v0 = v0.yx * rvxy; v1 = v1.yx * rvxy; float2 v2 = math.normalize(v0) + math.normalize(v1); if (math.length(v2) < 1e-30f) return false; v2 = math.normalize(v2); float2 whsizey = new float2(whsize.y * cph); float2 v3 = v2 * whsizey; rt = b - v3; rb = b + v3; float2 pivotSet = (rb - rt) * pivot; rt = rt + pivotSet; rb = rb + pivotSet; return true; } #endregion #region Input Preparation. void AppendCornerCoordinates(ref NativeArray corners, ref int cornerCount, float2 a, float2 b, float2 c, float2 d) { corners[cornerCount++] = a; corners[cornerCount++] = b; corners[cornerCount++] = c; corners[cornerCount++] = d; } void PrepareInput(SpriteShapeParameters shapeParams, int maxArrayCount, NativeArray shapePoints, bool optimizeGeometry, bool updateCollider, bool optimizeCollider, float colliderOffset, float colliderDetail) { kModeLinear = 0; kModeContinous = 1; kModeBroken = 2; kCornerTypeOuterTopLeft = 1; kCornerTypeOuterTopRight = 2; kCornerTypeOuterBottomLeft = 3; kCornerTypeOuterBottomRight = 4; kCornerTypeInnerTopLeft = 5; kCornerTypeInnerTopRight = 6; kCornerTypeInnerBottomLeft = 7; kCornerTypeInnerBottomRight = 8; kMaxArrayCount = maxArrayCount; m_IndexDataCount = 0; m_VertexDataCount = 0; m_ColliderDataCount = 0; m_ActiveIndexCount = 0; m_ActiveVertexCount = 0; kEpsilon = 0.00001f; kEpsilonOrder = -0.0001f; kEpsilonRelaxed = 0.001f; kExtendSegment = 10000.0f; kLowestQualityTolerance = 4.0f; kHighestQualityTolerance = 16.0f; kColliderQuality = math.clamp(colliderDetail, kLowestQualityTolerance, kHighestQualityTolerance); kOptimizeCollider = optimizeCollider ? 1 : 0; kColliderQuality = (kHighestQualityTolerance - kColliderQuality + 2.0f) * 0.002f; colliderOffset = (colliderOffset == 0) ? kEpsilonRelaxed : -colliderOffset; kOptimizeRender = optimizeGeometry ? 1 : 0; kRenderQuality = math.clamp(shapeParams.splineDetail, kLowestQualityTolerance, kHighestQualityTolerance); kRenderQuality = (kHighestQualityTolerance - kRenderQuality + 2.0f) * 0.0002f; m_ShapeParams.shapeData = new int4(shapeParams.carpet ? 1 : 0, shapeParams.adaptiveUV ? 1 : 0, shapeParams.spriteBorders ? 1 : 0, shapeParams.fillTexture != null ? 1 : 0); m_ShapeParams.splineData = new int4(shapeParams.stretchUV ? 1 : 0, (shapeParams.splineDetail > 4) ? (int)shapeParams.splineDetail : 4, 0, updateCollider ? 1 : 0); m_ShapeParams.curveData = new float4(colliderOffset, shapeParams.borderPivot, shapeParams.angleThreshold, 0); float fx = 0, fy = 0; if (shapeParams.fillTexture != null) { fx = (float)shapeParams.fillTexture.width * (1.0f / (float)shapeParams.fillScale); fy = (float)shapeParams.fillTexture.height * (1.0f / (float)shapeParams.fillScale); } m_ShapeParams.fillData = new float4(shapeParams.fillScale, fx, fy, 0); unsafe { UnsafeUtility.MemClear(m_GeomArray.GetUnsafePtr(), m_GeomArray.Length * UnsafeUtility.SizeOf()); } m_Transform = new float4x4(shapeParams.transform.m00, shapeParams.transform.m01, shapeParams.transform.m02, shapeParams.transform.m03, shapeParams.transform.m10, shapeParams.transform.m11, shapeParams.transform.m12, shapeParams.transform.m13, shapeParams.transform.m20, shapeParams.transform.m21, shapeParams.transform.m22, shapeParams.transform.m23, shapeParams.transform.m30, shapeParams.transform.m31, shapeParams.transform.m32, shapeParams.transform.m33); kControlPointCount = shapePoints.Length * (int)shapeParams.splineDetail * 32; m_Segments = new NativeArray(shapePoints.Length * 2, Allocator.TempJob, NativeArrayOptions.UninitializedMemory); m_ContourPoints = new NativeArray(kControlPointCount, Allocator.TempJob, NativeArrayOptions.UninitializedMemory); m_TessPoints = new NativeArray(shapePoints.Length * (int)shapeParams.splineDetail * 128, Allocator.TempJob, NativeArrayOptions.UninitializedMemory); m_CornerCoordinates = new NativeArray(32, Allocator.TempJob, NativeArrayOptions.UninitializedMemory); m_Intersectors = new NativeArray(kControlPointCount, Allocator.TempJob, NativeArrayOptions.ClearMemory); m_GeneratedControlPoints = new NativeArray(kControlPointCount, Allocator.TempJob, NativeArrayOptions.UninitializedMemory); m_SpriteIndices = new NativeArray(kControlPointCount, Allocator.TempJob, NativeArrayOptions.UninitializedMemory); int cornerCount = 0; AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(1f, 1f), new float2(0, 1f), new float2(1f, 0), new float2(0, 0)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(1f, 0), new float2(1f, 1f), new float2(0, 0), new float2(0, 1f)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(0, 1f), new float2(0, 0), new float2(1f, 1f), new float2(1f, 0)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(0, 0), new float2(1f, 0), new float2(0, 1f), new float2(1f, 1f)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(0, 0), new float2(0, 1f), new float2(1f, 0), new float2(1f, 1f)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(0, 1f), new float2(1f, 1f), new float2(0, 0), new float2(1f, 0)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(1f, 0), new float2(0, 0), new float2(1f, 1f), new float2(0, 1f)); AppendCornerCoordinates(ref m_CornerCoordinates, ref cornerCount, new float2(1f, 1f), new float2(1f, 0), new float2(0, 1f), new float2(0, 0)); } void TransferSprites(ref NativeArray spriteInfos, Sprite[] sprites, int maxCount) { for (int i = 0; i < sprites.Length && i < maxCount; ++i) { JobSpriteInfo spriteInfo = spriteInfos[i]; Sprite spr = sprites[i]; if (spr != null) { Texture2D tex = spr.texture; spriteInfo.texRect = new float4(spr.textureRect.x, spr.textureRect.y, spr.textureRect.width, spr.textureRect.height); spriteInfo.texData = new float4(tex.width, tex.height, tex.texelSize.x, tex.texelSize.y); spriteInfo.border = new float4(spr.border.x, spr.border.y, spr.border.z, spr.border.w); spriteInfo.uvInfo = new float4(spriteInfo.texRect.x / spriteInfo.texData.x, spriteInfo.texRect.y / spriteInfo.texData.y, spriteInfo.texRect.z / spriteInfo.texData.x, spriteInfo.texRect.w / spriteInfo.texData.y); spriteInfo.metaInfo = new float4(spr.pixelsPerUnit, spr.pivot.y / spr.textureRect.height, spr.rect.width, spr.rect.height); if (!math.any(spriteInfo.texRect)) { SetResult(SpriteShapeGeneratorResult.ErrorSpritesTightPacked); } } spriteInfos[i] = spriteInfo; } } void PrepareSprites(Sprite[] edgeSprites, Sprite[] cornerSprites) { m_SpriteInfos = new NativeArray(edgeSprites.Length, Allocator.TempJob); TransferSprites(ref m_SpriteInfos, edgeSprites, edgeSprites.Length); m_CornerSpriteInfos = new NativeArray(kCornerTypeInnerBottomRight, Allocator.TempJob); TransferSprites(ref m_CornerSpriteInfos, cornerSprites, cornerSprites.Length); } void PrepareAngleRanges(AngleRangeInfo[] angleRanges) { m_AngleRanges = new NativeArray(angleRanges.Length, Allocator.TempJob); for (int i = 0; i < angleRanges.Length; ++i) { JobAngleRange angleRange = m_AngleRanges[i]; AngleRangeInfo ari = angleRanges[i]; int[] spr = ari.sprites; if (ari.start > ari.end) { var sw = ari.start; ari.start = ari.end; ari.end = sw; } angleRange.spriteAngles = new float4(ari.start + 90f, ari.end + 90f, 0, 0); angleRange.spriteData = new int4((int)ari.order, spr.Length, 32, 0); m_AngleRanges[i] = angleRange; } } void PrepareControlPoints(NativeArray shapePoints, NativeArray metaData) { float2 zero = new float2(0, 0); m_ControlPoints = new NativeArray(kControlPointCount, Allocator.TempJob); for (int i = 0; i < shapePoints.Length; ++i) { JobControlPoint shapePoint = m_ControlPoints[i]; ShapeControlPoint sp = shapePoints[i]; SplinePointMetaData md = metaData[i]; shapePoint.position = new float2(sp.position.x, sp.position.y); shapePoint.tangentLt = (sp.mode == kModeLinear) ? zero : new float2(sp.leftTangent.x, sp.leftTangent.y); shapePoint.tangentRt = (sp.mode == kModeLinear) ? zero : new float2(sp.rightTangent.x, sp.rightTangent.y); shapePoint.cpInfo = new float2(md.height, 0); shapePoint.cpData = new int4((int)md.spriteIndex, md.cornerMode, sp.mode, 0); shapePoint.exData = new int4(-1, 0, 0, sp.mode); m_ControlPoints[i] = shapePoint; } m_ControlPointCount = shapePoints.Length; m_Corners = new NativeArray(shapePoints.Length, Allocator.TempJob); GenerateControlPoints(); } #endregion #region Resolve Angles for Points. bool WithinRange(JobAngleRange angleRange, float inputAngle) { float range = angleRange.spriteAngles.y - angleRange.spriteAngles.x; float angle = Mathf.Repeat(inputAngle - angleRange.spriteAngles.x, 360f); return (angle >= 0f && angle <= range); } bool AngleWithinRange(float t, float a, float b) { return (a != 0 && b != 0) && (t >= a && t <= b); } static float2 BezierPoint(float2 st, float2 sp, float2 ep, float2 et, float t) { float2 xt = new float2(t); float2 nt = new float2(1.0f - t); float2 x3 = new float2(3.0f); return (sp * nt * nt * nt) + (st * nt * nt * xt * x3) + (et * nt * xt * xt * x3) + (ep * xt * xt * xt); } static float SlopeAngle(float2 dirNormalized) { float2 dvup = new float2(0, 1f); float2 dvrt = new float2(1f, 0); float dr = math.dot(dirNormalized, dvrt); float du = math.dot(dirNormalized, dvup); float cu = math.acos(du); float sn = dr >= 0 ? 1.0f : -1.0f; float an = cu * Mathf.Rad2Deg * sn; // Adjust angles when direction is parallel to Up Axis. an = (du != 1f) ? an : 0; an = (du != -1f) ? an : -180f; return an; } static float SlopeAngle(float2 start, float2 end) { float2 dir = math.normalize(start - end); return SlopeAngle(dir); } bool ResolveAngle(float angle, int activeIndex, ref float renderOrder, ref int spriteIndex, ref int firstSpriteIndex) { int localRenderOrder = 0; int localSpriteIndex = 0; for (int i = 0; i < m_AngleRanges.Length; ++i) { bool withinRange = WithinRange(m_AngleRanges[i], angle); if (withinRange) { int validIndex = (activeIndex < m_AngleRanges[i].spriteData.y) ? activeIndex : 0; renderOrder = localRenderOrder + validIndex; spriteIndex = localSpriteIndex + validIndex; firstSpriteIndex = localSpriteIndex; return true; } localRenderOrder += m_AngleRanges[i].spriteData.z; localSpriteIndex += m_AngleRanges[i].spriteData.y; } return false; } int GetSpriteIndex(int index, int previousIndex, ref int resolved) { int next = (index + 1) % controlPointCount, spriteIndex = -1, firstSpriteIndex = -1; float order = 0; var cp = GetControlPoint(index); float angle = SlopeAngle(GetControlPoint(next).position, cp.position); bool resolve = ResolveAngle(angle, cp.cpData.x, ref order, ref spriteIndex, ref firstSpriteIndex); resolved = resolve ? 1 : 0; return resolve ? spriteIndex : previousIndex; } #endregion #region Segments. void GenerateSegments() { int activeSpriteIndex = 0, activeSegmentIndex = 0, firstSpriteIndex = -1; JobSegmentInfo activeSegment = m_Segments[0]; activeSegment.sgInfo = int4.zero; activeSegment.spriteInfo = int4.zero; float angle = 0; // Generate Segments. for (int i = 0; i < controlPointCount; ++i) { int next = (i + 1) % controlPointCount; // Check for Last Point and see if we need loop-back. bool skipSegmenting = false; if (next == 0) { if (!isCarpet) continue; next = 1; skipSegmenting = true; } JobControlPoint iscp = GetControlPoint(i); JobControlPoint iscpNext = GetControlPoint(next); // If this segment is corner, continue. if (iscp.exData.x > 0 && iscp.exData.x == iscpNext.exData.x && iscp.exData.z == 2) continue; // Resolve Angle and Order. int4 pointData = iscp.cpData; float2 pointInfo = iscp.cpInfo; // Get Min Max Segment. int mn = (i < next) ? i : next; int mx = (i > next) ? i : next; bool continueStrip = (iscp.cpData.z == kModeContinous), edgeUpdated = false; if (false == continueStrip || 0 == activeSegmentIndex) angle = SlopeAngle(iscpNext.position, iscp.position); bool resolved = ResolveAngle(angle, pointData.x, ref pointInfo.y, ref pointData.w, ref firstSpriteIndex); if (!resolved && !skipSegmenting) { // If we do not resolve SpriteIndex (AngleRange) just continue existing segment. pointData.w = activeSpriteIndex; iscp.cpData = pointData; m_ControlPoints[i] = iscp; // Insert Dummy Segment. activeSegment = m_Segments[activeSegmentIndex]; activeSegment.sgInfo.x = mn; activeSegment.sgInfo.y = mx; activeSegment.sgInfo.z = -1; m_Segments[activeSegmentIndex] = activeSegment; activeSegmentIndex++; continue; } // Update current Point. activeSpriteIndex = pointData.w; iscp.cpData = pointData; iscp.cpInfo = pointInfo; m_ControlPoints[i] = iscp; if (skipSegmenting) continue; // Check for Segments. Also check if the Segment Start has been resolved. Otherwise simply start with the next one. if (activeSegmentIndex != 0) continueStrip = continueStrip && (m_SpriteIndices[activeSegment.sgInfo.x].y != 0 && activeSpriteIndex == activeSegment.sgInfo.z); if (continueStrip && i != (controlPointCount - 1)) { for (int s = 0; s < activeSegmentIndex; ++s) { activeSegment = m_Segments[s]; if (activeSegment.sgInfo.x - mn == 1) { edgeUpdated = true; activeSegment.sgInfo.x = mn; m_Segments[s] = activeSegment; break; } if (mx - activeSegment.sgInfo.y == 1) { edgeUpdated = true; activeSegment.sgInfo.y = mx; m_Segments[s] = activeSegment; break; } } } if (!edgeUpdated) { activeSegment = m_Segments[activeSegmentIndex]; JobSpriteInfo sprLt = GetSpriteInfo(iscp.cpData.w); activeSegment.sgInfo.x = mn; activeSegment.sgInfo.y = mx; activeSegment.sgInfo.z = activeSpriteIndex; activeSegment.sgInfo.w = firstSpriteIndex; activeSegment.spriteInfo.x = sprLt.texRect.z; activeSegment.spriteInfo.y = sprLt.texRect.w; activeSegment.spriteInfo.z = pointInfo.y; m_Segments[activeSegmentIndex] = activeSegment; activeSegmentIndex++; } } m_SegmentCount = activeSegmentIndex; } void UpdateSegments() { // Determine Distance of Segment. for (int i = 0; i < segmentCount; ++i) { // Calculate Segment Distances. JobSegmentInfo isi = GetSegmentInfo(i); if (isi.spriteInfo.z >= 0) { isi.spriteInfo.w = SegmentDistance(isi); m_Segments[i] = isi; } } } bool GetSegmentBoundaryColumn(JobSegmentInfo segment, JobSpriteInfo sprInfo, float2 whsize, float2 startPos, float2 endPos, bool end, ref float2 top, ref float2 bottom) { bool res = false; float pivot = 0.5f - sprInfo.metaInfo.y; if (!end) { JobControlPoint icp = GetControlPoint(segment.sgInfo.x); if (math.any(icp.tangentRt)) endPos = icp.tangentRt + startPos; res = GenerateColumnsBi(startPos, endPos, whsize, end, ref top, ref bottom, icp.cpInfo.x * 0.5f, pivot); } else { JobControlPoint jcp = GetControlPoint(segment.sgInfo.y); if (math.any(jcp.tangentLt)) endPos = jcp.tangentLt + startPos; res = GenerateColumnsBi(startPos, endPos, whsize, end, ref top, ref bottom, jcp.cpInfo.x * 0.5f, pivot); } return res; } void GenerateControlPoints() { // Globals. int activePoint = 0, activeIndex = 0; int startPoint = 0, endPoint = controlPointCount, lastPoint = (controlPointCount - 1); int2 sprData = new int2(0, 0); // Calc and calculate Indices. for (int i = 0; i < controlPointCount; ++i) { var resolved = 0; int spriteIndex = GetSpriteIndex(i, activeIndex, ref resolved); sprData.x = activeIndex = spriteIndex; sprData.y = resolved; m_SpriteIndices[i] = sprData; } // Open-Ended. We simply dont allow Continous mode in End-points. if (!isCarpet) { JobControlPoint cp = GetControlPoint(0); cp.cpData.z = (cp.cpData.z == kModeContinous) ? kModeBroken : cp.cpData.z; m_GeneratedControlPoints[activePoint++] = cp; // If its not carpet, we already pre-insert start and endpoint. startPoint = 1; endPoint = controlPointCount - 1; } // Generate Intermediates. for (int i = startPoint; i < endPoint; ++i) { // Check if the Neighbor Points are all in Linear Mode. bool cornerCriteriaMet = false; bool vc = InsertCorner(i, ref m_SpriteIndices, ref m_GeneratedControlPoints, ref activePoint, ref cornerCriteriaMet); if (vc) continue; // NO Corners. var cp = GetControlPoint(i); cp.exData.z = (cornerCriteriaMet && cp.cpData.y == 2) ? 1 : 0; // Set this to stretched of Corner criteria met but no corner sprites but stretched corner. m_GeneratedControlPoints[activePoint++] = cp; } // Open-Ended. if (!isCarpet) { // Fixup for End-Points and Point-Mode. JobControlPoint sp = m_GeneratedControlPoints[0]; sp.exData.z = 1; m_GeneratedControlPoints[0] = sp; JobControlPoint cp = GetControlPoint(endPoint); cp.cpData.z = (cp.cpData.z == kModeContinous) ? kModeBroken : cp.cpData.z; cp.exData.z = 1; m_GeneratedControlPoints[activePoint++] = cp; } // If Closed Shape else { JobControlPoint cp = m_GeneratedControlPoints[0]; m_GeneratedControlPoints[activePoint++] = cp; } // Copy from these intermediate Points to main Control Points. for (int i = 0; i < activePoint; ++i) m_ControlPoints[i] = m_GeneratedControlPoints[i]; m_ControlPointCount = activePoint; // Calc and calculate Indices. for (int i = 0; i < controlPointCount; ++i) { var resolved = 0; int spriteIndex = GetSpriteIndex(i, activeIndex, ref resolved); sprData.x = activeIndex = spriteIndex; sprData.y = resolved; m_SpriteIndices[i] = sprData; } } float SegmentDistance(JobSegmentInfo isi) { float distance = 0; int stIx = GetContourIndex(isi.sgInfo.x); int enIx = GetEndContourIndexOfSegment(isi); for (int i = stIx; i < enIx; ++i) { int j = i + 1; JobContourPoint lt = GetContourPoint(i); JobContourPoint rt = GetContourPoint(j); distance = distance + math.distance(lt.position, rt.position); } return distance; } void GenerateContour() { int controlPointContour = controlPointCount - 1; // Expand the Bezier. int ap = 0; float fmax = (float)(splineDetail - 1); for (int i = 0; i < controlPointContour; ++i) { int j = i + 1; JobControlPoint cp = GetControlPoint(i); JobControlPoint pp = GetControlPoint(j); var smoothInterp = cp.exData.w == kModeContinous || pp.exData.w == kModeContinous; float2 p0 = cp.position; float2 p1 = pp.position; float2 sp = p0; float2 rt = p0 + cp.tangentRt; float2 lt = p1 + pp.tangentLt; int cap = ap; float spd = 0, cpd = 0; for (int n = 0; n < splineDetail; ++n) { JobContourPoint xp = m_ContourPoints[ap]; float t = (float)n / fmax; float2 bp = BezierPoint(rt, p0, p1, lt, t); xp.position = bp; spd += math.distance(bp, sp); m_ContourPoints[ap++] = xp; sp = bp; } sp = p0; for (int n = 0; n < splineDetail; ++n) { JobContourPoint xp = m_ContourPoints[cap]; cpd += math.distance(xp.position, sp); xp.ptData.x = smoothInterp ? InterpolateSmooth(cp.cpInfo.x, pp.cpInfo.x, cpd / spd) : InterpolateLinear(cp.cpInfo.x, pp.cpInfo.x, cpd / spd); m_ContourPoints[cap++] = xp; sp = xp.position; } } // End m_ContourPointCount = ap; int tessPoints = 0; // Create Tessallator if required. for (int i = 0; i < contourPointCount; ++i) { if ((i + 1) % splineDetail == 0) continue; int h = (i == 0) ? (contourPointCount - 1) : (i - 1); int j = (i + 1) % contourPointCount; h = (i % splineDetail == 0) ? (h - 1) : h; JobContourPoint pp = GetContourPoint(h); JobContourPoint cp = GetContourPoint(i); JobContourPoint np = GetContourPoint(j); float2 cpd = cp.position - pp.position; float2 npd = np.position - cp.position; if (math.length(cpd) < kEpsilon || math.length(npd) < kEpsilon) continue; float2 vl = math.normalize(cpd); float2 vr = math.normalize(npd); vl = new float2(-vl.y, vl.x); vr = new float2(-vr.y, vr.x); float2 va = math.normalize(vl) + math.normalize(vr); float2 vn = math.normalize(va); if (math.any(va) && math.any(vn)) m_TessPoints[tessPoints++] = cp.position + (vn * borderPivot); } m_TessPointCount = tessPoints; } // Prepare Contour. bool PrepareContour() { // Generate Contour GenerateContour(); // Fill Geom. Generate in Native code until we have a reasonably fast enough Tessellation in NativeArray based Jobs. SpriteShapeSegment geom = m_GeomArray[0]; geom.vertexCount = 0; geom.geomIndex = 0; geom.indexCount = 0; geom.spriteIndex = -1; m_GeomArray[0] = geom; // Fill Geometry. Check if Fill Texture and Fill Scale is Valid. if (math.all(m_ShapeParams.shapeData.xw) && m_TessPointCount > 0) { if (kOptimizeRender > 0) OptimizePoints(kRenderQuality, true, ref m_TessPoints, ref m_TessPointCount); return true; } return false; } [BurstCompile] // Tess static unsafe void UTessellator(ref SpriteShapeSegment geom, int maxCount, float2* tessPoints, int tessPointCount, ushort* indices, ref int iCount, byte* vertices, int stride, ref int vCount, Unity.Collections.Allocator label) { NativeArray edges = new NativeArray(tessPointCount - 1, label); NativeArray points = new NativeArray(tessPointCount - 1, label); float kPrecisionFudge = 1.0f; for (int i = 0; i < points.Length; ++i) points[i] = tessPoints[i] * kPrecisionFudge; for (int i = 0; i < tessPointCount - 2; ++i) { int2 te = edges[i]; te.x = i; te.y = i + 1; edges[i] = te; } int2 tee = edges[tessPointCount - 2]; tee.x = tessPointCount - 2; tee.y = 0; edges[tessPointCount - 2] = tee; NativeArray ov = new NativeArray(tessPointCount * 4, label); NativeArray oi = new NativeArray(tessPointCount * 4, label); NativeArray oe = new NativeArray(tessPointCount * 4, label); UnityEngine.U2D.Common.UTess.ModuleHandle.Tessellate(label, in points, in edges, ref ov, out var ovc, ref oi, out var oic, ref oe, out var oec); ovc = ovc < maxCount ? ovc : maxCount; oic = oic < maxCount ? oic : maxCount; if (oic > 0) { for (vCount = 0; vCount < ovc; ++vCount) { Vector3* pos = (Vector3*)vertices; *pos = new Vector3(ov[vCount].x, ov[vCount].y, 0) / kPrecisionFudge; vertices = vertices + stride; } for (iCount = 0; iCount < oic; ++iCount) indices[iCount] = (ushort)oi[iCount]; } ov.Dispose(); oi.Dispose(); oe.Dispose(); edges.Dispose(); points.Dispose(); } // Burstable UTess2D Version. bool TessellateContour(Unity.Collections.Allocator label) { // Generate Contour bool innerShape = PrepareContour(); SpriteShapeSegment geom = m_GeomArray[0]; if (innerShape) { unsafe { UTessellator(ref geom, kMaxArrayCount, (float2*)m_TessPoints.GetUnsafePtr(), m_TessPointCount, (ushort*)m_IndexArray.GetUnsafePtr(), ref m_IndexDataCount, (byte*)m_PosArray.GetUnsafePtr(), m_PosArray.Stride, ref m_VertexDataCount, label); } if (m_IndexDataCount == 0 || m_VertexDataCount == 0) { m_IndexDataCount = m_ActiveVertexCount = 0; SetResult(SpriteShapeGeneratorResult.ErrorDefaultQuadCreated); } else { geom.indexCount = m_ActiveIndexCount = m_IndexDataCount; geom.vertexCount = m_ActiveVertexCount = m_VertexDataCount; if (m_TanArray.Length > 1) { for (int i = 0; i < m_ActiveVertexCount; ++i) m_TanArray[i] = new Vector4(1.0f, 0, 0, -1.0f); } } m_GeomArray[0] = geom; } return innerShape; } void TessellateContourMainThread() { // Generate Contour bool innerShape = PrepareContour(); SpriteShapeSegment geom = m_GeomArray[0]; // Fallback only when there is InnerShape and Utess faced knots/overlaps. if (innerShape && 0 == m_ActiveVertexCount) { // UTess failed, trying fallback. SetResult(SpriteShapeGeneratorResult.Success); var inputs = new ContourVertex[m_TessPointCount]; for (int i = 0; i < m_TessPointCount; ++i) inputs[i] = new ContourVertex() { Position = new Vec3() { X = m_TessPoints[i].x, Y = m_TessPoints[i].y } }; Tess tess = new Tess(); tess.AddContour(inputs, ContourOrientation.Original); tess.Tessellate(WindingRule.NonZero, ElementType.Polygons, 3); var indices = tess.Elements.Select(i => (UInt16)i).ToArray(); var vertices = tess.Vertices.Select(v => new Vector2(v.Position.X, v.Position.Y)).ToArray(); m_IndexDataCount = indices.Length; m_VertexDataCount = vertices.Length; if (vertices.Length > 0) { for (m_ActiveIndexCount = 0; m_ActiveIndexCount < m_IndexDataCount; ++m_ActiveIndexCount) m_IndexArray[m_ActiveIndexCount] = indices[m_ActiveIndexCount]; for (m_ActiveVertexCount = 0; m_ActiveVertexCount < m_VertexDataCount; ++m_ActiveVertexCount) m_PosArray[m_ActiveVertexCount] = new Vector3(vertices[m_ActiveVertexCount].x, vertices[m_ActiveVertexCount].y, 0); geom.indexCount = m_ActiveIndexCount; geom.vertexCount = m_ActiveVertexCount; } if (m_TanArray.Length > 1) { for (int i = 0; i < m_ActiveVertexCount; ++i) m_TanArray[i] = new Vector4(1.0f, 0, 0, -1.0f); } m_GeomArray[0] = geom; } } void CalculateBoundingBox() { if (vertexArrayCount == 0 && contourPointCount == 0) return; var bounds = new Bounds(); var min = vertexArrayCount != 0 ? new float2(m_PosArray[0].x, m_PosArray[0].y) : new float2(m_ContourPoints[0].position.x, m_ContourPoints[0].position.y); var max = min; { for (int i = 0; i < vertexArrayCount; ++i) { float3 pos = m_PosArray[i]; min = math.min(min, pos.xy); max = math.max(max, pos.xy); } } { for (int i = 0; i < contourPointCount; ++i) { float2 pos = new float2(m_ContourPoints[i].position.x, m_ContourPoints[i].position.y); min = math.min(min, pos); max = math.max(max, pos); } } bounds.SetMinMax(new Vector3(min.x, min.y, 0), new Vector3(max.x, max.y, 0)); m_Bounds[0] = bounds; } void CalculateTexCoords() { SpriteShapeSegment geom = m_GeomArray[0]; if (m_ShapeParams.splineData.x > 0) { float3 ext = m_Bounds[0].extents * 2; float3 min = m_Bounds[0].center - m_Bounds[0].extents; for (int i = 0; i < geom.vertexCount; ++i) { Vector3 pos = m_PosArray[i]; Vector2 uv0 = m_Uv0Array[i]; float3 uv = ((new float3(pos.x, pos.y, pos.z) - min) / ext) * m_ShapeParams.fillData.x; uv0.x = uv.x; uv0.y = uv.y; m_Uv0Array[i] = uv0; } } else { for (int i = 0; i < geom.vertexCount; ++i) { Vector3 pos = m_PosArray[i]; Vector2 uv0 = m_Uv0Array[i]; float3 uv = math.transform(m_Transform, new float3(pos.x, pos.y, pos.z)); uv0.x = uv.x / m_ShapeParams.fillData.y; uv0.y = uv.y / m_ShapeParams.fillData.z; m_Uv0Array[i] = uv0; } } } void CopyVertexData(ref NativeSlice outPos, ref NativeSlice outUV0, ref NativeSlice outTan, int outIndex, ref Array inVertices, int inIndex, float sOrder) { Vector3 iscp = outPos[outIndex]; Vector2 iscu = outUV0[outIndex]; float3 v0 = new float3(inVertices[inIndex].pos.x, inVertices[inIndex].pos.y, sOrder); float3 v1 = new float3(inVertices[inIndex + 1].pos.x, inVertices[inIndex + 1].pos.y, sOrder); float3 v2 = new float3(inVertices[inIndex + 2].pos.x, inVertices[inIndex + 2].pos.y, sOrder); float3 v3 = new float3(inVertices[inIndex + 3].pos.x, inVertices[inIndex + 3].pos.y, sOrder); outPos[outIndex] = v0; outUV0[outIndex] = inVertices[inIndex].uv; outPos[outIndex + 1] = v1; outUV0[outIndex + 1] = inVertices[inIndex + 1].uv; outPos[outIndex + 2] = v2; outUV0[outIndex + 2] = inVertices[inIndex + 2].uv; outPos[outIndex + 3] = v3; outUV0[outIndex + 3] = inVertices[inIndex + 3].uv; if (outTan.Length > 1) { outTan[outIndex] = inVertices[inIndex].tan; outTan[outIndex + 1] = inVertices[inIndex + 1].tan; outTan[outIndex + 2] = inVertices[inIndex + 2].tan; outTan[outIndex + 3] = inVertices[inIndex + 3].tan; } } int CopySegmentRenderData(JobSpriteInfo ispr, ref NativeSlice outPos, ref NativeSlice outUV0, ref NativeSlice outTan, ref int outCount, ref NativeArray indexData, ref int indexCount, ref Array inVertices, int inCount, float sOrder) { if (inCount < 4) return -1; int localVertex = 0; int finalCount = indexCount + inCount + (inCount / 2); if (finalCount >= indexData.Length) { SetResult(SpriteShapeGeneratorResult.ErrorVertexLimitReached); return -1; } for (int i = 0; i < inCount; i = i + 4, outCount = outCount + 4, localVertex = localVertex + 4) { CopyVertexData(ref outPos, ref outUV0, ref outTan, outCount, ref inVertices, i, sOrder); indexData[indexCount++] = (ushort) (localVertex); indexData[indexCount++] = (ushort) (3 + localVertex); indexData[indexCount++] = (ushort) (1 + localVertex); indexData[indexCount++] = (ushort) (localVertex); indexData[indexCount++] = (ushort) (2 + localVertex); indexData[indexCount++] = (ushort) (3 + localVertex); } return outCount; } void GetLineSegments(JobSpriteInfo sprInfo, JobSegmentInfo segment, float2 whsize, ref float2 vlt, ref float2 vlb, ref float2 vrt, ref float2 vrb) { JobControlPoint scp = GetControlPoint(segment.sgInfo.x); JobControlPoint ecp = GetControlPoint(segment.sgInfo.y); GetSegmentBoundaryColumn(segment, sprInfo, whsize, scp.position, ecp.position, false, ref vlt, ref vlb); GetSegmentBoundaryColumn(segment, sprInfo, whsize, ecp.position, scp.position, true, ref vrt, ref vrb); } void TessellateSegment(int segmentIndex, JobSpriteInfo sprInfo, JobSegmentInfo segment, float2 whsize, float4 border, float pxlWidth, ref Array vertices, int vertexCount, bool useClosure, bool validHead, bool validTail, bool firstSegment, bool finalSegment, ref Array outputVertices, ref int outputCount) { int outputVertexCount = 0; float2 zero = float2.zero; float2 lt = zero, lb = zero, rt = zero, rb = zero; float4 stretcher = new float4(1.0f, 1.0f, 0, 0); var column0 = new JobShapeVertex(); var column1 = new JobShapeVertex(); var column2 = new JobShapeVertex(); var column3 = new JobShapeVertex(); int cms = vertexCount - 1; int lcm = cms - 1; int expectedCount = outputCount + (cms * 4); var sprite = vertices[0].sprite; if (expectedCount >= outputVertices.MaxSize) { SetResult(SpriteShapeGeneratorResult.ErrorVertexLimitReached); Debug.Log($"Mesh data has reached Limits. Please try dividing shape into smaller blocks."); return; } float uvDist = 0; float uvStart = border.x; float uvEnd = whsize.x - border.z; float uvTotal = whsize.x; float uvInter = uvEnd - uvStart; float uvNow = uvStart / uvTotal; float dt = uvInter / pxlWidth; float pivot = 0.5f - sprInfo.metaInfo.y; //// //// //// //// Stretch bool stretchCorners = false; bool stretchSegment = math.abs(segment.sgInfo.x - segment.sgInfo.y) == 1; if (stretchSegment && segmentCount > 1) stretchCorners = FetchStretcher(segmentIndex, sprInfo, segment, whsize, validHead, validTail, ref stretcher); //// //// //// //// Stretch // Generate Render Inputs. for (int i = 0; i < cms; ++i) { bool lc = (cms > 1) && (i == lcm); bool im = (i != 0 && !lc); JobShapeVertex cs = vertices[i]; JobShapeVertex ns = vertices[i + 1]; float2 es = lc ? cs.pos : vertices[i + 2].pos; lt = column1.pos; lb = column3.pos; if (im) { // Left from Previous. GenerateColumnsTri(cs.pos, ns.pos, es, whsize, lc, ref rt, ref rb, ns.meta.x * 0.5f, pivot); } else { if (!lc) { GetSegmentBoundaryColumn(segment, sprInfo, whsize, cs.pos, ns.pos, false, ref lt, ref lb); } if (lc && useClosure) { rb = m_FirstLB; rt = m_FirstLT; } else { GetSegmentBoundaryColumn(segment, sprInfo, whsize, ns.pos, es, lc, ref rt, ref rb); } } if (i == 0 && segment.sgInfo.x == 0) { m_FirstLB = lb; m_FirstLT = lt; } if (!((math.any(lt) || math.any(lb)) && (math.any(rt) || math.any(rb)))) continue; // default tan (1, 0, 0, -1) which is along uv. same here. float2 nlt = math.normalize(rt - lt); float4 tan = new float4(nlt.x, nlt.y, 0, -1.0f); column0.pos = lt; column0.meta = cs.meta; column0.sprite = sprite; column0.tan = tan; column1.pos = rt; column1.meta = ns.meta; column1.sprite = sprite; column1.tan = tan; column2.pos = lb; column2.meta = cs.meta; column2.sprite = sprite; column2.tan = tan; column3.pos = rb; column3.meta = ns.meta; column3.sprite = sprite; column3.tan = tan; // Calculate UV. if (validHead && i == 0) { column0.uv.x = column0.uv.y = column1.uv.y = column2.uv.x = 0; column1.uv.x = column3.uv.x = border.x / whsize.x; column2.uv.y = column3.uv.y = 1.0f; column0.sprite.z = column2.sprite.z = firstSegment ? 0 : 1; } else if (validTail && i == lcm) { column0.uv.y = column1.uv.y = 0; column0.uv.x = column2.uv.x = (whsize.x - border.z) / whsize.x; column1.uv.x = column2.uv.y = column3.uv.x = column3.uv.y = 1.0f; column1.sprite.z = column3.sprite.z = finalSegment ? 0 : 1; } else { if ((uvInter - uvDist) < kEpsilonRelaxed) { uvNow = uvStart / uvTotal; uvDist = 0; } uvDist = uvDist + (math.distance(ns.pos, cs.pos) * dt); float uvNext = (uvDist + uvStart) / uvTotal; if ((uvDist - uvInter) > kEpsilonRelaxed) { uvNext = uvEnd / uvTotal; uvDist = uvEnd; } column0.uv.y = column1.uv.y = 0; column0.uv.x = column2.uv.x = uvNow; column1.uv.x = column3.uv.x = uvNext; column2.uv.y = column3.uv.y = 1.0f; uvNow = uvNext; } { // Fix UV and Copy. column0.uv.x = (column0.uv.x * sprInfo.uvInfo.z) + sprInfo.uvInfo.x; column0.uv.y = (column0.uv.y * sprInfo.uvInfo.w) + sprInfo.uvInfo.y; outputVertices[outputVertexCount++] = column0; column1.uv.x = (column1.uv.x * sprInfo.uvInfo.z) + sprInfo.uvInfo.x; column1.uv.y = (column1.uv.y * sprInfo.uvInfo.w) + sprInfo.uvInfo.y; outputVertices[outputVertexCount++] = column1; column2.uv.x = (column2.uv.x * sprInfo.uvInfo.z) + sprInfo.uvInfo.x; column2.uv.y = (column2.uv.y * sprInfo.uvInfo.w) + sprInfo.uvInfo.y; outputVertices[outputVertexCount++] = column2; column3.uv.x = (column3.uv.x * sprInfo.uvInfo.z) + sprInfo.uvInfo.x; column3.uv.y = (column3.uv.y * sprInfo.uvInfo.w) + sprInfo.uvInfo.y; outputVertices[outputVertexCount++] = column3; } } //// //// //// //// Stretch if (stretchCorners) StretchCorners(segment, ref outputVertices, outputVertexCount, validHead, validTail, stretcher); //// //// //// //// Stretch outputCount = outputVertexCount; } bool SkipSegment(JobSegmentInfo isi) { // Start the Generation. bool skip = (isi.sgInfo.z < 0); if (!skip) { JobSpriteInfo ispr = GetSpriteInfo(isi.sgInfo.z); skip = (math.any(ispr.uvInfo) == false); } if (skip) { int cis = GetContourIndex(isi.sgInfo.x); int cie = GetEndContourIndexOfSegment(isi); while (cis < cie) { JobContourPoint icp = GetContourPoint(cis); m_ColliderPoints[m_ColliderDataCount++] = icp.position; cis++; } } return skip; } float InterpolateLinear(float a, float b, float t) { return math.lerp(a, b, t); } float InterpolateSmooth(float a, float b, float t) { float mu2 = (1.0f - math.cos(t * math.PI)) / 2.0f; return (a * (1 - mu2) + b * mu2); } void TessellateSegments() { JobControlPoint iscp = GetControlPoint(0); bool disableHead = (iscp.cpData.z == kModeContinous && isCarpet); float2 zero = new float2(0, 0); float2 ec = zero; var minArrayCount = (kControlPointCount > kMaxArrayCount) ? (kMaxArrayCount / 2) : kControlPointCount; var segVertexData = new Array(minArrayCount, kMaxArrayCount, Allocator.Temp, NativeArrayOptions.UninitializedMemory); var segOutputData = new Array(minArrayCount, kMaxArrayCount, Allocator.Temp, NativeArrayOptions.UninitializedMemory); for (int i = 0; i < segmentCount; ++i) { // Tessellate the Segment. JobSegmentInfo isi = GetSegmentInfo(i); bool skip = SkipSegment(isi); if (skip) continue; // Internal Data : x, y : pos z : height w : renderIndex JobShapeVertex isv = new JobShapeVertex(); JobSpriteInfo ispr = GetSpriteInfo(isi.sgInfo.z); int vertexCount = 0; int sprIx = isi.sgInfo.z; float rpunits = 1.0f / ispr.metaInfo.x; float2 whsize = new float2(ispr.metaInfo.z, ispr.metaInfo.w) * rpunits; float4 border = ispr.border * rpunits; JobControlPoint _scp = GetControlPoint(isi.sgInfo.x); JobControlPoint _ecp = GetControlPoint(isi.sgInfo.y); bool useClosure = (m_ControlPoints[0].cpData.z == kModeContinous) && (isi.sgInfo.y == controlPointCount - 1); bool firstSegment = (i == 0) && !isCarpet && !useClosure; bool validHead = hasSpriteBorder && (border.x > 0) && ((_scp.exData.z == 0) || firstSegment); validHead = (m_ControlPoints[0].cpData.z == kModeContinous) ? (validHead && !isCarpet) : validHead; bool finalSegment = (i == segmentCount - 1) && !isCarpet && !useClosure; bool validTail = hasSpriteBorder && (border.z > 0) && ((_ecp.exData.z == 0) || finalSegment); validTail = (m_ControlPoints[0].cpData.z == kModeContinous) ? (validTail && !isCarpet) : validTail; // Generate the UV Increments. float extendUV = 0; float stPixelU = border.x; float enPixelU = whsize.x - border.z; float pxlWidth = enPixelU - stPixelU; float segmentD = isi.spriteInfo.w; float uIncStep = math.floor(segmentD / pxlWidth); uIncStep = uIncStep == 0 ? 1f : uIncStep; pxlWidth = isAdaptive ? (segmentD / uIncStep) : pxlWidth; // Check for any invalid Sizes. if (pxlWidth < kEpsilon) { SetResult(SpriteShapeGeneratorResult.ErrorSpritesWrongBorder); Debug.Log($"One of the sprites seem to have Invalid Borders. Please check Input Sprites."); return; } // Start the Generation. int stIx = GetContourIndex(isi.sgInfo.x); int enIx = GetEndContourIndexOfSegment(isi); // Single Segment Loop. if (stIx == 0) validHead = (validHead && !disableHead); // Do we have a Sprite Head Slice if (validHead) { JobContourPoint icp = GetContourPoint(stIx); float2 v1 = icp.position; float2 v2 = GetContourPoint(stIx + 1).position; isv.pos = v1 + (math.normalize(v1 - v2) * border.x); isv.meta.x = icp.ptData.x; isv.sprite.x = sprIx; segVertexData[vertexCount++] = isv; } // Generate the Strip. float sl = 0; int it = stIx, nt = 0; isv.sprite.z = 0; while (it < enIx) { nt = it + 1; JobContourPoint icp = GetContourPoint(it); JobContourPoint ncp = GetContourPoint(nt); float2 sp = icp.position; float2 ip = sp; float2 ep = ncp.position; float2 df = ep - sp; float al = math.length(df); if (al > kEpsilon) { float sh = icp.ptData.x, eh = ncp.ptData.x, hl = 0; sl = sl + al; // Connect previously left out space when sl < pxlWidth var addtail = (0 == vertexCount); float2 step = math.normalize(df); isv.pos = icp.position; isv.meta.x = icp.ptData.x; isv.sprite.x = sprIx; if (vertexCount > 0) { var dt = math.length(segVertexData[vertexCount - 1].pos - isv.pos); addtail = dt > kEpsilonRelaxed; } if (addtail) segVertexData[vertexCount++] = isv; while (sl > pxlWidth) { float _uv = pxlWidth - extendUV; float2 uv = new float2(_uv); ip = sp + (step * uv); hl = hl + math.length(ip - sp); isv.pos = ip; isv.meta.x = InterpolateLinear(sh, eh, hl / al); isv.sprite.x = sprIx; if (math.any(segVertexData[vertexCount - 1].pos - isv.pos)) segVertexData[vertexCount++] = isv; sl = sl - pxlWidth; sp = ip; extendUV = 0; } extendUV = sl; } it++; } // The Remains from the above Loop. Finish the Curve. if (sl > kEpsilon) { JobContourPoint ecp = GetContourPoint(enIx); isv.pos = ecp.position; isv.meta.x = ecp.ptData.x; isv.sprite.x = sprIx; segVertexData[vertexCount++] = isv; } // Generate Tail if (validTail) { JobContourPoint icp = GetContourPoint(enIx); float2 v1 = icp.position; float2 v2 = GetContourPoint(enIx - 1).position; isv.pos = v1 + (math.normalize(v1 - v2) * border.z); isv.meta.x = icp.ptData.x; isv.sprite.x = sprIx; segVertexData[vertexCount++] = isv; } // Generate the Renderer Data. int outputCount = 0; TessellateSegment(i, ispr, isi, whsize, border, pxlWidth, ref segVertexData, vertexCount, useClosure, validHead, validTail, firstSegment, finalSegment, ref segOutputData, ref outputCount); if (outputCount == 0) continue; var z = ((float)(i + 1) * kEpsilonOrder) + ((float)isi.sgInfo.z * kEpsilonOrder * 0.001f); CopySegmentRenderData(ispr, ref m_PosArray, ref m_Uv0Array, ref m_TanArray, ref m_VertexDataCount, ref m_IndexArray, ref m_IndexDataCount, ref segOutputData, outputCount, z); if (hasCollider) { JobSpriteInfo isprc = (ispr.metaInfo.x == 0) ? GetSpriteInfo(isi.sgInfo.w) : ispr; outputCount = 0; rpunits = 1.0f / isprc.metaInfo.x; whsize = new float2(isprc.metaInfo.z, isprc.metaInfo.w) * rpunits; border = isprc.border * rpunits; stPixelU = border.x; enPixelU = whsize.x - border.z; pxlWidth = enPixelU - stPixelU; TessellateSegment(i, isprc, isi, whsize, border, pxlWidth, ref segVertexData, vertexCount, useClosure, validHead, validTail, firstSegment, finalSegment, ref segOutputData, ref outputCount); ec = UpdateCollider(isi, isprc, ref segOutputData, outputCount, ref m_ColliderPoints, ref m_ColliderDataCount); } // Geom Data var geom = m_GeomArray[i + 1]; geom.geomIndex = i + 1; geom.indexCount = m_IndexDataCount - m_ActiveIndexCount; geom.vertexCount = m_VertexDataCount - m_ActiveVertexCount; geom.spriteIndex = isi.sgInfo.z; m_GeomArray[i + 1] = geom; // Exit m_ActiveIndexCount = m_IndexDataCount; m_ActiveVertexCount = m_VertexDataCount; } segVertexData.Dispose(); segOutputData.Dispose(); // Copy Collider, Copy Render Data. m_GeomArrayCount = segmentCount + 1; m_IndexArrayCount = m_IndexDataCount; m_VertexArrayCount = m_VertexDataCount; m_ColliderPointCount = m_ColliderDataCount; } #endregion #region Stretch. bool FetchStretcher(int segmentIndex, JobSpriteInfo sprInfo, JobSegmentInfo segment, float2 whsize, bool validHead, bool validTail, ref float4 stretcher) { bool needsStretchL = false, needsStretchR = false; int lastSegmentIndex = segmentCount - 1; int prevSegmentIndex = segmentIndex == 0 ? lastSegmentIndex : segmentIndex - 1; int nextSegmentIndex = segmentIndex == lastSegmentIndex ? 0 : segmentIndex + 1; JobSegmentInfo prevSegment = GetSegmentInfo(prevSegmentIndex); JobSegmentInfo nextSegment = GetSegmentInfo(nextSegmentIndex); JobControlPoint scp = GetControlPoint(segment.sgInfo.x); JobControlPoint ecp = GetControlPoint(segment.sgInfo.y); var stretchLeft = (scp.cpData.y == 2) && math.abs(prevSegment.sgInfo.x - prevSegment.sgInfo.y) == 1; var stretchRight = (ecp.cpData.y == 2) && math.abs(nextSegment.sgInfo.x - nextSegment.sgInfo.y) == 1; var lastControlPoint = (controlPointCount - 1); if (!isCarpet) { stretchLeft = stretchLeft && segment.sgInfo.x != 0; stretchRight = stretchRight && segment.sgInfo.y != lastControlPoint; } if (stretchLeft || stretchRight) { // Get End points for current segment. float2 avlt = float2.zero, avlb = float2.zero, avrt = float2.zero, avrb = float2.zero; GetLineSegments(sprInfo, segment, whsize, ref avlt, ref avlb, ref avrt, ref avrb); float2 _avlt = avlt, _avlb = avlb, _avrt = avrt, _avrb = avrb; float2 ltp = avlt, lbt = avlb, rtp = avrt, rbt = avrb; ExtendSegment(ref avlt, ref avrt); ExtendSegment(ref avlb, ref avrb); // Check Neighbor Next if (stretchLeft) { if (math.any(m_Intersectors[segment.sgInfo.x].top) && math.any(m_Intersectors[segment.sgInfo.x].bottom)) { ltp = m_Intersectors[segment.sgInfo.x].top; lbt = m_Intersectors[segment.sgInfo.x].bottom; needsStretchL = true; } else { // Check end-points match for start and prev. if (1 == scp.exData.z) { // Intersection Test float2 pvlt = float2.zero, pvlb = float2.zero, pvrt = float2.zero, pvrb = float2.zero; GetLineSegments(sprInfo, prevSegment, whsize, ref pvlt, ref pvlb, ref pvrt, ref pvrb); ExtendSegment(ref pvlt, ref pvrt); ExtendSegment(ref pvlb, ref pvrb); bool _lt = LineIntersection(kEpsilon, pvlt, pvrt, avlt, avrt, ref ltp); bool _lb = LineIntersection(kEpsilon, pvlb, pvrb, avlb, avrb, ref lbt); needsStretchL = _lt && _lb; } if (needsStretchL) { JobIntersectPoint ip = m_Intersectors[segment.sgInfo.x]; ip.top = ltp; ip.bottom = lbt; m_Intersectors[segment.sgInfo.x] = ip; } } } // Check Neighbor Next if (stretchRight) { if (math.any(m_Intersectors[segment.sgInfo.y].top) && math.any(m_Intersectors[segment.sgInfo.y].bottom)) { rtp = m_Intersectors[segment.sgInfo.y].top; rbt = m_Intersectors[segment.sgInfo.y].bottom; needsStretchR = true; } else { // Check end-points match for end and next. if (1 == ecp.exData.z) { // Intersection Test float2 nvlt = float2.zero, nvlb = float2.zero, nvrt = float2.zero, nvrb = float2.zero; GetLineSegments(sprInfo, nextSegment, whsize, ref nvlt, ref nvlb, ref nvrt, ref nvrb); ExtendSegment(ref nvlt, ref nvrt); ExtendSegment(ref nvlb, ref nvrb); bool _rt = LineIntersection(kEpsilon, avlt, avrt, nvlt, nvrt, ref rtp); bool _rb = LineIntersection(kEpsilon, avlb, avrb, nvlb, nvrb, ref rbt); needsStretchR = _rt && _rb; } if (needsStretchR) { JobIntersectPoint ip = m_Intersectors[segment.sgInfo.y]; ip.top = rtp; ip.bottom = rbt; m_Intersectors[segment.sgInfo.y] = ip; } } } if (needsStretchL || needsStretchR) { float2 _lm = (_avlt + _avlb) * 0.5f; float2 _rm = (_avrt + _avrb) * 0.5f; float _m = math.length(_lm - _rm); float _t = math.length(ltp - rtp); float _b = math.length(lbt - rbt); stretcher.x = _t / _m; stretcher.y = _b / _m; stretcher.z = needsStretchL ? 1.0f : 0; stretcher.w = needsStretchR ? 1.0f : 0; } } return (needsStretchL || needsStretchR); } void StretchCorners(JobSegmentInfo segment, ref Array vertices, int vertexCount, bool validHead, bool validTail, float4 stretcher) { if (vertexCount > 0) { int lts = validHead ? 4 : 0; float2 lt = vertices[lts].pos, _lt = vertices[lts].pos; float2 rt = vertices[vertexCount - 3].pos, _rt = vertices[vertexCount - 3].pos; float2 lb = vertices[lts + 2].pos, _lb = vertices[lts + 2].pos; float2 rb = vertices[vertexCount - 1].pos, _rb = vertices[vertexCount - 1].pos; if (math.any(m_Intersectors[segment.sgInfo.x].top) && math.any(m_Intersectors[segment.sgInfo.x].bottom)) { lt = m_Intersectors[segment.sgInfo.x].top; lb = m_Intersectors[segment.sgInfo.x].bottom; } if (math.any(m_Intersectors[segment.sgInfo.y].top) && math.any(m_Intersectors[segment.sgInfo.y].bottom)) { rt = m_Intersectors[segment.sgInfo.y].top; rb = m_Intersectors[segment.sgInfo.y].bottom; } for (int i = lts; i < vertexCount; i = i + 4) { JobShapeVertex v0 = vertices[i + 0]; JobShapeVertex v1 = vertices[i + 1]; JobShapeVertex v2 = vertices[i + 2]; JobShapeVertex v3 = vertices[i + 3]; v0.pos = lt + ((vertices[i + 0].pos - _lt) * stretcher.x); v1.pos = lt + ((vertices[i + 1].pos - _lt) * stretcher.x); v2.pos = lb + ((vertices[i + 2].pos - _lb) * stretcher.y); v3.pos = lb + ((vertices[i + 3].pos - _lb) * stretcher.y); vertices[i + 0] = v0; vertices[i + 1] = v1; vertices[i + 2] = v2; vertices[i + 3] = v3; } JobShapeVertex vx = vertices[lts]; JobShapeVertex vy = vertices[lts + 2]; vx.pos = lt; vy.pos = lb; vertices[lts] = vx; vertices[lts + 2] = vy; JobShapeVertex vz = vertices[vertexCount - 3]; JobShapeVertex vw = vertices[vertexCount - 1]; vz.pos = rt; vw.pos = rb; vertices[vertexCount - 3] = vz; vertices[vertexCount - 1] = vw; } } #endregion #region Corners // Extend Segment. void ExtendSegment(ref float2 l0, ref float2 r0) { float2 _l0 = l0, _r0 = r0; float2 _x = math.normalize(_r0 - _l0); r0 = _r0 + (_x * kExtendSegment); l0 = _l0 + (-_x * kExtendSegment); } bool GetIntersection(int cp, int ct, JobSpriteInfo ispr, ref float2 lt0, ref float2 lb0, ref float2 rt0, ref float2 rb0, ref float2 lt1, ref float2 lb1, ref float2 rt1, ref float2 rb1, ref float2 tp, ref float2 bt) { // Correct Left. float2 zero = new float2(0, 0); int pp = (cp == 0) ? (controlPointCount - 1) : (cp - 1); int np = (cp + 1) % controlPointCount; float pivot = 0.5f - ispr.metaInfo.y; JobControlPoint lcp = GetControlPoint(pp); JobControlPoint ccp = GetControlPoint(cp); JobControlPoint rcp = GetControlPoint(np); float rpunits = 1.0f / ispr.metaInfo.x; float2 whsize = new float2(ispr.texRect.z, ispr.texRect.w) * rpunits; float4 border = ispr.border * rpunits; // Generate the UV Increments. float stPixelV = border.y; float enPixelV = whsize.y - border.y; float pxlWidth = enPixelV - stPixelV; // pxlWidth is the square size of the corner sprite. // Generate the LeftTop, LeftBottom, RightTop & RightBottom for both sides. GenerateColumnsBi(lcp.position, ccp.position, whsize, false, ref lb0, ref lt0, ccp.cpInfo.x * 0.5f, pivot); GenerateColumnsBi(ccp.position, lcp.position, whsize, false, ref rt0, ref rb0, ccp.cpInfo.x * 0.5f, pivot); GenerateColumnsBi(ccp.position, rcp.position, whsize, false, ref lb1, ref lt1, ccp.cpInfo.x * 0.5f, pivot); GenerateColumnsBi(rcp.position, ccp.position, whsize, false, ref rt1, ref rb1, ccp.cpInfo.x * 0.5f, pivot); rt0 = rt0 + (math.normalize(rt0 - lt0) * kExtendSegment); rb0 = rb0 + (math.normalize(rb0 - lb0) * kExtendSegment); lt1 = lt1 + (math.normalize(lt1 - rt1) * kExtendSegment); lb1 = lb1 + (math.normalize(lb1 - rb1) * kExtendSegment); // Generate Intersection of the Bottom Line Segments. bool t = LineIntersection(kEpsilon, lt0, rt0, lt1, rt1, ref tp); bool b = LineIntersection(kEpsilon, lb0, rb0, lb1, rb1, ref bt); if (!b && !t) return false; return true; } bool AttachCorner(int cp, int ct, JobSpriteInfo ispr, ref NativeArray newPoints, ref int activePoint) { // Correct Left. float2 zero = new float2(0, 0); float2 tp = zero, bt = zero; float2 lt0 = zero, lb0 = zero, rt0 = zero, rb0 = zero, lt1 = zero, lb1 = zero, rt1 = zero, rb1 = zero; float pivot = 0.5f - ispr.metaInfo.y; int pp = (cp == 0) ? (controlPointCount - 1) : (cp - 1); int np = (cp + 1) % controlPointCount; JobControlPoint lcp = GetControlPoint(pp); JobControlPoint ccp = GetControlPoint(cp); JobControlPoint rcp = GetControlPoint(np); float rpunits = 1.0f / ispr.metaInfo.x; float2 whsize = new float2(ispr.texRect.z, ispr.texRect.w) * rpunits; float4 border = ispr.border * rpunits; // Generate the UV Increments. float stPixelV = border.y; float enPixelV = whsize.y - border.y; float pxlWidth = enPixelV - stPixelV; // pxlWidth is the square size of the corner sprite. bool intersects = GetIntersection(cp, ct, ispr, ref lt0, ref lb0, ref rt0, ref rb0, ref lt1, ref lb1, ref rt1, ref rb1, ref tp, ref bt); if (!intersects) return false; float2 pt = ccp.position; float2 lt = lcp.position - pt; float2 rt = rcp.position - pt; float ld = math.length(lt); float rd = math.length(rt); if (ld < pxlWidth || rd < pxlWidth) return false; float lrd = 0, rrd = 0; float a = AngleBetweenVector(math.normalize(lcp.position - ccp.position), math.normalize(rcp.position - ccp.position)); if (a > 0) { lrd = ld - math.distance(lb0, bt); rrd = rd - math.distance(bt, rb1); } else { lrd = ld - math.distance(lt0, tp); rrd = rd - math.distance(tp, rt1); } float2 la = pt + (math.normalize(lt) * lrd); float2 ra = pt + (math.normalize(rt) * rrd); ccp.exData.x = ct; ccp.exData.z = 2; // Start ccp.position = la; newPoints[activePoint++] = ccp; ccp.exData.x = ct; ccp.exData.z = 3; // End ccp.position = ra; newPoints[activePoint++] = ccp; JobCornerInfo iscp = m_Corners[m_CornerCount]; if (a > 0) { iscp.bottom = bt; iscp.top = tp; GenerateColumnsBi(la, lcp.position, whsize, false, ref lt0, ref lb0, ccp.cpInfo.x * ispr.metaInfo.y, pivot); GenerateColumnsBi(ra, rcp.position, whsize, false, ref lt1, ref lb1, ccp.cpInfo.x * ispr.metaInfo.y, pivot); iscp.left = lt0; iscp.right = lb1; } else { iscp.bottom = tp; iscp.top = bt; GenerateColumnsBi(la, lcp.position, whsize, false, ref lt0, ref lb0, ccp.cpInfo.x * ispr.metaInfo.y, pivot); GenerateColumnsBi(ra, rcp.position, whsize, false, ref lt1, ref lb1, ccp.cpInfo.x * ispr.metaInfo.y, pivot); iscp.left = lb0; iscp.right = lt1; } iscp.cornerData.x = ct; iscp.cornerData.y = activePoint; m_Corners[m_CornerCount] = iscp; m_CornerCount++; return true; } float2 CornerTextureCoordinate(int cornerType, int index) { int cornerArrayIndex = (cornerType - 1) * 4; return m_CornerCoordinates[cornerArrayIndex + index]; } int CalculateCorner(int index, float angle, float2 lt, float2 rt) { float slope = SlopeAngle(lt); var slopePair0 = new float2(-135.0f, -35.0f); var cornerPair0 = new int2(kCornerTypeInnerTopLeft, kCornerTypeOuterBottomLeft); if ( slope > slopePair0.x && slope < slopePair0.y ) return (angle > 0) ? cornerPair0.x : cornerPair0.y; var slopePair1 = new float2(35.0f, 135.0f); var cornerPair1 = new int2(kCornerTypeInnerBottomRight, kCornerTypeOuterTopRight); if (slope > slopePair1.x && slope < slopePair1.y) return (angle > 0) ? cornerPair1.x : cornerPair1.y; var slopePair2 = new float2(-35.0f, 35.0f); var cornerPair2 = new int2(kCornerTypeInnerTopRight, kCornerTypeOuterTopLeft); if (slope > slopePair2.x && slope < slopePair2.y) return (angle > 0) ? cornerPair2.x : cornerPair2.y; var slopePair3 = new float2(-135.0f, 135.0f); var cornerPair3 = new int2(kCornerTypeInnerBottomLeft, kCornerTypeOuterBottomRight); if (slope > slopePair3.x && slope < slopePair3.y) return (angle > 0) ? cornerPair3.x : cornerPair3.y; return (angle > 0) ? kCornerTypeInnerBottomLeft : kCornerTypeOuterBottomRight; } bool InsertCorner(int index, ref NativeArray cpSpriteIndices, ref NativeArray newPoints, ref int activePoint, ref bool cornerConsidered) { int i = (index == 0) ? (controlPointCount - 1) : (index - 1); int k = (index + 1) % controlPointCount; // Check if we have valid Sprites. if (cpSpriteIndices[i].x >= spriteCount || cpSpriteIndices[index].x >= spriteCount) return false; // Check if they have been resolved. if (cpSpriteIndices[i].y == 0 || cpSpriteIndices[index].y == 0) return false; JobControlPoint pcp = GetControlPoint(i); JobControlPoint icp = GetControlPoint(index); JobControlPoint ncp = GetControlPoint(k); // Check if the Mode of control Point and previous neighbor is same. Also check if Corner Toggle is enabled. if (icp.cpData.y == 0 || pcp.cpData.z != kModeLinear || icp.cpData.z != kModeLinear || ncp.cpData.z != kModeLinear) return false; // Check if the Height of the Control Points match if (pcp.cpInfo.x != icp.cpInfo.x || icp.cpInfo.x != ncp.cpInfo.x) return false; JobSpriteInfo psi = GetSpriteInfo(cpSpriteIndices[i].x); JobSpriteInfo isi = GetSpriteInfo(cpSpriteIndices[index].x); // Check if the Sprites Pivot matches. Otherwise not allowed. // psi.uvInfo.w != isi.uvInfo.w && psi.metaInfo.y != 0.5f (no more height and pivot checks) if (psi.metaInfo.y != isi.metaInfo.y) return false; // Now perform expensive stuff like angles etc.. float2 idir = math.normalize(ncp.position - icp.position); float2 ndir = math.normalize(pcp.position - icp.position); float angle = AngleBetweenVector(idir, ndir); float angleAbs = math.abs(angle); cornerConsidered = AngleWithinRange(angleAbs, (90f - m_ShapeParams.curveData.z), (90f + m_ShapeParams.curveData.z)) || (m_ShapeParams.curveData.z == 90.0f); if (cornerConsidered && icp.cpData.y == 1) { float2 rdir = math.normalize(icp.position - pcp.position); int ct = CalculateCorner(index, angle, rdir, idir); // Check if we have a valid Sprite. if (ct > 0) { JobSpriteInfo cspr = GetCornerSpriteInfo(ct); return AttachCorner(index, ct, cspr, ref newPoints, ref activePoint); } } return false; } void TessellateCorners() { for (int corner = 1; corner <= kCornerTypeInnerBottomRight; ++corner) { JobSpriteInfo isi = GetCornerSpriteInfo(corner); if (isi.metaInfo.x == 0) continue; int ic = 0; int vc = 0; Vector3 pos = m_PosArray[ic]; Vector2 uv0 = m_Uv0Array[ic]; bool ccw = (corner <= kCornerTypeOuterBottomRight); int vArrayCount = m_VertexArrayCount; for (int i = 0; i < m_CornerCount; ++i) { JobCornerInfo isc = m_Corners[i]; if (isc.cornerData.x == corner) { // Vertices. pos.x = isc.top.x; pos.y = isc.top.y; uv0.x = (CornerTextureCoordinate(corner, 1).x * isi.uvInfo.z) + isi.uvInfo.x; uv0.y = (CornerTextureCoordinate(corner, 1).y * isi.uvInfo.w) + isi.uvInfo.y; m_PosArray[m_VertexArrayCount] = pos; m_Uv0Array[m_VertexArrayCount++] = uv0; pos.x = isc.right.x; pos.y = isc.right.y; uv0.x = (CornerTextureCoordinate(corner, 0).x * isi.uvInfo.z) + isi.uvInfo.x; uv0.y = (CornerTextureCoordinate(corner, 0).y * isi.uvInfo.w) + isi.uvInfo.y; m_PosArray[m_VertexArrayCount] = pos; m_Uv0Array[m_VertexArrayCount++] = uv0; pos.x = isc.left.x; pos.y = isc.left.y; uv0.x = (CornerTextureCoordinate(corner, 3).x * isi.uvInfo.z) + isi.uvInfo.x; uv0.y = (CornerTextureCoordinate(corner, 3).y * isi.uvInfo.w) + isi.uvInfo.y; m_PosArray[m_VertexArrayCount] = pos; m_Uv0Array[m_VertexArrayCount++] = uv0; pos.x = isc.bottom.x; pos.y = isc.bottom.y; uv0.x = (CornerTextureCoordinate(corner, 2).x * isi.uvInfo.z) + isi.uvInfo.x; uv0.y = (CornerTextureCoordinate(corner, 2).y * isi.uvInfo.w) + isi.uvInfo.y; m_PosArray[m_VertexArrayCount] = pos; m_Uv0Array[m_VertexArrayCount++] = uv0; // Indices. m_IndexArray[m_IndexArrayCount++] = (ushort)(vc + 0); m_IndexArray[m_IndexArrayCount++] = (ushort)(vc + (ccw ? 1 : 3)); m_IndexArray[m_IndexArrayCount++] = (ushort)(vc + (ccw ? 3 : 1)); m_IndexArray[m_IndexArrayCount++] = (ushort)(vc + 0); m_IndexArray[m_IndexArrayCount++] = (ushort)(vc + (ccw ? 3 : 2)); m_IndexArray[m_IndexArrayCount++] = (ushort)(vc + (ccw ? 2 : 3)); vc = vc + 4; ic = ic + 6; } } if (m_TanArray.Length > 1) { for (int i = vArrayCount; i < m_VertexArrayCount; ++i) m_TanArray[i] = new Vector4(1.0f, 0, 0, -1.0f); } // Geom Data if (ic > 0 && vc > 0) { var geom = m_GeomArray[m_GeomArrayCount]; geom.geomIndex = m_GeomArrayCount; geom.indexCount = ic; geom.vertexCount = vc; geom.spriteIndex = m_SpriteInfos.Length + (corner - 1); m_GeomArray[m_GeomArrayCount++] = geom; } } } #endregion #region Fast Optimizations bool AreCollinear(float2 a, float2 b, float2 c, float t) { float ax = (a.y - b.y) * (a.x - c.x); float bx = (a.y - c.y) * (a.x - b.x); float aa = math.abs(ax - bx); return aa < t; } // Check if points are co linear and reduce. void OptimizePoints(float tolerance, bool tess, ref NativeArray pointSet, ref int pointCount) { int kMinimumPointsRequired = 8; if (pointCount < kMinimumPointsRequired) return; var tmpPoints = new NativeArray(pointCount + 8, Allocator.Temp, NativeArrayOptions.UninitializedMemory); int optimizedColliderPointCount = 0; int endColliderPointCount = pointCount - 2; bool val = true; var fst = pointSet[0]; tmpPoints[0] = fst; for (int i = 0; i < endColliderPointCount; ++i) { float2 v0 = pointSet[i]; float2 v1 = pointSet[i + 1]; float2 v2 = pointSet[i + 2]; do { val = AreCollinear(v0, v1, v2, tolerance); if (false == val) { tmpPoints[++optimizedColliderPointCount] = v1; break; } i++; v1 = pointSet[i + 1]; v2 = pointSet[i + 2]; } while (val && i < endColliderPointCount); } // Test for the last 2 points. (N - 2) (N - 1) (N) var lb2 = tmpPoints[optimizedColliderPointCount]; var lb1 = pointSet[endColliderPointCount]; var lst = pointSet[endColliderPointCount + 1]; val = AreCollinear(lb2, lb1, lst, tolerance); if (!val) tmpPoints[++optimizedColliderPointCount] = lb1; if (isCarpet) { if (tess || optimizedColliderPointCount > 2) { val = AreCollinear(tmpPoints[optimizedColliderPointCount], lst, fst, tolerance); if (!val) tmpPoints[++optimizedColliderPointCount] = lst; } tmpPoints[++optimizedColliderPointCount] = fst; } else tmpPoints[++optimizedColliderPointCount] = lst; pointCount = optimizedColliderPointCount + 1; UnityEngine.U2D.Common.UTess.ModuleHandle.Copy(tmpPoints, pointSet, pointCount); tmpPoints.Dispose(); } #endregion #region Collider Specific. void AttachCornerToCollider(JobSegmentInfo isi, float pivot, ref NativeArray colliderPoints, ref int colliderPointCount) { float2 zero = new float2(0, 0); int cornerIndex = isi.sgInfo.x + 1; for (int i = 0; i < m_CornerCount; ++i) { JobCornerInfo isc = m_Corners[i]; if (cornerIndex == isc.cornerData.y) { float2 cp = zero; float2 v0 = zero; if (isc.cornerData.x > kCornerTypeOuterBottomRight) v0 = isc.top; else v0 = isc.bottom; float2 v2 = zero; if (isc.cornerData.x > kCornerTypeOuterBottomRight) v2 = isc.bottom; else v2 = isc.top; cp = (v0 - v2) * pivot; cp = (v2 + cp + v0 + cp) * 0.5f; colliderPoints[colliderPointCount++] = cp; break; } } } float2 UpdateCollider(JobSegmentInfo isi, JobSpriteInfo ispr, ref Array vertices, int count, ref NativeArray colliderPoints, ref int colliderPointCount) { float2 zero = new float2(0, 0); float pivot = 0; // 0.5f - ispr.metaInfo.y; // Follow processed geometry and only use ColliderPivot. pivot = pivot + colliderPivot; AttachCornerToCollider(isi, pivot, ref colliderPoints, ref colliderPointCount); float2 cp = zero; float2 v0 = zero; float2 v2 = zero; for (int k = 0; k < count; k = k + 4) { v0 = vertices[k].pos; v2 = vertices[k + 2].pos; cp = (v0 - v2) * pivot; if (vertices[k].sprite.z == 0) colliderPoints[colliderPointCount++] = (v2 + cp + v0 + cp) * 0.5f; } float2 v1 = vertices[count - 1].pos; float2 v3 = vertices[count - 3].pos; cp = (v3 - v1) * pivot; if (vertices[count - 1].sprite.z == 0) colliderPoints[colliderPointCount++] = (v1 + cp + v3 + cp) * 0.5f; return cp; } void TrimOverlaps(int cpCount) { int kMinimumPointTolerance = 4; if (m_ColliderPointCount < kMinimumPointTolerance) return; var tmpPoints = new NativeArray(m_ColliderPointCount, Allocator.Temp, NativeArrayOptions.UninitializedMemory); int trimmedPointCount = 0; int i = 0; int kColliderPointCountClamped = m_ColliderPointCount / 2; int kSplineDetailClamped = math.clamp(splineDetail * 3, 0, 8); int kNeighbors = kSplineDetailClamped > kColliderPointCountClamped ? kColliderPointCountClamped : kSplineDetailClamped; kNeighbors = (kNeighbors > cpCount) ? cpCount : kNeighbors; int testOverlapCount = m_ColliderPointCount; if (!isCarpet) { tmpPoints[trimmedPointCount++] = m_ColliderPoints[0]; testOverlapCount = m_ColliderPointCount - 1; } while (i < testOverlapCount) { int h = (i > 0) ? (i - 1) : (m_ColliderPointCount - 1); bool noIntersection = true; float2 v0 = m_ColliderPoints[h]; float2 v1 = m_ColliderPoints[i]; for (int n = kNeighbors; n > 1; --n) { int j = (i + n - 1) % m_ColliderPointCount; int k = (i + n) % m_ColliderPointCount; if (k == 0 || i == 0) continue; float2 v2 = m_ColliderPoints[j]; float2 v3 = m_ColliderPoints[k]; float2 vx = v0 - v3; if (math.abs(math.length(vx)) < kEpsilon) break; float2 vi = v0; bool overLaps = LineIntersection(kEpsilonRelaxed, v0, v1, v2, v3, ref vi); if (overLaps && IsPointOnLines(kEpsilonRelaxed, v0, v1, v2, v3, vi)) { noIntersection = false; tmpPoints[trimmedPointCount++] = vi; i = i + n; break; } } if (noIntersection) { if (0 != i || isCarpet) tmpPoints[trimmedPointCount++] = v1; i = i + 1; } } for (; i < m_ColliderPointCount; ++i) tmpPoints[trimmedPointCount++] = m_ColliderPoints[i]; i = 0; m_ColliderPoints[i++] = tmpPoints[0]; float2 prev = tmpPoints[0]; for (int j = 1; j < trimmedPointCount; ++j) { float dist = math.length(tmpPoints[j] - prev); if (dist > kEpsilon) m_ColliderPoints[i++] = tmpPoints[j]; prev = tmpPoints[j]; } trimmedPointCount = i; if (trimmedPointCount > 3 && isCarpet) { // Check intersection of first line Segment and last. float2 vin = m_ColliderPoints[0]; bool endOverLaps = LineIntersection(kEpsilonRelaxed, m_ColliderPoints[0], m_ColliderPoints[1], m_ColliderPoints[trimmedPointCount - 1], m_ColliderPoints[trimmedPointCount - 2], ref vin); if (endOverLaps) m_ColliderPoints[0] = m_ColliderPoints[trimmedPointCount - 1] = vin; } tmpPoints.Dispose(); m_ColliderPointCount = trimmedPointCount; } void OptimizeCollider() { if (hasCollider) { if (kColliderQuality > 0) { OptimizePoints(kColliderQuality, false, ref m_ColliderPoints, ref m_ColliderPointCount); TrimOverlaps(m_ControlPointCount - 1); m_ColliderPoints[m_ColliderPointCount++] = new float2(0, 0); m_ColliderPoints[m_ColliderPointCount++] = new float2(0, 0); } // If the resulting Colliders don't have enough points including the last 2 'end-points', just use Contours as Colliders. var minimumPointCount = isCarpet ? 5 : 3; if (m_ColliderPointCount <= minimumPointCount) { for (int i = 0; i < m_TessPointCount; ++i) m_ColliderPoints[i] = m_TessPoints[i]; m_ColliderPoints[m_TessPointCount] = new float2(0, 0); m_ColliderPoints[m_TessPointCount + 1] = new float2(0, 0); m_ColliderPointCount = m_TessPointCount + 2; } } } #endregion #region Entry, Exit Points. [Obsolete] public void Prepare(UnityEngine.U2D.SpriteShapeController controller, SpriteShapeParameters shapeParams, int maxArrayCount, NativeArray shapePoints, NativeArray metaData, AngleRangeInfo[] angleRanges, Sprite[] segmentSprites, Sprite[] cornerSprites) { // Prepare Inputs. PrepareInput(shapeParams, maxArrayCount, shapePoints, controller.optimizeGeometry, controller.autoUpdateCollider || controller.forceColliderShapeUpdate, controller.optimizeCollider, controller.colliderOffset, controller.colliderDetail); PrepareSprites(segmentSprites, cornerSprites); PrepareAngleRanges(angleRanges); NativeArray newMetaData = new NativeArray(metaData.Length, Allocator.Temp); for (int i = 0; i < metaData.Length; ++i) { SplinePointMetaData newData = new SplinePointMetaData(); newData.height = metaData[i].height; newData.spriteIndex = metaData[i].spriteIndex; newData.cornerMode = metaData[i].corner ? (int)Corner.Automatic : (int)Corner.Disable; newMetaData[i] = newData; } PrepareControlPoints(shapePoints, newMetaData); newMetaData.Dispose(); // Generate Fill. Obsolete API and let's stick with main-thread fill. kModeUTess = 0; TessellateContourMainThread(); } internal void Prepare(UnityEngine.U2D.SpriteShapeController controller, SpriteShapeParameters shapeParams, int maxArrayCount, NativeArray shapePoints, NativeArray metaData, AngleRangeInfo[] angleRanges, Sprite[] segmentSprites, Sprite[] cornerSprites, bool UseUTess) { // Prepare Inputs. SetResult(SpriteShapeGeneratorResult.Success); PrepareInput(shapeParams, maxArrayCount, shapePoints, controller.optimizeGeometry, controller.autoUpdateCollider || controller.forceColliderShapeUpdate, controller.optimizeCollider, controller.colliderOffset, controller.colliderDetail); PrepareSprites(segmentSprites, cornerSprites); PrepareAngleRanges(angleRanges); PrepareControlPoints(shapePoints, metaData); // Generate Fill. kModeUTess = UseUTess ? 1 : 0; if (0 == kModeUTess) TessellateContourMainThread(); } public void Execute() { generateGeometry.Begin(); { if (0 != kModeUTess) TessellateContour(Allocator.Temp); GenerateSegments(); UpdateSegments(); TessellateSegments(); TessellateCorners(); CalculateTexCoords(); } generateGeometry.End(); generateCollider.Begin(); { CalculateBoundingBox(); OptimizeCollider(); } generateCollider.End(); } // Only needed if Burst is disabled. // [BurstDiscard] public void Cleanup() { SafeDispose(m_Corners); SafeDispose(m_CornerSpriteInfos); SafeDispose(m_SpriteInfos); SafeDispose(m_AngleRanges); SafeDispose(m_Segments); SafeDispose(m_ControlPoints); SafeDispose(m_ContourPoints); SafeDispose(m_GeneratedControlPoints); SafeDispose(m_SpriteIndices); SafeDispose(m_Intersectors); SafeDispose(m_TessPoints); SafeDispose(m_CornerCoordinates); } #endregion } };