using System.Collections.Generic; using Unity.Jobs; using Unity.Collections; using Unity.Mathematics; using Unity.Collections.LowLevel.Unsafe; using Unity.Profiling; #if UNITY_EDITOR using UnityEditor.U2D; #endif namespace UnityEngine.U2D { /// /// SpriteShapeController component contains Spline and SpriteShape Profile information that is used when generating SpriteShape geometry. /// [ExecuteInEditMode] [RequireComponent(typeof(SpriteShapeRenderer))] [DisallowMultipleComponent] [HelpURLAttribute("https://docs.unity3d.com/Packages/com.unity.2d.spriteshape@latest/index.html?subfolder=/manual/SSController.html")] public class SpriteShapeController : MonoBehaviour { // Internal Dataset. const float s_DistanceTolerance = 0.001f; // Cached Objects. SpriteShape m_ActiveSpriteShape; EdgeCollider2D m_EdgeCollider2D; PolygonCollider2D m_PolygonCollider2D; SpriteShapeRenderer m_SpriteShapeRenderer; SpriteShapeGeometryCache m_SpriteShapeGeometryCache; Sprite[] m_SpriteArray = new Sprite[0]; Sprite[] m_EdgeSpriteArray = new Sprite[0]; Sprite[] m_CornerSpriteArray = new Sprite[0]; AngleRangeInfo[] m_AngleRangeInfoArray = new AngleRangeInfo[0]; // Required for Generation. NativeArray m_ColliderData; NativeArray m_TangentData; NativeArray m_Statistics; // Renderer Stuff. bool m_DynamicOcclusionLocal; bool m_DynamicOcclusionOverriden; // Added so that shadows will work correctly when no collider is assigned bool m_ForceColliderShapeUpdate = false; // Hash Check. int m_ActiveSplineHash = 0; int m_ActiveSpriteShapeHash = 0; int m_MaxArrayCount = 0; JobHandle m_JobHandle; SpriteShapeParameters m_ActiveShapeParameters; // Serialized Data. [SerializeField] Spline m_Spline = new Spline(); [SerializeField] SpriteShape m_SpriteShape; [SerializeField] float m_FillPixelPerUnit = 100.0f; [SerializeField] float m_StretchTiling = 1.0f; [SerializeField] int m_SplineDetail; [SerializeField] bool m_AdaptiveUV; [SerializeField] bool m_StretchUV; [SerializeField] bool m_WorldSpaceUV; [SerializeField] float m_CornerAngleThreshold = 30.0f; [SerializeField] int m_ColliderDetail; [SerializeField, Range(-0.5f, 0.5f)] float m_ColliderOffset; [SerializeField] bool m_UpdateCollider = true; [SerializeField] bool m_EnableTangents = false; [SerializeField] [HideInInspector] bool m_GeometryCached = false; [SerializeField] bool m_UTess2D = true; [SerializeField] SpriteShapeGeometryCreator m_Creator; [SerializeField] List m_Modifiers = new List(); [SerializeField] List m_ColliderSegment = new List(); internal static readonly ProfilerMarker generateGeometry = new ProfilerMarker("SpriteShape.GenerateGeometry"); internal static readonly ProfilerMarker generateCollider = new ProfilerMarker("SpriteShape.GenerateCollider"); #region GetSet internal int maxArrayCount { get { return m_MaxArrayCount; } set { m_MaxArrayCount = value; } } internal bool geometryCached { get { return m_GeometryCached; } set { m_GeometryCached = value; } } internal int splineHashCode { get { return m_ActiveSplineHash; } } internal Sprite[] spriteArray { get { return m_SpriteArray; } } internal SpriteShapeParameters spriteShapeParameters { get { return m_ActiveShapeParameters; } } internal SpriteShapeGeometryCache spriteShapeGeometryCache { get { if (!m_SpriteShapeGeometryCache) { bool b = TryGetComponent(typeof(SpriteShapeGeometryCache), out Component comp); m_SpriteShapeGeometryCache = b ? (comp as SpriteShapeGeometryCache) : null; } return m_SpriteShapeGeometryCache; } } internal Sprite[] cornerSpriteArray { get { return m_CornerSpriteArray; } } internal Sprite[] edgeSpriteArray { get { return m_EdgeSpriteArray; } } /// Angle Ranges public AngleRangeInfo[] angleRangeInfoArray { get { return m_AngleRangeInfoArray; } } /// Get/Set SpriteShape Geometry Creator. public SpriteShapeGeometryCreator spriteShapeCreator { get { if (m_Creator == null) m_Creator = SpriteShapeDefaultCreator.defaultInstance; return m_Creator; } set { if (value != null) m_Creator = value; } } /// Get a list of Modifiers. public List modifiers { get { return m_Modifiers; } } /// Hash code for SpriteShape used to check for changes. public int spriteShapeHashCode { get { return m_ActiveSpriteShapeHash; } } /// Defines whether UV for fill geometry uses local or global space. public bool worldSpaceUVs { get { return m_WorldSpaceUV; } set { m_WorldSpaceUV = value; } } /// Defines pixel per unit for fill geometry UV generation. public float fillPixelsPerUnit { get { return m_FillPixelPerUnit; } set { m_FillPixelPerUnit = value; } } /// Enable tangent channel when generating SpriteShape geometry (used in Shaders) public bool enableTangents { get { return m_EnableTangents; } set { m_EnableTangents = value; } } /// Stretch tiling for inner fill geometry UV generation. public float stretchTiling { get { return m_StretchTiling; } set { m_StretchTiling = value; } } /// Level of detail for generated geometry. public int splineDetail { get { return m_SplineDetail; } set { m_SplineDetail = Mathf.Max(0, value); } } /// Level of detail for geometry generated for colliders. public int colliderDetail { get { return m_ColliderDetail; } set { m_ColliderDetail = Mathf.Max(0, value); } } /// Offset for colliders. public float colliderOffset { get { return m_ColliderOffset; } set { m_ColliderOffset = value; } } /// Angle threshold within which corners are enabled. public float cornerAngleThreshold { get { return m_CornerAngleThreshold; } set { m_CornerAngleThreshold = value; } } /// Auto update colliders on any change to SpriteShape geometry. public bool autoUpdateCollider { get { return m_UpdateCollider; } set { m_UpdateCollider = value; } } /// Optimize generated collider geometry. public bool optimizeCollider { get { return true; } } /// Optimize generated SpriteShape geometry. public bool optimizeGeometry { get { return true; } } /// Does this SpriteShapeController object has colliders ? public bool hasCollider { get { return (edgeCollider != null) || (polygonCollider != null); } } /// Spline object that has data to create the Bezier curve of this SpriteShape Controller. public Spline spline { get { return m_Spline; } } /// SpriteShape Profile asset that contains information on how to generate/render SpriteShapes. public SpriteShape spriteShape { get { return m_SpriteShape; } set { m_SpriteShape = value; } } /// EdgeCollider2D component attached to this Object. public EdgeCollider2D edgeCollider { get { if (!m_EdgeCollider2D) { bool b = TryGetComponent(typeof(EdgeCollider2D), out Component comp); m_EdgeCollider2D = b ? (comp as EdgeCollider2D) : null; } return m_EdgeCollider2D; } } /// PolygonCollider2D component attached to this Object. public PolygonCollider2D polygonCollider { get { if (!m_PolygonCollider2D) { bool b = TryGetComponent(typeof(PolygonCollider2D), out Component comp); m_PolygonCollider2D = b ? (comp as PolygonCollider2D) : null; } return m_PolygonCollider2D; } } /// SpriteShapeRenderer component of this Object. public SpriteShapeRenderer spriteShapeRenderer { get { if (!m_SpriteShapeRenderer) m_SpriteShapeRenderer = GetComponent(); return m_SpriteShapeRenderer; } } internal bool forceColliderShapeUpdate { get { return m_ForceColliderShapeUpdate; } } internal NativeArray stats { get { if (!m_Statistics.IsCreated) m_Statistics = new NativeArray(1, Allocator.Persistent); return m_Statistics; } } #endregion #region EventHandles. void DisposeInternal() { m_JobHandle.Complete(); if (m_ColliderData.IsCreated) m_ColliderData.Dispose(); if (m_TangentData.IsCreated) m_TangentData.Dispose(); if (m_Statistics.IsCreated) m_Statistics.Dispose(); } void OnApplicationQuit() { DisposeInternal(); } void OnEnable() { m_DynamicOcclusionOverriden = true; m_DynamicOcclusionLocal = spriteShapeRenderer.allowOcclusionWhenDynamic; spriteShapeRenderer.allowOcclusionWhenDynamic = false; InitBounds(); UpdateSpriteData(); } void OnDisable() { UpdateGeometryCache(); DisposeInternal(); } void OnDestroy() { } void Reset() { m_SplineDetail = (int)QualityDetail.High; m_AdaptiveUV = true; m_StretchUV = false; m_FillPixelPerUnit = 100f; m_ColliderDetail = (int)QualityDetail.High; m_StretchTiling = 1.0f; m_WorldSpaceUV = false; m_CornerAngleThreshold = 30.0f; m_ColliderOffset = 0; m_UpdateCollider = true; m_EnableTangents = false; spline.Clear(); spline.InsertPointAt(0, Vector2.left + Vector2.down); spline.InsertPointAt(1, Vector2.left + Vector2.up); spline.InsertPointAt(2, Vector2.right + Vector2.up); spline.InsertPointAt(3, Vector2.right + Vector2.down); } static void SmartDestroy(UnityEngine.Object o) { if (o == null) return; #if UNITY_EDITOR if (!Application.isPlaying) DestroyImmediate(o); else #endif Destroy(o); } #endregion #region HashAndDataCheck internal Bounds InitBounds() { var pointCount = spline.GetPointCount(); if (pointCount > 1) { Bounds bounds = new Bounds(spline.GetPosition(0), Vector3.zero); for (int i = 1; i < pointCount; ++i) bounds.Encapsulate(spline.GetPosition(i)); bounds.Encapsulate(spriteShapeRenderer.localBounds); spriteShapeRenderer.SetLocalAABB(bounds); return bounds; } return new Bounds(); } /// /// Refresh SpriteShape Hash so its force generated again on the next frame if its visible. /// public void RefreshSpriteShape() { m_ActiveSplineHash = 0; } // Ensure Neighbor points are not too close to each other. bool ValidateSpline() { int pointCount = spline.GetPointCount(); if (pointCount < 2) return false; for (int i = 0; i < pointCount - 1; ++i) { var vec = spline.GetPosition(i) - spline.GetPosition(i + 1); if (vec.sqrMagnitude < s_DistanceTolerance) { Debug.LogWarningFormat(gameObject, "[SpriteShape] Control points {0} & {1} are too close. SpriteShape will not be generated for < {2} >.", i, i + 1, gameObject.name); return false; } } return true; } // Ensure SpriteShape is valid if not bool ValidateSpriteShapeTexture() { bool valid = false; // Check if SpriteShape Profile is valid. if (spriteShape != null) { // Open ended and no valid Sprites set. Check if it has a valid fill texture. if (!spline.isOpenEnded) { valid = (spriteShape.fillTexture != null); } } else { // Warn that no SpriteShape is set. Debug.LogWarningFormat(gameObject, "[SpriteShape] A valid SpriteShape profile has not been set for gameObject < {0} >.", gameObject.name); #if UNITY_EDITOR // We allow null SpriteShape for rapid prototyping in Editor. valid = true; #endif } return valid; } internal bool ValidateUTess2D() { bool uTess2D = m_UTess2D; // Check for all properties that can create overlaps/intersections. if (m_UTess2D && null != spriteShape) { uTess2D = (spriteShape.fillOffset == 0); } return uTess2D; } bool HasSpriteShapeChanged() { bool changed = (m_ActiveSpriteShape != spriteShape); if (changed) m_ActiveSpriteShape = spriteShape; return changed; } bool HasSpriteShapeDataChanged() { bool updateSprites = HasSpriteShapeChanged(); if (spriteShape) { var hashCode = SpriteShape.GetSpriteShapeHashCode(spriteShape); if (spriteShapeHashCode != hashCode) { m_ActiveSpriteShapeHash = hashCode; updateSprites = true; } } return updateSprites; } int GetCustomScriptHashCode() { int hashCode = 0; unchecked { hashCode = (int)2166136261 ^ spriteShapeCreator.GetVersion(); foreach (var mod in m_Modifiers) if (null != mod) hashCode = hashCode * 16777619 ^ mod.GetVersion(); } return hashCode; } bool HasSplineDataChanged() { unchecked { // Spline. int hashCode = (int)2166136261 ^ spline.GetHashCode(); // Local Stuff. hashCode = hashCode * 16777619 ^ (m_UTess2D ? 1 : 0); hashCode = hashCode * 16777619 ^ (m_WorldSpaceUV ? 1 : 0); hashCode = hashCode * 16777619 ^ (m_EnableTangents ? 1 : 0); hashCode = hashCode * 16777619 ^ (m_GeometryCached ? 1 : 0); hashCode = hashCode * 16777619 ^ (m_StretchTiling.GetHashCode()); hashCode = hashCode * 16777619 ^ (m_ColliderOffset.GetHashCode()); hashCode = hashCode * 16777619 ^ (m_ColliderDetail.GetHashCode()); hashCode = hashCode * 16777619 ^ (GetCustomScriptHashCode()); hashCode = hashCode * 16777619 ^ (edgeCollider == null ? 0 : 1); hashCode = hashCode * 16777619 ^ (polygonCollider == null ? 0 : 1); if (splineHashCode != hashCode) { m_ActiveSplineHash = hashCode; return true; } } return false; } void LateUpdate() { BakeCollider(); } void OnWillRenderObject() { BakeMesh(); } /// /// Generate geometry on a Job. /// /// JobHandle for the SpriteShape geometry generation job. public JobHandle BakeMesh() { JobHandle jobHandle = default; #if !UNITY_EDITOR if (spriteShapeGeometryCache) { // If SpriteShapeGeometry has already been uploaded, don't bother checking further. if (0 != m_ActiveSplineHash && 0 != spriteShapeGeometryCache.maxArrayCount) return jobHandle; } #endif bool valid = ValidateSpline(); if (valid) { bool splineChanged = HasSplineDataChanged(); bool spriteShapeChanged = HasSpriteShapeDataChanged(); bool spriteShapeParametersChanged = UpdateSpriteShapeParameters(); if (splineChanged || spriteShapeChanged || spriteShapeParametersChanged) { if (spriteShapeChanged) { UpdateSpriteData(); } jobHandle = ScheduleBake(); #if UNITY_EDITOR UpdateGeometryCache(); #endif } } return jobHandle; } #endregion #region UpdateData /// /// Update Cache. /// internal void UpdateGeometryCache() { if (spriteShapeGeometryCache && geometryCached) { m_JobHandle.Complete(); spriteShapeGeometryCache.UpdateGeometryCache(); } } /// /// Force update SpriteShape parameters. /// /// Returns true if there are changes public bool UpdateSpriteShapeParameters() { bool carpet = !spline.isOpenEnded; bool smartSprite = true; bool adaptiveUV = m_AdaptiveUV; bool stretchUV = m_StretchUV; bool spriteBorders = false; uint fillScale = 0; uint splineDetail = (uint)m_SplineDetail; float borderPivot = 0f; float angleThreshold = (m_CornerAngleThreshold >= 0 && m_CornerAngleThreshold < 90) ? m_CornerAngleThreshold : 89.9999f; Texture2D fillTexture = null; Matrix4x4 transformMatrix = Matrix4x4.identity; if (spriteShape) { if (worldSpaceUVs) transformMatrix = transform.localToWorldMatrix; fillTexture = spriteShape.fillTexture; fillScale = stretchUV ? (uint)stretchTiling : (uint)fillPixelsPerUnit; borderPivot = spriteShape.fillOffset; spriteBorders = spriteShape.useSpriteBorders; // If Corners are enabled, set smart-sprite to false. if (spriteShape.cornerSprites.Count > 0) smartSprite = false; } else { #if UNITY_EDITOR fillTexture = UnityEditor.EditorGUIUtility.whiteTexture; fillScale = 100; #endif } bool changed = m_ActiveShapeParameters.adaptiveUV != adaptiveUV || m_ActiveShapeParameters.angleThreshold != angleThreshold || m_ActiveShapeParameters.borderPivot != borderPivot || m_ActiveShapeParameters.carpet != carpet || m_ActiveShapeParameters.fillScale != fillScale || m_ActiveShapeParameters.fillTexture != fillTexture || m_ActiveShapeParameters.smartSprite != smartSprite || m_ActiveShapeParameters.splineDetail != splineDetail || m_ActiveShapeParameters.spriteBorders != spriteBorders || m_ActiveShapeParameters.transform != transformMatrix || m_ActiveShapeParameters.stretchUV != stretchUV; m_ActiveShapeParameters.adaptiveUV = adaptiveUV; m_ActiveShapeParameters.stretchUV = stretchUV; m_ActiveShapeParameters.angleThreshold = angleThreshold; m_ActiveShapeParameters.borderPivot = borderPivot; m_ActiveShapeParameters.carpet = carpet; m_ActiveShapeParameters.fillScale = fillScale; m_ActiveShapeParameters.fillTexture = fillTexture; m_ActiveShapeParameters.smartSprite = smartSprite; m_ActiveShapeParameters.splineDetail = splineDetail; m_ActiveShapeParameters.spriteBorders = spriteBorders; m_ActiveShapeParameters.transform = transformMatrix; return changed; } void UpdateSpriteData() { if (spriteShape) { List edgeSpriteList = new List(); List cornerSpriteList = new List(); List angleRangeInfoList = new List(); List sortedAngleRanges = new List(spriteShape.angleRanges); sortedAngleRanges.Sort((a, b) => a.order.CompareTo(b.order)); for (int i = 0; i < sortedAngleRanges.Count; i++) { bool validSpritesFound = false; AngleRange angleRange = sortedAngleRanges[i]; foreach (Sprite edgeSprite in angleRange.sprites) { if (edgeSprite != null) { validSpritesFound = true; break; } } if (validSpritesFound) { AngleRangeInfo angleRangeInfo = new AngleRangeInfo(); angleRangeInfo.start = angleRange.start; angleRangeInfo.end = angleRange.end; angleRangeInfo.order = (uint)angleRange.order; List spriteIndices = new List(); foreach (Sprite edgeSprite in angleRange.sprites) { edgeSpriteList.Add(edgeSprite); spriteIndices.Add(edgeSpriteList.Count - 1); } angleRangeInfo.sprites = spriteIndices.ToArray(); angleRangeInfoList.Add(angleRangeInfo); } } bool validCornerSpritesFound = false; foreach (CornerSprite cornerSprite in spriteShape.cornerSprites) { if (cornerSprite.sprites[0] != null) { validCornerSpritesFound = true; break; } } if (validCornerSpritesFound) { for (int i = 0; i < spriteShape.cornerSprites.Count; i++) { CornerSprite cornerSprite = spriteShape.cornerSprites[i]; cornerSpriteList.Add(cornerSprite.sprites[0]); } } m_EdgeSpriteArray = edgeSpriteList.ToArray(); m_CornerSpriteArray = cornerSpriteList.ToArray(); m_AngleRangeInfoArray = angleRangeInfoList.ToArray(); List spriteList = new List(); spriteList.AddRange(m_EdgeSpriteArray); spriteList.AddRange(m_CornerSpriteArray); m_SpriteArray = spriteList.ToArray(); } else { m_SpriteArray = new Sprite[0]; m_EdgeSpriteArray = new Sprite[0]; m_CornerSpriteArray = new Sprite[0]; m_AngleRangeInfoArray = new AngleRangeInfo[0]; } } internal NativeArray GetShapeControlPoints() { int pointCount = spline.GetPointCount(); NativeArray shapePoints = new NativeArray(pointCount, Allocator.Temp); for (int i = 0; i < pointCount; ++i) { ShapeControlPoint shapeControlPoint; shapeControlPoint.position = spline.GetPosition(i); shapeControlPoint.leftTangent = spline.GetLeftTangent(i); shapeControlPoint.rightTangent = spline.GetRightTangent(i); shapeControlPoint.mode = (int)spline.GetTangentMode(i); shapePoints[i] = shapeControlPoint; } return shapePoints; } internal NativeArray GetSplinePointMetaData() { int pointCount = spline.GetPointCount(); NativeArray shapeMetaData = new NativeArray(pointCount, Allocator.Temp); for (int i = 0; i < pointCount; ++i) { SplinePointMetaData metaData; metaData.height = m_Spline.GetHeight(i); metaData.spriteIndex = (uint)m_Spline.GetSpriteIndex(i); metaData.cornerMode = (int)m_Spline.GetCornerMode(i); shapeMetaData[i] = metaData; } return shapeMetaData; } internal int CalculateMaxArrayCount(NativeArray shapePoints) { int maxVertexCount = 1024 * 64; bool hasSprites = false; float smallestWidth = 99999.0f; if (null != spriteArray) { foreach (var sprite in m_SpriteArray) { if (sprite != null) { hasSprites = true; float pixelWidth = BezierUtility.GetSpritePixelWidth(sprite); smallestWidth = (smallestWidth > pixelWidth) ? pixelWidth : smallestWidth; } } } // Approximate vertex Array Count. Include Corners and Wide Sprites into account. float smallestSegment = smallestWidth; float shapeLength = BezierUtility.BezierLength(shapePoints, splineDetail, ref smallestSegment) * 4.0f; int adjustShape = shapePoints.Length * 5 * splineDetail; int adjustWidth = hasSprites ? ((int)(shapeLength / smallestSegment) * splineDetail) + adjustShape : 0; adjustShape = optimizeGeometry ? (adjustShape) : (adjustShape * 2); adjustShape = ValidateSpriteShapeTexture() ? adjustShape : 0; maxArrayCount = adjustShape + adjustWidth; maxArrayCount = math.min(maxArrayCount, maxVertexCount); return maxArrayCount; } #endregion #region ScheduleAndGenerate unsafe JobHandle ScheduleBake() { JobHandle jobHandle = default; bool staticUpload = Application.isPlaying; #if !UNITY_EDITOR staticUpload = true; #endif if (staticUpload && geometryCached) { if (spriteShapeGeometryCache) if (spriteShapeGeometryCache.maxArrayCount != 0) return spriteShapeGeometryCache.Upload(spriteShapeRenderer, this); } maxArrayCount = spriteShapeCreator.GetVertexArrayCount(this); if (maxArrayCount > 0 && enabled) { // Complate previos m_JobHandle.Complete(); // Collider Data if (m_ColliderData.IsCreated) m_ColliderData.Dispose(); m_ColliderData = new NativeArray(maxArrayCount, Allocator.Persistent); // Tangent Data if (!m_TangentData.IsCreated) m_TangentData = new NativeArray(1, Allocator.Persistent); NativeArray indexArray; NativeSlice posArray; NativeSlice uv0Array; NativeArray geomArray = spriteShapeRenderer.GetSegments(spline.GetPointCount() * 8); NativeSlice tanArray = new NativeSlice(m_TangentData); if (m_EnableTangents) { spriteShapeRenderer.GetChannels(maxArrayCount, out indexArray, out posArray, out uv0Array, out tanArray); } else { spriteShapeRenderer.GetChannels(maxArrayCount, out indexArray, out posArray, out uv0Array); } m_JobHandle = jobHandle = spriteShapeCreator.MakeCreatorJob(this, indexArray, posArray, uv0Array, tanArray, geomArray, m_ColliderData); foreach (var geomMod in m_Modifiers) if (null != geomMod) m_JobHandle = geomMod.MakeModifierJob(m_JobHandle, this, indexArray, posArray, uv0Array, tanArray, geomArray, m_ColliderData); // Prepare Renderer. spriteShapeRenderer.Prepare(m_JobHandle, m_ActiveShapeParameters, m_SpriteArray); jobHandle = m_JobHandle; #if UNITY_EDITOR if (spriteShapeGeometryCache && geometryCached) spriteShapeGeometryCache.SetGeometryCache(maxArrayCount, posArray, uv0Array, tanArray, indexArray, geomArray); #endif JobHandle.ScheduleBatchedJobs(); } if (m_DynamicOcclusionOverriden) { spriteShapeRenderer.allowOcclusionWhenDynamic = m_DynamicOcclusionLocal; m_DynamicOcclusionOverriden = false; } return jobHandle; } /// /// Update Collider of this Object. /// public void BakeCollider() { // Previously this must be explicitly called if using BakeMesh. // But now we do it internally. BakeCollider_CanBeCalledMultipleTimesWithoutJobComplete m_JobHandle.Complete(); if (m_ColliderData.IsCreated) { if ((autoUpdateCollider && hasCollider) || forceColliderShapeUpdate) { int maxCount = short.MaxValue - 1; float2 last = (float2)0; m_ColliderSegment.Clear(); for (int i = 0; i < maxCount; ++i) { float2 now = m_ColliderData[i]; if (!math.any(last) && !math.any(now)) { if ((i+1) < maxCount) { float2 next = m_ColliderData[i+1]; if (!math.any(next) && !math.any(next)) break; } else break; } m_ColliderSegment.Add(new Vector2(now.x, now.y)); } if (autoUpdateCollider) { if (edgeCollider != null) edgeCollider.points = m_ColliderSegment.ToArray(); if (polygonCollider != null) polygonCollider.points = m_ColliderSegment.ToArray(); } #if UNITY_EDITOR UnityEditor.SceneView.RepaintAll(); #endif } // Dispose Collider as its no longer needed. m_ColliderData.Dispose(); // Print Once. if (m_Statistics.IsCreated) { var stats = m_Statistics[0]; switch (stats.status) { case SpriteShapeGeneratorResult.ErrorNativeDataOverflow: Debug.LogWarningFormat(gameObject, "NativeArray access not within range. Please submit a bug report."); break; case SpriteShapeGeneratorResult.ErrorSpritesTightPacked: Debug.LogWarningFormat(gameObject, "Sprites used in SpriteShape profile must use FullRect."); break; case SpriteShapeGeneratorResult.ErrorSpritesWrongBorder: Debug.LogWarningFormat(gameObject, "Sprites used in SpriteShape profile have invalid borders. Please check SpriteShape profile."); break; case SpriteShapeGeneratorResult.ErrorVertexLimitReached: Debug.LogWarningFormat(gameObject, "Mesh data has reached Limits. Please try dividing shape into smaller blocks."); break; case SpriteShapeGeneratorResult.ErrorDefaultQuadCreated: Debug.LogWarningFormat(gameObject, "Fill tessellation (C# Job) encountered errors. Please disable it to use default tessellation for fill geometry."); break; } } } } internal void BakeMeshForced() { if (spriteShapeRenderer != null) { var hasSplineChanged = HasSplineDataChanged(); if (hasSplineChanged) { BakeMesh(); Rendering.CommandBuffer rc = new Rendering.CommandBuffer(); rc.GetTemporaryRT(0, 256, 256, 0); rc.SetRenderTarget(0); rc.DrawRenderer(spriteShapeRenderer, spriteShapeRenderer.sharedMaterial); rc.ReleaseTemporaryRT(0); Graphics.ExecuteCommandBuffer(rc); } } } #endregion internal void ForceColliderShapeUpdate(bool forceUpdate) { m_ForceColliderShapeUpdate = forceUpdate; } internal NativeArray GetColliderShapeData() { if (m_ColliderData.IsCreated) { JobHandle handle = BakeMesh(); handle.Complete(); BakeCollider(); } NativeArray retNativeArray = new NativeArray(m_ColliderSegment.Count, Allocator.Temp); for (int i = 0; i < retNativeArray.Length; i++) retNativeArray[i] = m_ColliderSegment[i]; return retNativeArray; } } }