// ----------------------------------------------------------------------- // // Triangle.NET code by Christian Woltering, http://triangle.codeplex.com/ // // ----------------------------------------------------------------------- namespace TriangleNet.Tools { using System; using System.Collections.Generic; using TriangleNet.Geometry; public static class PolygonValidator { /// /// Test the polygon for consistency. /// public static bool IsConsistent(IPolygon poly) { var logger = Log.Instance; var points = poly.Points; int horrors = 0; int i = 0; int count = points.Count; if (count < 3) { logger.Warning("Polygon must have at least 3 vertices.", "PolygonValidator.IsConsistent()"); return false; } foreach (var p in points) { if (p == null) { horrors++; logger.Warning(String.Format("Point {0} is null.", i), "PolygonValidator.IsConsistent()"); } else if (double.IsNaN(p.x) || double.IsNaN(p.y)) { horrors++; logger.Warning(String.Format("Point {0} has invalid coordinates.", i), "PolygonValidator.IsConsistent()"); } else if (double.IsInfinity(p.x) || double.IsInfinity(p.y)) { horrors++; logger.Warning(String.Format("Point {0} has invalid coordinates.", i), "PolygonValidator.IsConsistent()"); } i++; } i = 0; foreach (var seg in poly.Segments) { if (seg == null) { horrors++; logger.Warning(String.Format("Segment {0} is null.", i), "PolygonValidator.IsConsistent()"); // Always abort if a NULL-segment is found. return false; } var p = seg.GetVertex(0); var q = seg.GetVertex(1); if ((p.x == q.x) && (p.y == q.y)) { horrors++; logger.Warning(String.Format("Endpoints of segment {0} are coincident (IDs {1} / {2}).", i, p.id, q.id), "PolygonValidator.IsConsistent()"); } i++; } if (points[0].id == points[1].id) { horrors += CheckVertexIDs(poly, count); } else { horrors += CheckDuplicateIDs(poly); } return horrors == 0; } /// /// Test the polygon for duplicate vertices. /// public static bool HasDuplicateVertices(IPolygon poly) { var logger = Log.Instance; int horrors = 0; var points = poly.Points.ToArray(); VertexSorter.Sort(points); for (int i = 1; i < points.Length; i++) { if (points[i - 1] == points[i]) { horrors++; logger.Warning(String.Format("Found duplicate point {0}.", points[i]), "PolygonValidator.HasDuplicateVertices()"); } } return horrors > 0; } /// /// Test the polygon for 360 degree angles. /// /// The polygon. /// The angle threshold. public static bool HasBadAngles(IPolygon poly, double threshold = 2e-12) { var logger = Log.Instance; int horrors = 0; int i = 0; Point p0 = null, p1 = null; Point q0, q1; int count = poly.Points.Count; foreach (var seg in poly.Segments) { q0 = p0; // Previous segment start point. q1 = p1; // Previous segment end point. p0 = seg.GetVertex(0); // Current segment start point. p1 = seg.GetVertex(1); // Current segment end point. if (p0 == p1 || q0 == q1) { // Ignore zero-length segments. continue; } if (q0 != null && q1 != null) { // The two segments are connected. if (p0 == q1 && p1 != null) { if (IsBadAngle(q0, p0, p1,threshold)) { horrors++; logger.Warning(String.Format("Bad segment angle found at index {0}.", i), "PolygonValidator.HasBadAngles()"); } } } i++; } return horrors > 0; } private static bool IsBadAngle(Point a, Point b, Point c, double threshold = 0.0) { double x = DotProduct(a, b, c); double y = CrossProductLength(a, b, c); return Math.Abs(Math.Atan2(y, x)) <= threshold; } // Returns the dot product . private static double DotProduct(Point a, Point b, Point c) { // Calculate the dot product. return (a.x - b.x) * (c.x - b.x) + (a.y - b.y) * (c.y - b.y); } // Returns the length of cross product AB x BC. private static double CrossProductLength(Point a, Point b, Point c) { // Calculate the Z coordinate of the cross product. return (a.x - b.x) * (c.y - b.y) - (a.y - b.y) * (c.x - b.x); } private static int CheckVertexIDs(IPolygon poly, int count) { var logger = Log.Instance; int horrors = 0; int i = 0; Vertex p, q; foreach (var seg in poly.Segments) { p = seg.GetVertex(0); q = seg.GetVertex(1); if (p.id < 0 || p.id >= count) { horrors++; logger.Warning(String.Format("Segment {0} has invalid startpoint.", i), "PolygonValidator.IsConsistent()"); } if (q.id < 0 || q.id >= count) { horrors++; logger.Warning(String.Format("Segment {0} has invalid endpoint.", i), "PolygonValidator.IsConsistent()"); } i++; } return horrors; } private static int CheckDuplicateIDs(IPolygon poly) { var ids = new HashSet(); // Check for duplicate ids. foreach (var p in poly.Points) { if (!ids.Add(p.id)) { Log.Instance.Warning("Found duplicate vertex ids.", "PolygonValidator.IsConsistent()"); return 1; } } return 0; } } }