1229 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			C#
		
	
	
	
		
		
			
		
	
	
			1229 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			C#
		
	
	
	
|   | // ----------------------------------------------------------------------- | |||
|  | // <copyright file="ConstraintMesher.cs"> | |||
|  | // Original Triangle code by Jonathan Richard Shewchuk, http://www.cs.cmu.edu/~quake/triangle.html | |||
|  | // Triangle.NET code by Christian Woltering, http://triangle.codeplex.com/ | |||
|  | // </copyright> | |||
|  | // ----------------------------------------------------------------------- | |||
|  | 
 | |||
|  | namespace TriangleNet.Meshing | |||
|  | { | |||
|  |     using System; | |||
|  |     using System.Collections.Generic; | |||
|  |     using TriangleNet.Geometry; | |||
|  |     using TriangleNet.Logging; | |||
|  |     using TriangleNet.Meshing.Iterators; | |||
|  |     using TriangleNet.Topology; | |||
|  | 
 | |||
|  |     internal class ConstraintMesher | |||
|  |     { | |||
|  |         IPredicates predicates; | |||
|  | 
 | |||
|  |         Mesh mesh; | |||
|  |         Behavior behavior; | |||
|  |         TriangleLocator locator; | |||
|  | 
 | |||
|  |         List<Triangle> viri; | |||
|  | 
 | |||
|  |         ILog<LogItem> logger; | |||
|  | 
 | |||
|  |         public ConstraintMesher(Mesh mesh, Configuration config) | |||
|  |         { | |||
|  |             this.mesh = mesh; | |||
|  |             this.predicates = config.Predicates(); | |||
|  | 
 | |||
|  |             this.behavior = mesh.behavior; | |||
|  |             this.locator = mesh.locator; | |||
|  | 
 | |||
|  |             this.viri = new List<Triangle>(); | |||
|  | 
 | |||
|  |             logger = Log.Instance; | |||
|  |         } | |||
|  | 
 | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Insert segments into the mesh. | |||
|  |         /// </summary> | |||
|  |         /// <param name="input">The polygon.</param> | |||
|  |         /// <param name="options">Constraint options.</param> | |||
|  |         public void Apply(IPolygon input, ConstraintOptions options) | |||
|  |         { | |||
|  |             behavior.Poly = input.Segments.Count > 0; | |||
|  | 
 | |||
|  |             // Copy constraint options | |||
|  |             if (options != null) | |||
|  |             { | |||
|  |                 behavior.ConformingDelaunay = options.ConformingDelaunay; | |||
|  |                 behavior.Convex = options.Convex; | |||
|  |                 behavior.NoBisect = options.SegmentSplitting; | |||
|  | 
 | |||
|  |                 if (behavior.ConformingDelaunay) | |||
|  |                 { | |||
|  |                     behavior.Quality = true; | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             //if (input.EdgeMarkers != null) | |||
|  |             //{ | |||
|  |             //    behavior.UseBoundaryMarkers = true; | |||
|  |             //} | |||
|  | 
 | |||
|  |             behavior.useRegions = input.Regions.Count > 0; | |||
|  | 
 | |||
|  |             // Ensure that no vertex can be mistaken for a triangular bounding | |||
|  |             // box vertex in insertvertex(). | |||
|  |             mesh.infvertex1 = null; | |||
|  |             mesh.infvertex2 = null; | |||
|  |             mesh.infvertex3 = null; | |||
|  | 
 | |||
|  |             if (behavior.useSegments) | |||
|  |             { | |||
|  |                 // Segments will be introduced next. | |||
|  |                 mesh.checksegments = true; | |||
|  | 
 | |||
|  |                 // Insert PSLG segments and/or convex hull segments. | |||
|  |                 FormSkeleton(input); | |||
|  |             } | |||
|  | 
 | |||
|  |             if (behavior.Poly && (mesh.triangles.Count > 0)) | |||
|  |             { | |||
|  |                 // Copy holes and regions | |||
|  |                 mesh.holes.AddRange(input.Holes); | |||
|  |                 mesh.regions.AddRange(input.Regions); | |||
|  | 
 | |||
|  |                 // Carve out holes and concavities. | |||
|  |                 CarveHoles(); | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Find the holes and infect them. Find the area constraints and infect  | |||
|  |         /// them. Infect the convex hull. Spread the infection and kill triangles.  | |||
|  |         /// Spread the area constraints. | |||
|  |         /// </summary> | |||
|  |         private void CarveHoles() | |||
|  |         { | |||
|  |             Otri searchtri = default(Otri); | |||
|  |             Vertex searchorg, searchdest; | |||
|  |             LocateResult intersect; | |||
|  | 
 | |||
|  |             Triangle[] regionTris = null; | |||
|  | 
 | |||
|  |             var dummytri = mesh.dummytri; | |||
|  | 
 | |||
|  |             if (!mesh.behavior.Convex) | |||
|  |             { | |||
|  |                 // Mark as infected any unprotected triangles on the boundary. | |||
|  |                 // This is one way by which concavities are created. | |||
|  |                 InfectHull(); | |||
|  |             } | |||
|  | 
 | |||
|  |             if (!mesh.behavior.NoHoles) | |||
|  |             { | |||
|  |                 // Infect each triangle in which a hole lies. | |||
|  |                 foreach (var hole in mesh.holes) | |||
|  |                 { | |||
|  |                     // Ignore holes that aren't within the bounds of the mesh. | |||
|  |                     if (mesh.bounds.Contains(hole)) | |||
|  |                     { | |||
|  |                         // Start searching from some triangle on the outer boundary. | |||
|  |                         searchtri.tri = dummytri; | |||
|  |                         searchtri.orient = 0; | |||
|  |                         searchtri.Sym(); | |||
|  |                         // Ensure that the hole is to the left of this boundary edge; | |||
|  |                         // otherwise, locate() will falsely report that the hole | |||
|  |                         // falls within the starting triangle. | |||
|  |                         searchorg = searchtri.Org(); | |||
|  |                         searchdest = searchtri.Dest(); | |||
|  |                         if (predicates.CounterClockwise(searchorg, searchdest, hole) > 0.0) | |||
|  |                         { | |||
|  |                             // Find a triangle that contains the hole. | |||
|  |                             intersect = mesh.locator.Locate(hole, ref searchtri); | |||
|  |                             if ((intersect != LocateResult.Outside) && (!searchtri.IsInfected())) | |||
|  |                             { | |||
|  |                                 // Infect the triangle. This is done by marking the triangle | |||
|  |                                 // as infected and including the triangle in the virus pool. | |||
|  |                                 searchtri.Infect(); | |||
|  |                                 viri.Add(searchtri.tri); | |||
|  |                             } | |||
|  |                         } | |||
|  |                     } | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             // Now, we have to find all the regions BEFORE we carve the holes, because locate() won't | |||
|  |             // work when the triangulation is no longer convex. (Incidentally, this is the reason why | |||
|  |             // regional attributes and area constraints can't be used when refining a preexisting mesh, | |||
|  |             // which might not be convex; they can only be used with a freshly triangulated PSLG.) | |||
|  |             if (mesh.regions.Count > 0) | |||
|  |             { | |||
|  |                 int i = 0; | |||
|  | 
 | |||
|  |                 regionTris = new Triangle[mesh.regions.Count]; | |||
|  | 
 | |||
|  |                 // Find the starting triangle for each region. | |||
|  |                 foreach (var region in mesh.regions) | |||
|  |                 { | |||
|  |                     regionTris[i] = dummytri; | |||
|  |                     // Ignore region points that aren't within the bounds of the mesh. | |||
|  |                     if (mesh.bounds.Contains(region.point)) | |||
|  |                     { | |||
|  |                         // Start searching from some triangle on the outer boundary. | |||
|  |                         searchtri.tri = dummytri; | |||
|  |                         searchtri.orient = 0; | |||
|  |                         searchtri.Sym(); | |||
|  |                         // Ensure that the region point is to the left of this boundary | |||
|  |                         // edge; otherwise, locate() will falsely report that the | |||
|  |                         // region point falls within the starting triangle. | |||
|  |                         searchorg = searchtri.Org(); | |||
|  |                         searchdest = searchtri.Dest(); | |||
|  |                         if (predicates.CounterClockwise(searchorg, searchdest, region.point) > 0.0) | |||
|  |                         { | |||
|  |                             // Find a triangle that contains the region point. | |||
|  |                             intersect = mesh.locator.Locate(region.point, ref searchtri); | |||
|  |                             if ((intersect != LocateResult.Outside) && (!searchtri.IsInfected())) | |||
|  |                             { | |||
|  |                                 // Record the triangle for processing after the | |||
|  |                                 // holes have been carved. | |||
|  |                                 regionTris[i] = searchtri.tri; | |||
|  |                                 regionTris[i].label = region.id; | |||
|  |                                 regionTris[i].area = region.area; | |||
|  |                             } | |||
|  |                         } | |||
|  |                     } | |||
|  | 
 | |||
|  |                     i++; | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             if (viri.Count > 0) | |||
|  |             { | |||
|  |                 // Carve the holes and concavities. | |||
|  |                 Plague(); | |||
|  |             } | |||
|  | 
 | |||
|  |             if (regionTris != null) | |||
|  |             { | |||
|  |                 var iterator = new RegionIterator(mesh); | |||
|  | 
 | |||
|  |                 for (int i = 0; i < regionTris.Length; i++) | |||
|  |                 { | |||
|  |                     if (regionTris[i].id != Mesh.DUMMY) | |||
|  |                     { | |||
|  |                         // Make sure the triangle under consideration still exists. | |||
|  |                         // It may have been eaten by the virus. | |||
|  |                         if (!Otri.IsDead(regionTris[i])) | |||
|  |                         { | |||
|  |                             // Apply one region's attribute and/or area constraint. | |||
|  |                             iterator.Process(regionTris[i]); | |||
|  |                         } | |||
|  |                     } | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             // Free up memory (virus pool should be empty anyway). | |||
|  |             viri.Clear(); | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Create the segments of a triangulation, including PSLG segments and edges  | |||
|  |         /// on the convex hull. | |||
|  |         /// </summary> | |||
|  |         private void FormSkeleton(IPolygon input) | |||
|  |         { | |||
|  |             // The segment endpoints. | |||
|  |             Vertex p, q; | |||
|  | 
 | |||
|  |             mesh.insegments = 0; | |||
|  | 
 | |||
|  |             if (behavior.Poly) | |||
|  |             { | |||
|  |                 // If the input vertices are collinear, there is no triangulation, | |||
|  |                 // so don't try to insert segments. | |||
|  |                 if (mesh.triangles.Count == 0) | |||
|  |                 { | |||
|  |                     return; | |||
|  |                 } | |||
|  | 
 | |||
|  |                 // If segments are to be inserted, compute a mapping | |||
|  |                 // from vertices to triangles. | |||
|  |                 if (input.Segments.Count > 0) | |||
|  |                 { | |||
|  |                     mesh.MakeVertexMap(); | |||
|  |                 } | |||
|  | 
 | |||
|  |                 // Read and insert the segments. | |||
|  |                 foreach (var seg in input.Segments) | |||
|  |                 { | |||
|  |                     mesh.insegments++; | |||
|  | 
 | |||
|  |                     p = seg.GetVertex(0); | |||
|  |                     q = seg.GetVertex(1); | |||
|  | 
 | |||
|  |                     if ((p.x == q.x) && (p.y == q.y)) | |||
|  |                     { | |||
|  |                         if (Log.Verbose) | |||
|  |                         { | |||
|  |                             logger.Warning("Endpoints of segment (IDs " + p.id + "/" + q.id + ") are coincident.", | |||
|  |                                 "Mesh.FormSkeleton()"); | |||
|  |                         } | |||
|  |                     } | |||
|  |                     else | |||
|  |                     { | |||
|  |                         InsertSegment(p, q, seg.Label); | |||
|  |                     } | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             if (behavior.Convex || !behavior.Poly) | |||
|  |             { | |||
|  |                 // Enclose the convex hull with subsegments. | |||
|  |                 MarkHull(); | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         #region Carving holes | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Virally infect all of the triangles of the convex hull that are not  | |||
|  |         /// protected by subsegments. Where there are subsegments, set boundary  | |||
|  |         /// markers as appropriate. | |||
|  |         /// </summary> | |||
|  |         private void InfectHull() | |||
|  |         { | |||
|  |             Otri hulltri = default(Otri); | |||
|  |             Otri nexttri = default(Otri); | |||
|  |             Otri starttri = default(Otri); | |||
|  |             Osub hullsubseg = default(Osub); | |||
|  |             Vertex horg, hdest; | |||
|  | 
 | |||
|  |             var dummytri = mesh.dummytri; | |||
|  | 
 | |||
|  |             // Find a triangle handle on the hull. | |||
|  |             hulltri.tri = dummytri; | |||
|  |             hulltri.orient = 0; | |||
|  |             hulltri.Sym(); | |||
|  | 
 | |||
|  |             // Remember where we started so we know when to stop. | |||
|  |             hulltri.Copy(ref starttri); | |||
|  |             // Go once counterclockwise around the convex hull. | |||
|  |             do | |||
|  |             { | |||
|  |                 // Ignore triangles that are already infected. | |||
|  |                 if (!hulltri.IsInfected()) | |||
|  |                 { | |||
|  |                     // Is the triangle protected by a subsegment? | |||
|  |                     hulltri.Pivot(ref hullsubseg); | |||
|  |                     if (hullsubseg.seg.hash == Mesh.DUMMY) | |||
|  |                     { | |||
|  |                         // The triangle is not protected; infect it. | |||
|  |                         if (!hulltri.IsInfected()) | |||
|  |                         { | |||
|  |                             hulltri.Infect(); | |||
|  |                             viri.Add(hulltri.tri); | |||
|  |                         } | |||
|  |                     } | |||
|  |                     else | |||
|  |                     { | |||
|  |                         // The triangle is protected; set boundary markers if appropriate. | |||
|  |                         if (hullsubseg.seg.boundary == 0) | |||
|  |                         { | |||
|  |                             hullsubseg.seg.boundary = 1; | |||
|  |                             horg = hulltri.Org(); | |||
|  |                             hdest = hulltri.Dest(); | |||
|  |                             if (horg.label == 0) | |||
|  |                             { | |||
|  |                                 horg.label = 1; | |||
|  |                             } | |||
|  |                             if (hdest.label == 0) | |||
|  |                             { | |||
|  |                                 hdest.label = 1; | |||
|  |                             } | |||
|  |                         } | |||
|  |                     } | |||
|  |                 } | |||
|  |                 // To find the next hull edge, go clockwise around the next vertex. | |||
|  |                 hulltri.Lnext(); | |||
|  |                 hulltri.Oprev(ref nexttri); | |||
|  |                 while (nexttri.tri.id != Mesh.DUMMY) | |||
|  |                 { | |||
|  |                     nexttri.Copy(ref hulltri); | |||
|  |                     hulltri.Oprev(ref nexttri); | |||
|  |                 } | |||
|  | 
 | |||
|  |             } while (!hulltri.Equals(starttri)); | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Spread the virus from all infected triangles to any neighbors not  | |||
|  |         /// protected by subsegments. Delete all infected triangles. | |||
|  |         /// </summary> | |||
|  |         /// <remarks> | |||
|  |         /// This is the procedure that actually creates holes and concavities. | |||
|  |         /// | |||
|  |         /// This procedure operates in two phases. The first phase identifies all | |||
|  |         /// the triangles that will die, and marks them as infected. They are | |||
|  |         /// marked to ensure that each triangle is added to the virus pool only | |||
|  |         /// once, so the procedure will terminate. | |||
|  |         /// | |||
|  |         /// The second phase actually eliminates the infected triangles. It also | |||
|  |         /// eliminates orphaned vertices. | |||
|  |         /// </remarks> | |||
|  |         void Plague() | |||
|  |         { | |||
|  |             Otri testtri = default(Otri); | |||
|  |             Otri neighbor = default(Otri); | |||
|  |             Osub neighborsubseg = default(Osub); | |||
|  |             Vertex testvertex; | |||
|  |             Vertex norg, ndest; | |||
|  | 
 | |||
|  |             var dummysub = mesh.dummysub; | |||
|  |             var dummytri = mesh.dummytri; | |||
|  | 
 | |||
|  |             bool killorg; | |||
|  | 
 | |||
|  |             // Loop through all the infected triangles, spreading the virus to | |||
|  |             // their neighbors, then to their neighbors' neighbors. | |||
|  |             for (int i = 0; i < viri.Count; i++) | |||
|  |             { | |||
|  |                 // WARNING: Don't use foreach, mesh.viri list may get modified. | |||
|  | 
 | |||
|  |                 testtri.tri = viri[i]; | |||
|  |                 // A triangle is marked as infected by messing with one of its pointers | |||
|  |                 // to subsegments, setting it to an illegal value.  Hence, we have to | |||
|  |                 // temporarily uninfect this triangle so that we can examine its | |||
|  |                 // adjacent subsegments. | |||
|  |                 // TODO: Not true in the C# version (so we could skip this). | |||
|  |                 testtri.Uninfect(); | |||
|  | 
 | |||
|  |                 // Check each of the triangle's three neighbors. | |||
|  |                 for (testtri.orient = 0; testtri.orient < 3; testtri.orient++) | |||
|  |                 { | |||
|  |                     // Find the neighbor. | |||
|  |                     testtri.Sym(ref neighbor); | |||
|  |                     // Check for a subsegment between the triangle and its neighbor. | |||
|  |                     testtri.Pivot(ref neighborsubseg); | |||
|  |                     // Check if the neighbor is nonexistent or already infected. | |||
|  |                     if ((neighbor.tri.id == Mesh.DUMMY) || neighbor.IsInfected()) | |||
|  |                     { | |||
|  |                         if (neighborsubseg.seg.hash != Mesh.DUMMY) | |||
|  |                         { | |||
|  |                             // There is a subsegment separating the triangle from its | |||
|  |                             // neighbor, but both triangles are dying, so the subsegment | |||
|  |                             // dies too. | |||
|  |                             mesh.SubsegDealloc(neighborsubseg.seg); | |||
|  |                             if (neighbor.tri.id != Mesh.DUMMY) | |||
|  |                             { | |||
|  |                                 // Make sure the subsegment doesn't get deallocated again | |||
|  |                                 // later when the infected neighbor is visited. | |||
|  |                                 neighbor.Uninfect(); | |||
|  |                                 neighbor.SegDissolve(dummysub); | |||
|  |                                 neighbor.Infect(); | |||
|  |                             } | |||
|  |                         } | |||
|  |                     } | |||
|  |                     else | |||
|  |                     {   // The neighbor exists and is not infected. | |||
|  |                         if (neighborsubseg.seg.hash == Mesh.DUMMY) | |||
|  |                         { | |||
|  |                             // There is no subsegment protecting the neighbor, so | |||
|  |                             // the neighbor becomes infected. | |||
|  |                             neighbor.Infect(); | |||
|  |                             // Ensure that the neighbor's neighbors will be infected. | |||
|  |                             viri.Add(neighbor.tri); | |||
|  |                         } | |||
|  |                         else | |||
|  |                         { | |||
|  |                             // The neighbor is protected by a subsegment. | |||
|  |                             // Remove this triangle from the subsegment. | |||
|  |                             neighborsubseg.TriDissolve(dummytri); | |||
|  |                             // The subsegment becomes a boundary.  Set markers accordingly. | |||
|  |                             if (neighborsubseg.seg.boundary == 0) | |||
|  |                             { | |||
|  |                                 neighborsubseg.seg.boundary = 1; | |||
|  |                             } | |||
|  |                             norg = neighbor.Org(); | |||
|  |                             ndest = neighbor.Dest(); | |||
|  |                             if (norg.label == 0) | |||
|  |                             { | |||
|  |                                 norg.label = 1; | |||
|  |                             } | |||
|  |                             if (ndest.label == 0) | |||
|  |                             { | |||
|  |                                 ndest.label = 1; | |||
|  |                             } | |||
|  |                         } | |||
|  |                     } | |||
|  |                 } | |||
|  |                 // Remark the triangle as infected, so it doesn't get added to the | |||
|  |                 // virus pool again. | |||
|  |                 testtri.Infect(); | |||
|  |             } | |||
|  | 
 | |||
|  |             foreach (var virus in viri) | |||
|  |             { | |||
|  |                 testtri.tri = virus; | |||
|  | 
 | |||
|  |                 // Check each of the three corners of the triangle for elimination. | |||
|  |                 // This is done by walking around each vertex, checking if it is | |||
|  |                 // still connected to at least one live triangle. | |||
|  |                 for (testtri.orient = 0; testtri.orient < 3; testtri.orient++) | |||
|  |                 { | |||
|  |                     testvertex = testtri.Org(); | |||
|  |                     // Check if the vertex has already been tested. | |||
|  |                     if (testvertex != null) | |||
|  |                     { | |||
|  |                         killorg = true; | |||
|  |                         // Mark the corner of the triangle as having been tested. | |||
|  |                         testtri.SetOrg(null); | |||
|  |                         // Walk counterclockwise about the vertex. | |||
|  |                         testtri.Onext(ref neighbor); | |||
|  |                         // Stop upon reaching a boundary or the starting triangle. | |||
|  |                         while ((neighbor.tri.id != Mesh.DUMMY) && | |||
|  |                                (!neighbor.Equals(testtri))) | |||
|  |                         { | |||
|  |                             if (neighbor.IsInfected()) | |||
|  |                             { | |||
|  |                                 // Mark the corner of this triangle as having been tested. | |||
|  |                                 neighbor.SetOrg(null); | |||
|  |                             } | |||
|  |                             else | |||
|  |                             { | |||
|  |                                 // A live triangle.  The vertex survives. | |||
|  |                                 killorg = false; | |||
|  |                             } | |||
|  |                             // Walk counterclockwise about the vertex. | |||
|  |                             neighbor.Onext(); | |||
|  |                         } | |||
|  |                         // If we reached a boundary, we must walk clockwise as well. | |||
|  |                         if (neighbor.tri.id == Mesh.DUMMY) | |||
|  |                         { | |||
|  |                             // Walk clockwise about the vertex. | |||
|  |                             testtri.Oprev(ref neighbor); | |||
|  |                             // Stop upon reaching a boundary. | |||
|  |                             while (neighbor.tri.id != Mesh.DUMMY) | |||
|  |                             { | |||
|  |                                 if (neighbor.IsInfected()) | |||
|  |                                 { | |||
|  |                                     // Mark the corner of this triangle as having been tested. | |||
|  |                                     neighbor.SetOrg(null); | |||
|  |                                 } | |||
|  |                                 else | |||
|  |                                 { | |||
|  |                                     // A live triangle.  The vertex survives. | |||
|  |                                     killorg = false; | |||
|  |                                 } | |||
|  |                                 // Walk clockwise about the vertex. | |||
|  |                                 neighbor.Oprev(); | |||
|  |                             } | |||
|  |                         } | |||
|  |                         if (killorg) | |||
|  |                         { | |||
|  |                             // Deleting vertex | |||
|  |                             testvertex.type = VertexType.UndeadVertex; | |||
|  |                             mesh.undeads++; | |||
|  |                         } | |||
|  |                     } | |||
|  |                 } | |||
|  | 
 | |||
|  |                 // Record changes in the number of boundary edges, and disconnect | |||
|  |                 // dead triangles from their neighbors. | |||
|  |                 for (testtri.orient = 0; testtri.orient < 3; testtri.orient++) | |||
|  |                 { | |||
|  |                     testtri.Sym(ref neighbor); | |||
|  |                     if (neighbor.tri.id == Mesh.DUMMY) | |||
|  |                     { | |||
|  |                         // There is no neighboring triangle on this edge, so this edge | |||
|  |                         // is a boundary edge. This triangle is being deleted, so this | |||
|  |                         // boundary edge is deleted. | |||
|  |                         mesh.hullsize--; | |||
|  |                     } | |||
|  |                     else | |||
|  |                     { | |||
|  |                         // Disconnect the triangle from its neighbor. | |||
|  |                         neighbor.Dissolve(dummytri); | |||
|  |                         // There is a neighboring triangle on this edge, so this edge | |||
|  |                         // becomes a boundary edge when this triangle is deleted. | |||
|  |                         mesh.hullsize++; | |||
|  |                     } | |||
|  |                 } | |||
|  |                 // Return the dead triangle to the pool of triangles. | |||
|  |                 mesh.TriangleDealloc(testtri.tri); | |||
|  |             } | |||
|  | 
 | |||
|  |             // Empty the virus pool. | |||
|  |             viri.Clear(); | |||
|  |         } | |||
|  | 
 | |||
|  |         #endregion | |||
|  | 
 | |||
|  |         #region Segment insertion | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Find the first triangle on the path from one point to another. | |||
|  |         /// </summary> | |||
|  |         /// <param name="searchtri"></param> | |||
|  |         /// <param name="searchpoint"></param> | |||
|  |         /// <returns> | |||
|  |         /// The return value notes whether the destination or apex of the found | |||
|  |         /// triangle is collinear with the two points in question.</returns> | |||
|  |         /// <remarks> | |||
|  |         /// Finds the triangle that intersects a line segment drawn from the | |||
|  |         /// origin of 'searchtri' to the point 'searchpoint', and returns the result | |||
|  |         /// in 'searchtri'. The origin of 'searchtri' does not change, even though | |||
|  |         /// the triangle returned may differ from the one passed in. This routine | |||
|  |         /// is used to find the direction to move in to get from one point to | |||
|  |         /// another. | |||
|  |         /// </remarks> | |||
|  |         private FindDirectionResult FindDirection(ref Otri searchtri, Vertex searchpoint) | |||
|  |         { | |||
|  |             Otri checktri = default(Otri); | |||
|  |             Vertex startvertex; | |||
|  |             Vertex leftvertex, rightvertex; | |||
|  |             double leftccw, rightccw; | |||
|  |             bool leftflag, rightflag; | |||
|  | 
 | |||
|  |             startvertex = searchtri.Org(); | |||
|  |             rightvertex = searchtri.Dest(); | |||
|  |             leftvertex = searchtri.Apex(); | |||
|  |             // Is 'searchpoint' to the left? | |||
|  |             leftccw = predicates.CounterClockwise(searchpoint, startvertex, leftvertex); | |||
|  |             leftflag = leftccw > 0.0; | |||
|  |             // Is 'searchpoint' to the right? | |||
|  |             rightccw = predicates.CounterClockwise(startvertex, searchpoint, rightvertex); | |||
|  |             rightflag = rightccw > 0.0; | |||
|  |             if (leftflag && rightflag) | |||
|  |             { | |||
|  |                 // 'searchtri' faces directly away from 'searchpoint'. We could go left | |||
|  |                 // or right. Ask whether it's a triangle or a boundary on the left. | |||
|  |                 searchtri.Onext(ref checktri); | |||
|  |                 if (checktri.tri.id == Mesh.DUMMY) | |||
|  |                 { | |||
|  |                     leftflag = false; | |||
|  |                 } | |||
|  |                 else | |||
|  |                 { | |||
|  |                     rightflag = false; | |||
|  |                 } | |||
|  |             } | |||
|  |             while (leftflag) | |||
|  |             { | |||
|  |                 // Turn left until satisfied. | |||
|  |                 searchtri.Onext(); | |||
|  |                 if (searchtri.tri.id == Mesh.DUMMY) | |||
|  |                 { | |||
|  |                     logger.Error("Unable to find a triangle on path.", "Mesh.FindDirection().1"); | |||
|  |                     throw new Exception("Unable to find a triangle on path."); | |||
|  |                 } | |||
|  |                 leftvertex = searchtri.Apex(); | |||
|  |                 rightccw = leftccw; | |||
|  |                 leftccw = predicates.CounterClockwise(searchpoint, startvertex, leftvertex); | |||
|  |                 leftflag = leftccw > 0.0; | |||
|  |             } | |||
|  |             while (rightflag) | |||
|  |             { | |||
|  |                 // Turn right until satisfied. | |||
|  |                 searchtri.Oprev(); | |||
|  |                 if (searchtri.tri.id == Mesh.DUMMY) | |||
|  |                 { | |||
|  |                     logger.Error("Unable to find a triangle on path.", "Mesh.FindDirection().2"); | |||
|  |                     throw new Exception("Unable to find a triangle on path."); | |||
|  |                 } | |||
|  |                 rightvertex = searchtri.Dest(); | |||
|  |                 leftccw = rightccw; | |||
|  |                 rightccw = predicates.CounterClockwise(startvertex, searchpoint, rightvertex); | |||
|  |                 rightflag = rightccw > 0.0; | |||
|  |             } | |||
|  |             if (leftccw == 0.0) | |||
|  |             { | |||
|  |                 return FindDirectionResult.Leftcollinear; | |||
|  |             } | |||
|  |             else if (rightccw == 0.0) | |||
|  |             { | |||
|  |                 return FindDirectionResult.Rightcollinear; | |||
|  |             } | |||
|  |             else | |||
|  |             { | |||
|  |                 return FindDirectionResult.Within; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Find the intersection of an existing segment and a segment that is being  | |||
|  |         /// inserted. Insert a vertex at the intersection, splitting an existing subsegment. | |||
|  |         /// </summary> | |||
|  |         /// <param name="splittri"></param> | |||
|  |         /// <param name="splitsubseg"></param> | |||
|  |         /// <param name="endpoint2"></param> | |||
|  |         /// <remarks> | |||
|  |         /// The segment being inserted connects the apex of splittri to endpoint2. | |||
|  |         /// splitsubseg is the subsegment being split, and MUST adjoin splittri. | |||
|  |         /// Hence, endpoints of the subsegment being split are the origin and | |||
|  |         /// destination of splittri. | |||
|  |         /// | |||
|  |         /// On completion, splittri is a handle having the newly inserted | |||
|  |         /// intersection point as its origin, and endpoint1 as its destination. | |||
|  |         /// </remarks> | |||
|  |         private void SegmentIntersection(ref Otri splittri, ref Osub splitsubseg, Vertex endpoint2) | |||
|  |         { | |||
|  |             Osub opposubseg = default(Osub); | |||
|  |             Vertex endpoint1; | |||
|  |             Vertex torg, tdest; | |||
|  |             Vertex leftvertex, rightvertex; | |||
|  |             Vertex newvertex; | |||
|  |             InsertVertexResult success; | |||
|  | 
 | |||
|  |             var dummysub = mesh.dummysub; | |||
|  | 
 | |||
|  |             double ex, ey; | |||
|  |             double tx, ty; | |||
|  |             double etx, ety; | |||
|  |             double split, denom; | |||
|  | 
 | |||
|  |             // Find the other three segment endpoints. | |||
|  |             endpoint1 = splittri.Apex(); | |||
|  |             torg = splittri.Org(); | |||
|  |             tdest = splittri.Dest(); | |||
|  |             // Segment intersection formulae; see the Antonio reference. | |||
|  |             tx = tdest.x - torg.x; | |||
|  |             ty = tdest.y - torg.y; | |||
|  |             ex = endpoint2.x - endpoint1.x; | |||
|  |             ey = endpoint2.y - endpoint1.y; | |||
|  |             etx = torg.x - endpoint2.x; | |||
|  |             ety = torg.y - endpoint2.y; | |||
|  |             denom = ty * ex - tx * ey; | |||
|  |             if (denom == 0.0) | |||
|  |             { | |||
|  |                 logger.Error("Attempt to find intersection of parallel segments.", | |||
|  |                     "Mesh.SegmentIntersection()"); | |||
|  |                 throw new Exception("Attempt to find intersection of parallel segments."); | |||
|  |             } | |||
|  |             split = (ey * etx - ex * ety) / denom; | |||
|  | 
 | |||
|  |             // Create the new vertex. | |||
|  |             newvertex = new Vertex( | |||
|  |                 torg.x + split * (tdest.x - torg.x), | |||
|  |                 torg.y + split * (tdest.y - torg.y), | |||
|  |                 splitsubseg.seg.boundary | |||
|  | #if USE_ATTRIBS | |||
|  |                 , mesh.nextras | |||
|  | #endif | |||
|  |                 ); | |||
|  | 
 | |||
|  |             newvertex.hash = mesh.hash_vtx++; | |||
|  |             newvertex.id = newvertex.hash; | |||
|  | 
 | |||
|  | #if USE_ATTRIBS | |||
|  |             // Interpolate its attributes. | |||
|  |             for (int i = 0; i < mesh.nextras; i++) | |||
|  |             { | |||
|  |                 newvertex.attributes[i] = torg.attributes[i] + split * (tdest.attributes[i] - torg.attributes[i]); | |||
|  |             } | |||
|  | #endif | |||
|  | #if USE_Z | |||
|  |             newvertex.z = torg.z + split * (tdest.z - torg.z); | |||
|  | #endif | |||
|  | 
 | |||
|  |             mesh.vertices.Add(newvertex.hash, newvertex); | |||
|  | 
 | |||
|  |             // Insert the intersection vertex.  This should always succeed. | |||
|  |             success = mesh.InsertVertex(newvertex, ref splittri, ref splitsubseg, false, false); | |||
|  |             if (success != InsertVertexResult.Successful) | |||
|  |             { | |||
|  |                 logger.Error("Failure to split a segment.", "Mesh.SegmentIntersection()"); | |||
|  |                 throw new Exception("Failure to split a segment."); | |||
|  |             } | |||
|  |             // Record a triangle whose origin is the new vertex. | |||
|  |             newvertex.tri = splittri; | |||
|  |             if (mesh.steinerleft > 0) | |||
|  |             { | |||
|  |                 mesh.steinerleft--; | |||
|  |             } | |||
|  | 
 | |||
|  |             // Divide the segment into two, and correct the segment endpoints. | |||
|  |             splitsubseg.Sym(); | |||
|  |             splitsubseg.Pivot(ref opposubseg); | |||
|  |             splitsubseg.Dissolve(dummysub); | |||
|  |             opposubseg.Dissolve(dummysub); | |||
|  |             do | |||
|  |             { | |||
|  |                 splitsubseg.SetSegOrg(newvertex); | |||
|  |                 splitsubseg.Next(); | |||
|  |             } while (splitsubseg.seg.hash != Mesh.DUMMY); | |||
|  |             do | |||
|  |             { | |||
|  |                 opposubseg.SetSegOrg(newvertex); | |||
|  |                 opposubseg.Next(); | |||
|  |             } while (opposubseg.seg.hash != Mesh.DUMMY); | |||
|  | 
 | |||
|  |             // Inserting the vertex may have caused edge flips.  We wish to rediscover | |||
|  |             // the edge connecting endpoint1 to the new intersection vertex. | |||
|  |             FindDirection(ref splittri, endpoint1); | |||
|  | 
 | |||
|  |             rightvertex = splittri.Dest(); | |||
|  |             leftvertex = splittri.Apex(); | |||
|  |             if ((leftvertex.x == endpoint1.x) && (leftvertex.y == endpoint1.y)) | |||
|  |             { | |||
|  |                 splittri.Onext(); | |||
|  |             } | |||
|  |             else if ((rightvertex.x != endpoint1.x) || (rightvertex.y != endpoint1.y)) | |||
|  |             { | |||
|  |                 logger.Error("Topological inconsistency after splitting a segment.", "Mesh.SegmentIntersection()"); | |||
|  |                 throw new Exception("Topological inconsistency after splitting a segment."); | |||
|  |             } | |||
|  |             // 'splittri' should have destination endpoint1. | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Scout the first triangle on the path from one endpoint to another, and check  | |||
|  |         /// for completion (reaching the second endpoint), a collinear vertex, or the  | |||
|  |         /// intersection of two segments. | |||
|  |         /// </summary> | |||
|  |         /// <param name="searchtri"></param> | |||
|  |         /// <param name="endpoint2"></param> | |||
|  |         /// <param name="newmark"></param> | |||
|  |         /// <returns>Returns true if the entire segment is successfully inserted, and false  | |||
|  |         /// if the job must be finished by ConstrainedEdge().</returns> | |||
|  |         /// <remarks> | |||
|  |         /// If the first triangle on the path has the second endpoint as its | |||
|  |         /// destination or apex, a subsegment is inserted and the job is done. | |||
|  |         /// | |||
|  |         /// If the first triangle on the path has a destination or apex that lies on | |||
|  |         /// the segment, a subsegment is inserted connecting the first endpoint to | |||
|  |         /// the collinear vertex, and the search is continued from the collinear | |||
|  |         /// vertex. | |||
|  |         /// | |||
|  |         /// If the first triangle on the path has a subsegment opposite its origin, | |||
|  |         /// then there is a segment that intersects the segment being inserted. | |||
|  |         /// Their intersection vertex is inserted, splitting the subsegment. | |||
|  |         /// </remarks> | |||
|  |         private bool ScoutSegment(ref Otri searchtri, Vertex endpoint2, int newmark) | |||
|  |         { | |||
|  |             Otri crosstri = default(Otri); | |||
|  |             Osub crosssubseg = default(Osub); | |||
|  |             Vertex leftvertex, rightvertex; | |||
|  |             FindDirectionResult collinear; | |||
|  | 
 | |||
|  |             collinear = FindDirection(ref searchtri, endpoint2); | |||
|  |             rightvertex = searchtri.Dest(); | |||
|  |             leftvertex = searchtri.Apex(); | |||
|  |             if (((leftvertex.x == endpoint2.x) && (leftvertex.y == endpoint2.y)) || | |||
|  |                 ((rightvertex.x == endpoint2.x) && (rightvertex.y == endpoint2.y))) | |||
|  |             { | |||
|  |                 // The segment is already an edge in the mesh. | |||
|  |                 if ((leftvertex.x == endpoint2.x) && (leftvertex.y == endpoint2.y)) | |||
|  |                 { | |||
|  |                     searchtri.Lprev(); | |||
|  |                 } | |||
|  |                 // Insert a subsegment, if there isn't already one there. | |||
|  |                 mesh.InsertSubseg(ref searchtri, newmark); | |||
|  |                 return true; | |||
|  |             } | |||
|  |             else if (collinear == FindDirectionResult.Leftcollinear) | |||
|  |             { | |||
|  |                 // We've collided with a vertex between the segment's endpoints. | |||
|  |                 // Make the collinear vertex be the triangle's origin. | |||
|  |                 searchtri.Lprev(); | |||
|  |                 mesh.InsertSubseg(ref searchtri, newmark); | |||
|  |                 // Insert the remainder of the segment. | |||
|  |                 return ScoutSegment(ref searchtri, endpoint2, newmark); | |||
|  |             } | |||
|  |             else if (collinear == FindDirectionResult.Rightcollinear) | |||
|  |             { | |||
|  |                 // We've collided with a vertex between the segment's endpoints. | |||
|  |                 mesh.InsertSubseg(ref searchtri, newmark); | |||
|  |                 // Make the collinear vertex be the triangle's origin. | |||
|  |                 searchtri.Lnext(); | |||
|  |                 // Insert the remainder of the segment. | |||
|  |                 return ScoutSegment(ref searchtri, endpoint2, newmark); | |||
|  |             } | |||
|  |             else | |||
|  |             { | |||
|  |                 searchtri.Lnext(ref crosstri); | |||
|  |                 crosstri.Pivot(ref crosssubseg); | |||
|  |                 // Check for a crossing segment. | |||
|  |                 if (crosssubseg.seg.hash == Mesh.DUMMY) | |||
|  |                 { | |||
|  |                     return false; | |||
|  |                 } | |||
|  |                 else | |||
|  |                 { | |||
|  |                     // Insert a vertex at the intersection. | |||
|  |                     SegmentIntersection(ref crosstri, ref crosssubseg, endpoint2); | |||
|  |                     crosstri.Copy(ref searchtri); | |||
|  |                     mesh.InsertSubseg(ref searchtri, newmark); | |||
|  |                     // Insert the remainder of the segment. | |||
|  |                     return ScoutSegment(ref searchtri, endpoint2, newmark); | |||
|  |                 } | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Enforce the Delaunay condition at an edge, fanning out recursively from  | |||
|  |         /// an existing vertex. Pay special attention to stacking inverted triangles. | |||
|  |         /// </summary> | |||
|  |         /// <param name="fixuptri"></param> | |||
|  |         /// <param name="leftside">Indicates whether or not fixuptri is to the left of  | |||
|  |         /// the segment being inserted. (Imagine that the segment is pointing up from | |||
|  |         /// endpoint1 to endpoint2.)</param> | |||
|  |         /// <remarks> | |||
|  |         /// This is a support routine for inserting segments into a constrained | |||
|  |         /// Delaunay triangulation. | |||
|  |         /// | |||
|  |         /// The origin of fixuptri is treated as if it has just been inserted, and | |||
|  |         /// the local Delaunay condition needs to be enforced. It is only enforced | |||
|  |         /// in one sector, however, that being the angular range defined by | |||
|  |         /// fixuptri. | |||
|  |         /// | |||
|  |         /// This routine also needs to make decisions regarding the "stacking" of | |||
|  |         /// triangles. (Read the description of ConstrainedEdge() below before | |||
|  |         /// reading on here, so you understand the algorithm.) If the position of | |||
|  |         /// the new vertex (the origin of fixuptri) indicates that the vertex before | |||
|  |         /// it on the polygon is a reflex vertex, then "stack" the triangle by | |||
|  |         /// doing nothing.  (fixuptri is an inverted triangle, which is how stacked | |||
|  |         /// triangles are identified.) | |||
|  |         /// | |||
|  |         /// Otherwise, check whether the vertex before that was a reflex vertex. | |||
|  |         /// If so, perform an edge flip, thereby eliminating an inverted triangle | |||
|  |         /// (popping it off the stack). The edge flip may result in the creation | |||
|  |         /// of a new inverted triangle, depending on whether or not the new vertex | |||
|  |         /// is visible to the vertex three edges behind on the polygon. | |||
|  |         /// | |||
|  |         /// If neither of the two vertices behind the new vertex are reflex | |||
|  |         /// vertices, fixuptri and fartri, the triangle opposite it, are not | |||
|  |         /// inverted; hence, ensure that the edge between them is locally Delaunay. | |||
|  |         /// </remarks> | |||
|  |         private void DelaunayFixup(ref Otri fixuptri, bool leftside) | |||
|  |         { | |||
|  |             Otri neartri = default(Otri); | |||
|  |             Otri fartri = default(Otri); | |||
|  |             Osub faredge = default(Osub); | |||
|  |             Vertex nearvertex, leftvertex, rightvertex, farvertex; | |||
|  | 
 | |||
|  |             fixuptri.Lnext(ref neartri); | |||
|  |             neartri.Sym(ref fartri); | |||
|  |             // Check if the edge opposite the origin of fixuptri can be flipped. | |||
|  |             if (fartri.tri.id == Mesh.DUMMY) | |||
|  |             { | |||
|  |                 return; | |||
|  |             } | |||
|  |             neartri.Pivot(ref faredge); | |||
|  |             if (faredge.seg.hash != Mesh.DUMMY) | |||
|  |             { | |||
|  |                 return; | |||
|  |             } | |||
|  |             // Find all the relevant vertices. | |||
|  |             nearvertex = neartri.Apex(); | |||
|  |             leftvertex = neartri.Org(); | |||
|  |             rightvertex = neartri.Dest(); | |||
|  |             farvertex = fartri.Apex(); | |||
|  |             // Check whether the previous polygon vertex is a reflex vertex. | |||
|  |             if (leftside) | |||
|  |             { | |||
|  |                 if (predicates.CounterClockwise(nearvertex, leftvertex, farvertex) <= 0.0) | |||
|  |                 { | |||
|  |                     // leftvertex is a reflex vertex too. Nothing can | |||
|  |                     // be done until a convex section is found. | |||
|  |                     return; | |||
|  |                 } | |||
|  |             } | |||
|  |             else | |||
|  |             { | |||
|  |                 if (predicates.CounterClockwise(farvertex, rightvertex, nearvertex) <= 0.0) | |||
|  |                 { | |||
|  |                     // rightvertex is a reflex vertex too.  Nothing can | |||
|  |                     // be done until a convex section is found. | |||
|  |                     return; | |||
|  |                 } | |||
|  |             } | |||
|  |             if (predicates.CounterClockwise(rightvertex, leftvertex, farvertex) > 0.0) | |||
|  |             { | |||
|  |                 // fartri is not an inverted triangle, and farvertex is not a reflex | |||
|  |                 // vertex.  As there are no reflex vertices, fixuptri isn't an | |||
|  |                 // inverted triangle, either.  Hence, test the edge between the | |||
|  |                 // triangles to ensure it is locally Delaunay. | |||
|  |                 if (predicates.InCircle(leftvertex, farvertex, rightvertex, nearvertex) <= 0.0) | |||
|  |                 { | |||
|  |                     return; | |||
|  |                 } | |||
|  |                 // Not locally Delaunay; go on to an edge flip. | |||
|  |             } | |||
|  |             // else fartri is inverted; remove it from the stack by flipping. | |||
|  |             mesh.Flip(ref neartri); | |||
|  |             fixuptri.Lprev();    // Restore the origin of fixuptri after the flip. | |||
|  |             // Recursively process the two triangles that result from the flip. | |||
|  |             DelaunayFixup(ref fixuptri, leftside); | |||
|  |             DelaunayFixup(ref fartri, leftside); | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Force a segment into a constrained Delaunay triangulation by deleting the  | |||
|  |         /// triangles it intersects, and triangulating the polygons that form on each  | |||
|  |         /// side of it. | |||
|  |         /// </summary> | |||
|  |         /// <param name="starttri"></param> | |||
|  |         /// <param name="endpoint2"></param> | |||
|  |         /// <param name="newmark"></param> | |||
|  |         /// <remarks> | |||
|  |         /// Generates a single subsegment connecting 'endpoint1' to 'endpoint2'. | |||
|  |         /// The triangle 'starttri' has 'endpoint1' as its origin.  'newmark' is the | |||
|  |         /// boundary marker of the segment. | |||
|  |         /// | |||
|  |         /// To insert a segment, every triangle whose interior intersects the | |||
|  |         /// segment is deleted. The union of these deleted triangles is a polygon | |||
|  |         /// (which is not necessarily monotone, but is close enough), which is | |||
|  |         /// divided into two polygons by the new segment. This routine's task is | |||
|  |         /// to generate the Delaunay triangulation of these two polygons. | |||
|  |         /// | |||
|  |         /// You might think of this routine's behavior as a two-step process.  The | |||
|  |         /// first step is to walk from endpoint1 to endpoint2, flipping each edge | |||
|  |         /// encountered.  This step creates a fan of edges connected to endpoint1, | |||
|  |         /// including the desired edge to endpoint2. The second step enforces the | |||
|  |         /// Delaunay condition on each side of the segment in an incremental manner: | |||
|  |         /// proceeding along the polygon from endpoint1 to endpoint2 (this is done | |||
|  |         /// independently on each side of the segment), each vertex is "enforced" | |||
|  |         /// as if it had just been inserted, but affecting only the previous | |||
|  |         /// vertices. The result is the same as if the vertices had been inserted | |||
|  |         /// in the order they appear on the polygon, so the result is Delaunay. | |||
|  |         /// | |||
|  |         /// In truth, ConstrainedEdge() interleaves these two steps. The procedure | |||
|  |         /// walks from endpoint1 to endpoint2, and each time an edge is encountered | |||
|  |         /// and flipped, the newly exposed vertex (at the far end of the flipped | |||
|  |         /// edge) is "enforced" upon the previously flipped edges, usually affecting | |||
|  |         /// only one side of the polygon (depending upon which side of the segment | |||
|  |         /// the vertex falls on). | |||
|  |         /// | |||
|  |         /// The algorithm is complicated by the need to handle polygons that are not | |||
|  |         /// convex.  Although the polygon is not necessarily monotone, it can be | |||
|  |         /// triangulated in a manner similar to the stack-based algorithms for | |||
|  |         /// monotone polygons. For each reflex vertex (local concavity) of the | |||
|  |         /// polygon, there will be an inverted triangle formed by one of the edge | |||
|  |         /// flips. (An inverted triangle is one with negative area - that is, its | |||
|  |         /// vertices are arranged in clockwise order - and is best thought of as a | |||
|  |         /// wrinkle in the fabric of the mesh.)  Each inverted triangle can be | |||
|  |         /// thought of as a reflex vertex pushed on the stack, waiting to be fixed | |||
|  |         /// later. | |||
|  |         /// | |||
|  |         /// A reflex vertex is popped from the stack when a vertex is inserted that | |||
|  |         /// is visible to the reflex vertex. (However, if the vertex behind the | |||
|  |         /// reflex vertex is not visible to the reflex vertex, a new inverted | |||
|  |         /// triangle will take its place on the stack.) These details are handled | |||
|  |         /// by the DelaunayFixup() routine above. | |||
|  |         /// </remarks> | |||
|  |         private void ConstrainedEdge(ref Otri starttri, Vertex endpoint2, int newmark) | |||
|  |         { | |||
|  |             Otri fixuptri = default(Otri), fixuptri2 = default(Otri); | |||
|  |             Osub crosssubseg = default(Osub); | |||
|  |             Vertex endpoint1; | |||
|  |             Vertex farvertex; | |||
|  |             double area; | |||
|  |             bool collision; | |||
|  |             bool done; | |||
|  | 
 | |||
|  |             endpoint1 = starttri.Org(); | |||
|  |             starttri.Lnext(ref fixuptri); | |||
|  |             mesh.Flip(ref fixuptri); | |||
|  |             // 'collision' indicates whether we have found a vertex directly | |||
|  |             // between endpoint1 and endpoint2. | |||
|  |             collision = false; | |||
|  |             done = false; | |||
|  |             do | |||
|  |             { | |||
|  |                 farvertex = fixuptri.Org(); | |||
|  |                 // 'farvertex' is the extreme point of the polygon we are "digging" | |||
|  |                 //  to get from endpoint1 to endpoint2. | |||
|  |                 if ((farvertex.x == endpoint2.x) && (farvertex.y == endpoint2.y)) | |||
|  |                 { | |||
|  |                     fixuptri.Oprev(ref fixuptri2); | |||
|  |                     // Enforce the Delaunay condition around endpoint2. | |||
|  |                     DelaunayFixup(ref fixuptri, false); | |||
|  |                     DelaunayFixup(ref fixuptri2, true); | |||
|  |                     done = true; | |||
|  |                 } | |||
|  |                 else | |||
|  |                 { | |||
|  |                     // Check whether farvertex is to the left or right of the segment being | |||
|  |                     // inserted, to decide which edge of fixuptri to dig through next. | |||
|  |                     area = predicates.CounterClockwise(endpoint1, endpoint2, farvertex); | |||
|  |                     if (area == 0.0) | |||
|  |                     { | |||
|  |                         // We've collided with a vertex between endpoint1 and endpoint2. | |||
|  |                         collision = true; | |||
|  |                         fixuptri.Oprev(ref fixuptri2); | |||
|  |                         // Enforce the Delaunay condition around farvertex. | |||
|  |                         DelaunayFixup(ref fixuptri, false); | |||
|  |                         DelaunayFixup(ref fixuptri2, true); | |||
|  |                         done = true; | |||
|  |                     } | |||
|  |                     else | |||
|  |                     { | |||
|  |                         if (area > 0.0) | |||
|  |                         { | |||
|  |                             // farvertex is to the left of the segment. | |||
|  |                             fixuptri.Oprev(ref fixuptri2); | |||
|  |                             // Enforce the Delaunay condition around farvertex, on the | |||
|  |                             // left side of the segment only. | |||
|  |                             DelaunayFixup(ref fixuptri2, true); | |||
|  |                             // Flip the edge that crosses the segment. After the edge is | |||
|  |                             // flipped, one of its endpoints is the fan vertex, and the | |||
|  |                             // destination of fixuptri is the fan vertex. | |||
|  |                             fixuptri.Lprev(); | |||
|  |                         } | |||
|  |                         else | |||
|  |                         { | |||
|  |                             // farvertex is to the right of the segment. | |||
|  |                             DelaunayFixup(ref fixuptri, false); | |||
|  |                             // Flip the edge that crosses the segment. After the edge is | |||
|  |                             // flipped, one of its endpoints is the fan vertex, and the | |||
|  |                             // destination of fixuptri is the fan vertex. | |||
|  |                             fixuptri.Oprev(); | |||
|  |                         } | |||
|  |                         // Check for two intersecting segments. | |||
|  |                         fixuptri.Pivot(ref crosssubseg); | |||
|  |                         if (crosssubseg.seg.hash == Mesh.DUMMY) | |||
|  |                         { | |||
|  |                             mesh.Flip(ref fixuptri);    // May create inverted triangle at left. | |||
|  |                         } | |||
|  |                         else | |||
|  |                         { | |||
|  |                             // We've collided with a segment between endpoint1 and endpoint2. | |||
|  |                             collision = true; | |||
|  |                             // Insert a vertex at the intersection. | |||
|  |                             SegmentIntersection(ref fixuptri, ref crosssubseg, endpoint2); | |||
|  |                             done = true; | |||
|  |                         } | |||
|  |                     } | |||
|  |                 } | |||
|  |             } while (!done); | |||
|  |             // Insert a subsegment to make the segment permanent. | |||
|  |             mesh.InsertSubseg(ref fixuptri, newmark); | |||
|  |             // If there was a collision with an interceding vertex, install another | |||
|  |             // segment connecting that vertex with endpoint2. | |||
|  |             if (collision) | |||
|  |             { | |||
|  |                 // Insert the remainder of the segment. | |||
|  |                 if (!ScoutSegment(ref fixuptri, endpoint2, newmark)) | |||
|  |                 { | |||
|  |                     ConstrainedEdge(ref fixuptri, endpoint2, newmark); | |||
|  |                 } | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Insert a PSLG segment into a triangulation. | |||
|  |         /// </summary> | |||
|  |         /// <param name="endpoint1"></param> | |||
|  |         /// <param name="endpoint2"></param> | |||
|  |         /// <param name="newmark"></param> | |||
|  |         private void InsertSegment(Vertex endpoint1, Vertex endpoint2, int newmark) | |||
|  |         { | |||
|  |             Otri searchtri1 = default(Otri), searchtri2 = default(Otri); | |||
|  |             Vertex checkvertex = null; | |||
|  | 
 | |||
|  |             var dummytri = mesh.dummytri; | |||
|  | 
 | |||
|  |             // Find a triangle whose origin is the segment's first endpoint. | |||
|  |             searchtri1 = endpoint1.tri; | |||
|  |             if (searchtri1.tri != null) | |||
|  |             { | |||
|  |                 checkvertex = searchtri1.Org(); | |||
|  |             } | |||
|  | 
 | |||
|  |             if (checkvertex != endpoint1) | |||
|  |             { | |||
|  |                 // Find a boundary triangle to search from. | |||
|  |                 searchtri1.tri = dummytri; | |||
|  |                 searchtri1.orient = 0; | |||
|  |                 searchtri1.Sym(); | |||
|  |                 // Search for the segment's first endpoint by point location. | |||
|  |                 if (locator.Locate(endpoint1, ref searchtri1) != LocateResult.OnVertex) | |||
|  |                 { | |||
|  |                     logger.Error("Unable to locate PSLG vertex in triangulation.", "Mesh.InsertSegment().1"); | |||
|  |                     throw new Exception("Unable to locate PSLG vertex in triangulation."); | |||
|  |                 } | |||
|  |             } | |||
|  |             // Remember this triangle to improve subsequent point location. | |||
|  |             locator.Update(ref searchtri1); | |||
|  | 
 | |||
|  |             // Scout the beginnings of a path from the first endpoint | |||
|  |             // toward the second. | |||
|  |             if (ScoutSegment(ref searchtri1, endpoint2, newmark)) | |||
|  |             { | |||
|  |                 // The segment was easily inserted. | |||
|  |                 return; | |||
|  |             } | |||
|  |             // The first endpoint may have changed if a collision with an intervening | |||
|  |             // vertex on the segment occurred. | |||
|  |             endpoint1 = searchtri1.Org(); | |||
|  | 
 | |||
|  |             // Find a triangle whose origin is the segment's second endpoint. | |||
|  |             checkvertex = null; | |||
|  |             searchtri2 = endpoint2.tri; | |||
|  |             if (searchtri2.tri != null) | |||
|  |             { | |||
|  |                 checkvertex = searchtri2.Org(); | |||
|  |             } | |||
|  |             if (checkvertex != endpoint2) | |||
|  |             { | |||
|  |                 // Find a boundary triangle to search from. | |||
|  |                 searchtri2.tri = dummytri; | |||
|  |                 searchtri2.orient = 0; | |||
|  |                 searchtri2.Sym(); | |||
|  |                 // Search for the segment's second endpoint by point location. | |||
|  |                 if (locator.Locate(endpoint2, ref searchtri2) != LocateResult.OnVertex) | |||
|  |                 { | |||
|  |                     logger.Error("Unable to locate PSLG vertex in triangulation.", "Mesh.InsertSegment().2"); | |||
|  |                     throw new Exception("Unable to locate PSLG vertex in triangulation."); | |||
|  |                 } | |||
|  |             } | |||
|  |             // Remember this triangle to improve subsequent point location. | |||
|  |             locator.Update(ref searchtri2); | |||
|  |             // Scout the beginnings of a path from the second endpoint | |||
|  |             // toward the first. | |||
|  |             if (ScoutSegment(ref searchtri2, endpoint1, newmark)) | |||
|  |             { | |||
|  |                 // The segment was easily inserted. | |||
|  |                 return; | |||
|  |             } | |||
|  |             // The second endpoint may have changed if a collision with an intervening | |||
|  |             // vertex on the segment occurred. | |||
|  |             endpoint2 = searchtri2.Org(); | |||
|  | 
 | |||
|  |             // Insert the segment directly into the triangulation. | |||
|  |             ConstrainedEdge(ref searchtri1, endpoint2, newmark); | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Cover the convex hull of a triangulation with subsegments. | |||
|  |         /// </summary> | |||
|  |         private void MarkHull() | |||
|  |         { | |||
|  |             Otri hulltri = default(Otri); | |||
|  |             Otri nexttri = default(Otri); | |||
|  |             Otri starttri = default(Otri); | |||
|  | 
 | |||
|  |             // Find a triangle handle on the hull. | |||
|  |             hulltri.tri = mesh.dummytri; | |||
|  |             hulltri.orient = 0; | |||
|  |             hulltri.Sym(); | |||
|  |             // Remember where we started so we know when to stop. | |||
|  |             hulltri.Copy(ref starttri); | |||
|  |             // Go once counterclockwise around the convex hull. | |||
|  |             do | |||
|  |             { | |||
|  |                 // Create a subsegment if there isn't already one here. | |||
|  |                 mesh.InsertSubseg(ref hulltri, 1); | |||
|  |                 // To find the next hull edge, go clockwise around the next vertex. | |||
|  |                 hulltri.Lnext(); | |||
|  |                 hulltri.Oprev(ref nexttri); | |||
|  |                 while (nexttri.tri.id != Mesh.DUMMY) | |||
|  |                 { | |||
|  |                     nexttri.Copy(ref hulltri); | |||
|  |                     hulltri.Oprev(ref nexttri); | |||
|  |                 } | |||
|  |             } while (!hulltri.Equals(starttri)); | |||
|  |         } | |||
|  | 
 | |||
|  |         #endregion | |||
|  |     } | |||
|  | } |