// ----------------------------------------------------------------------- // // Triangle.NET code by Christian Woltering, http://triangle.codeplex.com/ // // ----------------------------------------------------------------------- namespace TriangleNet.Topology.DCEL { using System.Collections.Generic; using TriangleNet.Geometry; public class DcelMesh { protected List vertices; protected List edges; protected List faces; /// /// Initializes a new instance of the class. /// public DcelMesh() : this(true) { } /// /// Initializes a new instance of the class. /// /// If false, lists will not be initialized. protected DcelMesh(bool initialize) { if (initialize) { vertices = new List(); edges = new List(); faces = new List(); } } /// /// Gets the vertices of the Voronoi diagram. /// public List Vertices { get { return vertices; } } /// /// Gets the list of half-edges specify the Voronoi diagram topology. /// public List HalfEdges { get { return edges; } } /// /// Gets the faces of the Voronoi diagram. /// public List Faces { get { return faces; } } /// /// Gets the collection of edges of the Voronoi diagram. /// public IEnumerable Edges { get { return EnumerateEdges(); } } /// /// Check if the DCEL is consistend. /// /// If true, faces are assumed to be closed (i.e. all edges must have /// a valid next pointer). /// Maximum edge count of faces (default = 0 means skip check). /// public virtual bool IsConsistent(bool closed = true, int depth = 0) { // Check vertices for null pointers. foreach (var vertex in vertices) { if (vertex.id < 0) { continue; } if (vertex.leaving == null) { return false; } if (vertex.Leaving.Origin.id != vertex.id) { return false; } } // Check faces for null pointers. foreach (var face in faces) { if (face.ID < 0) { continue; } if (face.edge == null) { return false; } if (face.id != face.edge.face.id) { return false; } } // Check half-edges for null pointers. foreach (var edge in edges) { if (edge.id < 0) { continue; } if (edge.twin == null) { return false; } if (edge.origin == null) { return false; } if (edge.face == null) { return false; } if (closed && edge.next == null) { return false; } } // Check half-edges (topology). foreach (var edge in edges) { if (edge.id < 0) { continue; } var twin = edge.twin; var next = edge.next; if (edge.id != twin.twin.id) { return false; } if (closed) { if (next.origin.id != twin.origin.id) { return false; } if (next.twin.next.origin.id != edge.twin.origin.id) { return false; } } } if (closed && depth > 0) { // Check if faces are closed. foreach (var face in faces) { if (face.id < 0) { continue; } var edge = face.edge; var next = edge.next; int id = edge.id; int k = 0; while (next.id != id && k < depth) { next = next.next; k++; } if (next.id != id) { return false; } } } return true; } /// /// Search for half-edge without twin and add a twin. Connect twins to form connected /// boundary contours. /// /// /// This method assumes that all faces are closed (i.e. no edge.next pointers are null). /// public void ResolveBoundaryEdges() { // Maps vertices to leaving boundary edge. var map = new Dictionary(); // TODO: parallel? foreach (var edge in this.edges) { if (edge.twin == null) { var twin = edge.twin = new HalfEdge(edge.next.origin, Face.Empty); twin.twin = edge; map.Add(twin.origin.id, twin); } } int j = edges.Count; foreach (var edge in map.Values) { edge.id = j++; edge.next = map[edge.twin.origin.id]; this.edges.Add(edge); } } /// /// Enumerates all edges of the DCEL. /// /// /// This method assumes that each half-edge has a twin (i.e. NOT null). /// protected virtual IEnumerable EnumerateEdges() { var edges = new List(this.edges.Count / 2); foreach (var edge in this.edges) { var twin = edge.twin; // Report edge only once. if (edge.id < twin.id) { edges.Add(new Edge(edge.origin.id, twin.origin.id)); } } return edges; } } }