// ----------------------------------------------------------------------- // // 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/ // // ----------------------------------------------------------------------- namespace TriangleNet.Topology { using System; using TriangleNet.Geometry; /// /// An oriented subsegment. /// /// /// Includes a pointer to a subsegment and an orientation. The orientation denotes a /// side of the edge. Hence, there are two possible orientations. By convention, the /// edge is always directed so that the "side" denoted is the right side of the edge. /// public struct Osub { internal SubSegment seg; internal int orient; // Ranges from 0 to 1. public SubSegment Segment { get { return seg; } } public override string ToString() { if (seg == null) { return "O-TID [null]"; } return String.Format("O-SID {0}", seg.hash); } #region Osub primitives /// /// Reverse the orientation of a subsegment. [sym(ab) -> ba] /// public void Sym(ref Osub os) { os.seg = seg; os.orient = 1 - orient; } /// /// Reverse the orientation of a subsegment. [sym(ab) -> ba] /// public void Sym() { orient = 1 - orient; } /// /// Find adjoining subsegment with the same origin. [pivot(ab) -> a*] /// /// spivot() finds the other subsegment (from the same segment) /// that shares the same origin. /// public void Pivot(ref Osub os) { os = seg.subsegs[orient]; } /// /// Finds a triangle abutting a subsegment. /// internal void Pivot(ref Otri ot) { ot = seg.triangles[orient]; } /// /// Find next subsegment in sequence. [next(ab) -> b*] /// public void Next(ref Osub ot) { ot = seg.subsegs[1 - orient]; } /// /// Find next subsegment in sequence. [next(ab) -> b*] /// public void Next() { this = seg.subsegs[1 - orient]; } /// /// Get the origin of a subsegment /// public Vertex Org() { return seg.vertices[orient]; } /// /// Get the destination of a subsegment /// public Vertex Dest() { return seg.vertices[1 - orient]; } #endregion #region Osub primitives (internal) /// /// Set the origin or destination of a subsegment. /// internal void SetOrg(Vertex vertex) { seg.vertices[orient] = vertex; } /// /// Set destination of a subsegment. /// internal void SetDest(Vertex vertex) { seg.vertices[1 - orient] = vertex; } /// /// Get the origin of the segment that includes the subsegment. /// internal Vertex SegOrg() { return seg.vertices[2 + orient]; } /// /// Get the destination of the segment that includes the subsegment. /// internal Vertex SegDest() { return seg.vertices[3 - orient]; } /// /// Set the origin of the segment that includes the subsegment. /// internal void SetSegOrg(Vertex vertex) { seg.vertices[2 + orient] = vertex; } /// /// Set the destination of the segment that includes the subsegment. /// internal void SetSegDest(Vertex vertex) { seg.vertices[3 - orient] = vertex; } /* Unused primitives. /// /// Find adjoining subsegment with the same origin. [pivot(ab) -> a*] /// public void PivotSelf() { this = seg.subsegs[orient]; } /// /// Read a boundary marker. /// /// Boundary markers are used to hold user-defined tags for /// setting boundary conditions in finite element solvers. public int Mark() { return seg.boundary; } /// /// Set a boundary marker. /// public void SetMark(int value) { seg.boundary = value; } /// /// Copy a subsegment. /// public void Copy(ref Osub o2) { o2.seg = seg; o2.orient = orient; } //*/ /// /// Bond two subsegments together. [bond(abc, ba)] /// internal void Bond(ref Osub os) { seg.subsegs[orient] = os; os.seg.subsegs[os.orient] = this; } /// /// Dissolve a subsegment bond (from one side). /// /// Note that the other subsegment will still think it's /// connected to this subsegment. internal void Dissolve(SubSegment dummy) { seg.subsegs[orient].seg = dummy; } /// /// Test for equality of subsegments. /// internal bool Equal(Osub os) { return ((seg == os.seg) && (orient == os.orient)); } /// /// Dissolve a bond (from the subsegment side). /// internal void TriDissolve(Triangle dummy) { seg.triangles[orient].tri = dummy; } /// /// Check a subsegment's deallocation. /// internal static bool IsDead(SubSegment sub) { return sub.subsegs[0].seg == null; } /// /// Set a subsegment's deallocation. /// internal static void Kill(SubSegment sub) { sub.subsegs[0].seg = null; sub.subsegs[1].seg = null; } #endregion } }