-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFace2.cs
182 lines (167 loc) · 4.93 KB
/
Face2.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
using System;
using System.Text;
/// <summary>
/// Organizes the components of a 2D mesh into a group of edges.
/// This is not used by a mesh internally; it is created upon
/// retrieval from a mesh.
/// </summary>
[Serializable]
public readonly struct Face2 : IComparable<Face2>, IEquatable<Face2>
{
/// <summary>
/// The edges array.
/// </summary>
private readonly Edge2[] edges;
/// <summary>
/// Constructs a face from an array of edges.
/// </summary>
/// <param name="edges">edges array</param>
public Face2(in Edge2[] edges) { this.edges = edges; }
/// <summary>
/// The number of edges in this face.
/// </summary>
/// <value>length</value>
public int Length { get { return this.edges.Length; } }
/// <summary>
/// Gets an edge at an index. Wraps around, so
/// negative indices may be used.
/// </summary>
/// <value>the edge</value>
public Edge2 this[int i]
{
get
{
return this.edges[Utils.RemFloor(i, this.edges.Length)];
}
}
/// <summary>
/// Tests this face for equivalence with an object.
/// </summary>
/// <param name="value">the object</param>
/// <returns>equivalence</returns>
public override bool Equals(object value)
{
if (Object.ReferenceEquals(this, value)) { return true; }
if (value is null) { return false; }
if (value is Vert3) { return this.Equals((Face2)value); }
return false;
}
/// <summary>
/// Returns a hash code representing this face.
/// </summary>
/// <returns>hash code</returns>
public override int GetHashCode()
{
return this.edges.GetHashCode();
}
/// <summary>
/// Returns a string representation of this face.
/// </summary>
/// <returns>string</returns>
public override string ToString()
{
return Face2.ToString(this);
}
/// <summary>
/// Compares this face to another in compliance with the IComparable
/// interface.
/// </summary>
/// <param name="f">comparisand</param>
/// <returns>evaluation</returns>
public int CompareTo(Face2 f)
{
return Face2.Center(this).CompareTo(Face2.Center(f));
}
/// <summary>
/// Tests this face for equivalence with another in compliance with the
/// IEquatable interface.
/// </summary>
/// <param name="f">face</param>
/// <returns>equivalence</returns>
public bool Equals(Face2 f)
{
return this.edges.Equals(f.edges);
}
/// <summary>
/// Finds the center of a face.
/// </summary>
/// <param name="f">face</param>
/// <returns>center</returns>
public static Vec2 Center(in Face2 f)
{
return Face2.CenterMean(f);
}
/// <summary>
/// Finds the mean center of a face.
/// Sums the origin coordinate of its edges,
/// then finds the average.
/// </summary>
/// <param name="f">face</param>
/// <returns>center</returns>
public static Vec2 CenterMean(in Face2 f)
{
int len = f.edges.Length;
Vec2 sum = Vec2.Zero;
for (int i = 0; i < len; ++i)
{
sum += f.edges[i].Origin.Coord;
}
return sum / len;
}
/// <summary>
/// Finds a point on a face's perimeter given a factor
/// in [0.0, 1.0].
/// </summary>
/// <param name="f">face</param>
/// <param name="t">factor</param>
/// <returns>point</returns>
public static Vec2 Eval(in Face2 f, in float t)
{
float tScaled = f.Length * Utils.RemFloor(t, 1.0f);
int i = (int)tScaled;
return Edge2.Eval(f[i], tScaled - i);
}
/// <summary>
/// Finds the perimeter of a face. Sums the magnitude
/// of its edges.
/// </summary>
/// <param name="f">face</param>
/// <returns>perimeter</returns>
public static float Perimeter(in Face2 f)
{
int len = f.Length;
float sum = 0.0f;
for (int i = 0; i < len; ++i)
{
sum += Edge2.Mag(f[i]);
}
return sum;
}
/// <summary>
/// Returns a string representation of a face.
/// </summary>
/// <param name="f">face</param>
/// <param name="places">number of decimal places</param>
/// <returns>string</returns>
public static string ToString(in Face2 f, in int places = 4)
{
return Face2.ToString(new StringBuilder(1024), f, places).ToString();
}
/// <summary>
/// Appends a representation of an face to a string builder.
/// </summary>
/// <param name="sb">string builder</param>
/// <param name="f">face</param>
/// <param name="places">number of decimal places</param>
/// <returns>string builder</returns>
public static StringBuilder ToString(
in StringBuilder sb,
in Face2 f,
in int places = 4)
{
sb.Append("{\"edges\":");
Edge2.ToString(sb, f.edges, places);
sb.Append('}');
return sb;
}
}