192 lines
7.5 KiB
C#
192 lines
7.5 KiB
C#
|
// -----------------------------------------------------------------------
|
|||
|
// <copyright file="Incremental.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.Algorithm
|
|||
|
{
|
|||
|
using System.Collections.Generic;
|
|||
|
using TriangleNet.Topology;
|
|||
|
using TriangleNet.Geometry;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Builds a delaunay triangulation using the incremental algorithm.
|
|||
|
/// </summary>
|
|||
|
public class Incremental : ITriangulator
|
|||
|
{
|
|||
|
Mesh mesh;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Form a Delaunay triangulation by incrementally inserting vertices.
|
|||
|
/// </summary>
|
|||
|
/// <returns>Returns the number of edges on the convex hull of the
|
|||
|
/// triangulation.</returns>
|
|||
|
public IMesh Triangulate(IList<Vertex> points, Configuration config)
|
|||
|
{
|
|||
|
this.mesh = new Mesh(config);
|
|||
|
this.mesh.TransferNodes(points);
|
|||
|
|
|||
|
Otri starttri = new Otri();
|
|||
|
|
|||
|
// Create a triangular bounding box.
|
|||
|
GetBoundingBox();
|
|||
|
|
|||
|
foreach (var v in mesh.vertices.Values)
|
|||
|
{
|
|||
|
starttri.tri = mesh.dummytri;
|
|||
|
Osub tmp = default(Osub);
|
|||
|
if (mesh.InsertVertex(v, ref starttri, ref tmp, false, false) == InsertVertexResult.Duplicate)
|
|||
|
{
|
|||
|
if (Log.Verbose)
|
|||
|
{
|
|||
|
Log.Instance.Warning("A duplicate vertex appeared and was ignored.",
|
|||
|
"Incremental.Triangulate()");
|
|||
|
}
|
|||
|
v.type = VertexType.UndeadVertex;
|
|||
|
mesh.undeads++;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Remove the bounding box.
|
|||
|
this.mesh.hullsize = RemoveBox();
|
|||
|
|
|||
|
return this.mesh;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Form an "infinite" bounding triangle to insert vertices into.
|
|||
|
/// </summary>
|
|||
|
/// <remarks>
|
|||
|
/// The vertices at "infinity" are assigned finite coordinates, which are
|
|||
|
/// used by the point location routines, but (mostly) ignored by the
|
|||
|
/// Delaunay edge flip routines.
|
|||
|
/// </remarks>
|
|||
|
void GetBoundingBox()
|
|||
|
{
|
|||
|
Otri inftri = default(Otri); // Handle for the triangular bounding box.
|
|||
|
Rectangle box = mesh.bounds;
|
|||
|
|
|||
|
// Find the width (or height, whichever is larger) of the triangulation.
|
|||
|
double width = box.Width;
|
|||
|
if (box.Height > width)
|
|||
|
{
|
|||
|
width = box.Height;
|
|||
|
}
|
|||
|
if (width == 0.0)
|
|||
|
{
|
|||
|
width = 1.0;
|
|||
|
}
|
|||
|
// Create the vertices of the bounding box.
|
|||
|
mesh.infvertex1 = new Vertex(box.Left - 50.0 * width, box.Bottom - 40.0 * width);
|
|||
|
mesh.infvertex2 = new Vertex(box.Right + 50.0 * width, box.Bottom - 40.0 * width);
|
|||
|
mesh.infvertex3 = new Vertex(0.5 * (box.Left + box.Right), box.Top + 60.0 * width);
|
|||
|
|
|||
|
// Create the bounding box.
|
|||
|
mesh.MakeTriangle(ref inftri);
|
|||
|
|
|||
|
inftri.SetOrg(mesh.infvertex1);
|
|||
|
inftri.SetDest(mesh.infvertex2);
|
|||
|
inftri.SetApex(mesh.infvertex3);
|
|||
|
|
|||
|
// Link dummytri to the bounding box so we can always find an
|
|||
|
// edge to begin searching (point location) from.
|
|||
|
mesh.dummytri.neighbors[0] = inftri;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Remove the "infinite" bounding triangle, setting boundary markers as appropriate.
|
|||
|
/// </summary>
|
|||
|
/// <returns>Returns the number of edges on the convex hull of the triangulation.</returns>
|
|||
|
/// <remarks>
|
|||
|
/// The triangular bounding box has three boundary triangles (one for each
|
|||
|
/// side of the bounding box), and a bunch of triangles fanning out from
|
|||
|
/// the three bounding box vertices (one triangle for each edge of the
|
|||
|
/// convex hull of the inner mesh). This routine removes these triangles.
|
|||
|
/// </remarks>
|
|||
|
int RemoveBox()
|
|||
|
{
|
|||
|
Otri deadtriangle = default(Otri);
|
|||
|
Otri searchedge = default(Otri);
|
|||
|
Otri checkedge = default(Otri);
|
|||
|
Otri nextedge = default(Otri), finaledge = default(Otri), dissolveedge = default(Otri);
|
|||
|
Vertex markorg;
|
|||
|
int hullsize;
|
|||
|
|
|||
|
bool noPoly = !mesh.behavior.Poly;
|
|||
|
|
|||
|
// Find a boundary triangle.
|
|||
|
nextedge.tri = mesh.dummytri;
|
|||
|
nextedge.orient = 0;
|
|||
|
nextedge.Sym();
|
|||
|
|
|||
|
// Mark a place to stop.
|
|||
|
nextedge.Lprev(ref finaledge);
|
|||
|
nextedge.Lnext();
|
|||
|
nextedge.Sym();
|
|||
|
// Find a triangle (on the boundary of the vertex set) that isn't
|
|||
|
// a bounding box triangle.
|
|||
|
nextedge.Lprev(ref searchedge);
|
|||
|
searchedge.Sym();
|
|||
|
// Check whether nextedge is another boundary triangle
|
|||
|
// adjacent to the first one.
|
|||
|
nextedge.Lnext(ref checkedge);
|
|||
|
checkedge.Sym();
|
|||
|
if (checkedge.tri.id == Mesh.DUMMY)
|
|||
|
{
|
|||
|
// Go on to the next triangle. There are only three boundary
|
|||
|
// triangles, and this next triangle cannot be the third one,
|
|||
|
// so it's safe to stop here.
|
|||
|
searchedge.Lprev();
|
|||
|
searchedge.Sym();
|
|||
|
}
|
|||
|
|
|||
|
// Find a new boundary edge to search from, as the current search
|
|||
|
// edge lies on a bounding box triangle and will be deleted.
|
|||
|
mesh.dummytri.neighbors[0] = searchedge;
|
|||
|
|
|||
|
hullsize = -2;
|
|||
|
while (!nextedge.Equals(finaledge))
|
|||
|
{
|
|||
|
hullsize++;
|
|||
|
nextedge.Lprev(ref dissolveedge);
|
|||
|
dissolveedge.Sym();
|
|||
|
// If not using a PSLG, the vertices should be marked now.
|
|||
|
// (If using a PSLG, markhull() will do the job.)
|
|||
|
if (noPoly)
|
|||
|
{
|
|||
|
// Be careful! One must check for the case where all the input
|
|||
|
// vertices are collinear, and thus all the triangles are part of
|
|||
|
// the bounding box. Otherwise, the setvertexmark() call below
|
|||
|
// will cause a bad pointer reference.
|
|||
|
if (dissolveedge.tri.id != Mesh.DUMMY)
|
|||
|
{
|
|||
|
markorg = dissolveedge.Org();
|
|||
|
if (markorg.label == 0)
|
|||
|
{
|
|||
|
markorg.label = 1;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
// Disconnect the bounding box triangle from the mesh triangle.
|
|||
|
dissolveedge.Dissolve(mesh.dummytri);
|
|||
|
nextedge.Lnext(ref deadtriangle);
|
|||
|
deadtriangle.Sym(ref nextedge);
|
|||
|
// Get rid of the bounding box triangle.
|
|||
|
mesh.TriangleDealloc(deadtriangle.tri);
|
|||
|
// Do we need to turn the corner?
|
|||
|
if (nextedge.tri.id == Mesh.DUMMY)
|
|||
|
{
|
|||
|
// Turn the corner.
|
|||
|
dissolveedge.Copy(ref nextedge);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
mesh.TriangleDealloc(finaledge.tri);
|
|||
|
|
|||
|
return hullsize;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|