364 lines
16 KiB
C#
364 lines
16 KiB
C#
// -----------------------------------------------------------------------
|
|
// <copyright file="TriangleLocator.cs" company="">
|
|
// 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
|
|
{
|
|
using TriangleNet.Geometry;
|
|
using TriangleNet.Topology;
|
|
|
|
/// <summary>
|
|
/// Locate triangles in a mesh.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// WARNING: This routine is designed for convex triangulations, and will
|
|
/// not generally work after the holes and concavities have been carved.
|
|
///
|
|
/// Based on a paper by Ernst P. Mucke, Isaac Saias, and Binhai Zhu, "Fast
|
|
/// Randomized Point Location Without Preprocessing in Two- and Three-Dimensional
|
|
/// Delaunay Triangulations," Proceedings of the Twelfth Annual Symposium on
|
|
/// Computational Geometry, ACM, May 1996.
|
|
/// </remarks>
|
|
public class TriangleLocator
|
|
{
|
|
TriangleSampler sampler;
|
|
Mesh mesh;
|
|
|
|
IPredicates predicates;
|
|
|
|
// Pointer to a recently visited triangle. Improves point location if
|
|
// proximate vertices are inserted sequentially.
|
|
internal Otri recenttri;
|
|
|
|
public TriangleLocator(Mesh mesh)
|
|
: this(mesh, RobustPredicates.Default)
|
|
{
|
|
}
|
|
|
|
public TriangleLocator(Mesh mesh, IPredicates predicates)
|
|
{
|
|
this.mesh = mesh;
|
|
this.predicates = predicates;
|
|
|
|
sampler = new TriangleSampler(mesh);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Suggest the given triangle as a starting triangle for point location.
|
|
/// </summary>
|
|
/// <param name="otri"></param>
|
|
public void Update(ref Otri otri)
|
|
{
|
|
otri.Copy(ref recenttri);
|
|
}
|
|
|
|
public void Reset()
|
|
{
|
|
sampler.Reset();
|
|
recenttri.tri = null; // No triangle has been visited yet.
|
|
}
|
|
|
|
/// <summary>
|
|
/// Find a triangle or edge containing a given point.
|
|
/// </summary>
|
|
/// <param name="searchpoint">The point to locate.</param>
|
|
/// <param name="searchtri">The triangle to start the search at.</param>
|
|
/// <param name="stopatsubsegment"> If 'stopatsubsegment' is set, the search
|
|
/// will stop if it tries to walk through a subsegment, and will return OUTSIDE.</param>
|
|
/// <returns>Location information.</returns>
|
|
/// <remarks>
|
|
/// Begins its search from 'searchtri'. It is important that 'searchtri'
|
|
/// be a handle with the property that 'searchpoint' is strictly to the left
|
|
/// of the edge denoted by 'searchtri', or is collinear with that edge and
|
|
/// does not intersect that edge. (In particular, 'searchpoint' should not
|
|
/// be the origin or destination of that edge.)
|
|
///
|
|
/// These conditions are imposed because preciselocate() is normally used in
|
|
/// one of two situations:
|
|
///
|
|
/// (1) To try to find the location to insert a new point. Normally, we
|
|
/// know an edge that the point is strictly to the left of. In the
|
|
/// incremental Delaunay algorithm, that edge is a bounding box edge.
|
|
/// In Ruppert's Delaunay refinement algorithm for quality meshing,
|
|
/// that edge is the shortest edge of the triangle whose circumcenter
|
|
/// is being inserted.
|
|
///
|
|
/// (2) To try to find an existing point. In this case, any edge on the
|
|
/// convex hull is a good starting edge. You must screen out the
|
|
/// possibility that the vertex sought is an endpoint of the starting
|
|
/// edge before you call preciselocate().
|
|
///
|
|
/// On completion, 'searchtri' is a triangle that contains 'searchpoint'.
|
|
///
|
|
/// This implementation differs from that given by Guibas and Stolfi. It
|
|
/// walks from triangle to triangle, crossing an edge only if 'searchpoint'
|
|
/// is on the other side of the line containing that edge. After entering
|
|
/// a triangle, there are two edges by which one can leave that triangle.
|
|
/// If both edges are valid ('searchpoint' is on the other side of both
|
|
/// edges), one of the two is chosen by drawing a line perpendicular to
|
|
/// the entry edge (whose endpoints are 'forg' and 'fdest') passing through
|
|
/// 'fapex'. Depending on which side of this perpendicular 'searchpoint'
|
|
/// falls on, an exit edge is chosen.
|
|
///
|
|
/// This implementation is empirically faster than the Guibas and Stolfi
|
|
/// point location routine (which I originally used), which tends to spiral
|
|
/// in toward its target.
|
|
///
|
|
/// Returns ONVERTEX if the point lies on an existing vertex. 'searchtri'
|
|
/// is a handle whose origin is the existing vertex.
|
|
///
|
|
/// Returns ONEDGE if the point lies on a mesh edge. 'searchtri' is a
|
|
/// handle whose primary edge is the edge on which the point lies.
|
|
///
|
|
/// Returns INTRIANGLE if the point lies strictly within a triangle.
|
|
/// 'searchtri' is a handle on the triangle that contains the point.
|
|
///
|
|
/// Returns OUTSIDE if the point lies outside the mesh. 'searchtri' is a
|
|
/// handle whose primary edge the point is to the right of. This might
|
|
/// occur when the circumcenter of a triangle falls just slightly outside
|
|
/// the mesh due to floating-point roundoff error. It also occurs when
|
|
/// seeking a hole or region point that a foolish user has placed outside
|
|
/// the mesh.
|
|
///
|
|
/// WARNING: This routine is designed for convex triangulations, and will
|
|
/// not generally work after the holes and concavities have been carved.
|
|
/// However, it can still be used to find the circumcenter of a triangle, as
|
|
/// long as the search is begun from the triangle in question.</remarks>
|
|
public LocateResult PreciseLocate(Point searchpoint, ref Otri searchtri,
|
|
bool stopatsubsegment)
|
|
{
|
|
Otri backtracktri = default(Otri);
|
|
Osub checkedge = default(Osub);
|
|
Vertex forg, fdest, fapex;
|
|
double orgorient, destorient;
|
|
bool moveleft;
|
|
|
|
// Where are we?
|
|
forg = searchtri.Org();
|
|
fdest = searchtri.Dest();
|
|
fapex = searchtri.Apex();
|
|
while (true)
|
|
{
|
|
// Check whether the apex is the point we seek.
|
|
if ((fapex.x == searchpoint.x) && (fapex.y == searchpoint.y))
|
|
{
|
|
searchtri.Lprev();
|
|
return LocateResult.OnVertex;
|
|
}
|
|
// Does the point lie on the other side of the line defined by the
|
|
// triangle edge opposite the triangle's destination?
|
|
destorient = predicates.CounterClockwise(forg, fapex, searchpoint);
|
|
// Does the point lie on the other side of the line defined by the
|
|
// triangle edge opposite the triangle's origin?
|
|
orgorient = predicates.CounterClockwise(fapex, fdest, searchpoint);
|
|
if (destorient > 0.0)
|
|
{
|
|
if (orgorient > 0.0)
|
|
{
|
|
// Move left if the inner product of (fapex - searchpoint) and
|
|
// (fdest - forg) is positive. This is equivalent to drawing
|
|
// a line perpendicular to the line (forg, fdest) and passing
|
|
// through 'fapex', and determining which side of this line
|
|
// 'searchpoint' falls on.
|
|
moveleft = (fapex.x - searchpoint.x) * (fdest.x - forg.x) +
|
|
(fapex.y - searchpoint.y) * (fdest.y - forg.y) > 0.0;
|
|
}
|
|
else
|
|
{
|
|
moveleft = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (orgorient > 0.0)
|
|
{
|
|
moveleft = false;
|
|
}
|
|
else
|
|
{
|
|
// The point we seek must be on the boundary of or inside this
|
|
// triangle.
|
|
if (destorient == 0.0)
|
|
{
|
|
searchtri.Lprev();
|
|
return LocateResult.OnEdge;
|
|
}
|
|
if (orgorient == 0.0)
|
|
{
|
|
searchtri.Lnext();
|
|
return LocateResult.OnEdge;
|
|
}
|
|
return LocateResult.InTriangle;
|
|
}
|
|
}
|
|
|
|
// Move to another triangle. Leave a trace 'backtracktri' in case
|
|
// floating-point roundoff or some such bogey causes us to walk
|
|
// off a boundary of the triangulation.
|
|
if (moveleft)
|
|
{
|
|
searchtri.Lprev(ref backtracktri);
|
|
fdest = fapex;
|
|
}
|
|
else
|
|
{
|
|
searchtri.Lnext(ref backtracktri);
|
|
forg = fapex;
|
|
}
|
|
backtracktri.Sym(ref searchtri);
|
|
|
|
if (mesh.checksegments && stopatsubsegment)
|
|
{
|
|
// Check for walking through a subsegment.
|
|
backtracktri.Pivot(ref checkedge);
|
|
if (checkedge.seg.hash != Mesh.DUMMY)
|
|
{
|
|
// Go back to the last triangle.
|
|
backtracktri.Copy(ref searchtri);
|
|
return LocateResult.Outside;
|
|
}
|
|
}
|
|
// Check for walking right out of the triangulation.
|
|
if (searchtri.tri.id == Mesh.DUMMY)
|
|
{
|
|
// Go back to the last triangle.
|
|
backtracktri.Copy(ref searchtri);
|
|
return LocateResult.Outside;
|
|
}
|
|
|
|
fapex = searchtri.Apex();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Find a triangle or edge containing a given point.
|
|
/// </summary>
|
|
/// <param name="searchpoint">The point to locate.</param>
|
|
/// <param name="searchtri">The triangle to start the search at.</param>
|
|
/// <returns>Location information.</returns>
|
|
/// <remarks>
|
|
/// Searching begins from one of: the input 'searchtri', a recently
|
|
/// encountered triangle 'recenttri', or from a triangle chosen from a
|
|
/// random sample. The choice is made by determining which triangle's
|
|
/// origin is closest to the point we are searching for. Normally,
|
|
/// 'searchtri' should be a handle on the convex hull of the triangulation.
|
|
///
|
|
/// Details on the random sampling method can be found in the Mucke, Saias,
|
|
/// and Zhu paper cited in the header of this code.
|
|
///
|
|
/// On completion, 'searchtri' is a triangle that contains 'searchpoint'.
|
|
///
|
|
/// Returns ONVERTEX if the point lies on an existing vertex. 'searchtri'
|
|
/// is a handle whose origin is the existing vertex.
|
|
///
|
|
/// Returns ONEDGE if the point lies on a mesh edge. 'searchtri' is a
|
|
/// handle whose primary edge is the edge on which the point lies.
|
|
///
|
|
/// Returns INTRIANGLE if the point lies strictly within a triangle.
|
|
/// 'searchtri' is a handle on the triangle that contains the point.
|
|
///
|
|
/// Returns OUTSIDE if the point lies outside the mesh. 'searchtri' is a
|
|
/// handle whose primary edge the point is to the right of. This might
|
|
/// occur when the circumcenter of a triangle falls just slightly outside
|
|
/// the mesh due to floating-point roundoff error. It also occurs when
|
|
/// seeking a hole or region point that a foolish user has placed outside
|
|
/// the mesh.
|
|
///
|
|
/// WARNING: This routine is designed for convex triangulations, and will
|
|
/// not generally work after the holes and concavities have been carved.
|
|
/// </remarks>
|
|
public LocateResult Locate(Point searchpoint, ref Otri searchtri)
|
|
{
|
|
Otri sampletri = default(Otri);
|
|
Vertex torg, tdest;
|
|
double searchdist, dist;
|
|
double ahead;
|
|
|
|
// Record the distance from the suggested starting triangle to the
|
|
// point we seek.
|
|
torg = searchtri.Org();
|
|
searchdist = (searchpoint.x - torg.x) * (searchpoint.x - torg.x) +
|
|
(searchpoint.y - torg.y) * (searchpoint.y - torg.y);
|
|
|
|
// If a recently encountered triangle has been recorded and has not been
|
|
// deallocated, test it as a good starting point.
|
|
if (recenttri.tri != null)
|
|
{
|
|
if (!Otri.IsDead(recenttri.tri))
|
|
{
|
|
torg = recenttri.Org();
|
|
if ((torg.x == searchpoint.x) && (torg.y == searchpoint.y))
|
|
{
|
|
recenttri.Copy(ref searchtri);
|
|
return LocateResult.OnVertex;
|
|
}
|
|
dist = (searchpoint.x - torg.x) * (searchpoint.x - torg.x) +
|
|
(searchpoint.y - torg.y) * (searchpoint.y - torg.y);
|
|
if (dist < searchdist)
|
|
{
|
|
recenttri.Copy(ref searchtri);
|
|
searchdist = dist;
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: Improve sampling.
|
|
sampler.Update();
|
|
|
|
foreach (var t in sampler)
|
|
{
|
|
sampletri.tri = t;
|
|
if (!Otri.IsDead(sampletri.tri))
|
|
{
|
|
torg = sampletri.Org();
|
|
dist = (searchpoint.x - torg.x) * (searchpoint.x - torg.x) +
|
|
(searchpoint.y - torg.y) * (searchpoint.y - torg.y);
|
|
if (dist < searchdist)
|
|
{
|
|
sampletri.Copy(ref searchtri);
|
|
searchdist = dist;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Where are we?
|
|
torg = searchtri.Org();
|
|
tdest = searchtri.Dest();
|
|
|
|
// Check the starting triangle's vertices.
|
|
if ((torg.x == searchpoint.x) && (torg.y == searchpoint.y))
|
|
{
|
|
return LocateResult.OnVertex;
|
|
}
|
|
if ((tdest.x == searchpoint.x) && (tdest.y == searchpoint.y))
|
|
{
|
|
searchtri.Lnext();
|
|
return LocateResult.OnVertex;
|
|
}
|
|
|
|
// Orient 'searchtri' to fit the preconditions of calling preciselocate().
|
|
ahead = predicates.CounterClockwise(torg, tdest, searchpoint);
|
|
if (ahead < 0.0)
|
|
{
|
|
// Turn around so that 'searchpoint' is to the left of the
|
|
// edge specified by 'searchtri'.
|
|
searchtri.Sym();
|
|
}
|
|
else if (ahead == 0.0)
|
|
{
|
|
// Check if 'searchpoint' is between 'torg' and 'tdest'.
|
|
if (((torg.x < searchpoint.x) == (searchpoint.x < tdest.x)) &&
|
|
((torg.y < searchpoint.y) == (searchpoint.y < tdest.y)))
|
|
{
|
|
return LocateResult.OnEdge;
|
|
}
|
|
}
|
|
|
|
return PreciseLocate(searchpoint, ref searchtri, false);
|
|
}
|
|
}
|
|
}
|