Plane.cs 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. // MIT License - Copyright (C) The Mono.Xna Team
  2. // This file is subject to the terms and conditions defined in
  3. // file 'LICENSE.txt', which is part of this source code package.
  4. using System;
  5. using System.Diagnostics;
  6. using System.Runtime.Serialization;
  7. namespace CommonLang.Geometry
  8. {
  9. internal class PlaneHelper
  10. {
  11. /// <summary>
  12. /// Returns a value indicating what side (positive/negative) of a plane a point is
  13. /// </summary>
  14. /// <param name="point">The point to check with</param>
  15. /// <param name="plane">The plane to check against</param>
  16. /// <returns>Greater than zero if on the positive side, less than zero if on the negative size, 0 otherwise</returns>
  17. public static float ClassifyPoint(ref Vector3 point, ref Plane plane)
  18. {
  19. return point.X * plane.Normal.X + point.Y * plane.Normal.Y + point.Z * plane.Normal.Z + plane.D;
  20. }
  21. /// <summary>
  22. /// Returns the perpendicular distance from a point to a plane
  23. /// </summary>
  24. /// <param name="point">The point to check</param>
  25. /// <param name="plane">The place to check</param>
  26. /// <returns>The perpendicular distance from the point to the plane</returns>
  27. public static float PerpendicularDistance(ref Vector3 point, ref Plane plane)
  28. {
  29. // dist = (ax + by + cz + d) / sqrt(a*a + b*b + c*c)
  30. return (float)Math.Abs((plane.Normal.X * point.X + plane.Normal.Y * point.Y + plane.Normal.Z * point.Z)
  31. / Math.Sqrt(plane.Normal.X * plane.Normal.X + plane.Normal.Y * plane.Normal.Y + plane.Normal.Z * plane.Normal.Z));
  32. }
  33. }
  34. public struct Plane : IEquatable<Plane>
  35. {
  36. #region Public Fields
  37. public float D;
  38. public Vector3 Normal;
  39. #endregion Public Fields
  40. #region Constructors
  41. public Plane(Vector4 value)
  42. : this(new Vector3(value.X, value.Y, value.Z), value.W)
  43. {
  44. }
  45. public Plane(Vector3 normal, float d)
  46. {
  47. Normal = normal;
  48. D = d;
  49. }
  50. public Plane(Vector3 a, Vector3 b, Vector3 c)
  51. {
  52. Vector3 ab = b - a;
  53. Vector3 ac = c - a;
  54. Vector3 cross = Vector3.Cross(ab, ac);
  55. Normal = Vector3.Normalize(cross);
  56. D = -(Vector3.Dot(Normal, a));
  57. }
  58. public Plane(float a, float b, float c, float d)
  59. : this(new Vector3(a, b, c), d)
  60. {
  61. }
  62. #endregion Constructors
  63. #region Public Methods
  64. public float Dot(Vector4 value)
  65. {
  66. return ((((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + (this.D * value.W));
  67. }
  68. public void Dot(ref Vector4 value, out float result)
  69. {
  70. result = (((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + (this.D * value.W);
  71. }
  72. public float DotCoordinate(Vector3 value)
  73. {
  74. return ((((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + this.D);
  75. }
  76. public void DotCoordinate(ref Vector3 value, out float result)
  77. {
  78. result = (((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + this.D;
  79. }
  80. public float DotNormal(Vector3 value)
  81. {
  82. return (((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z));
  83. }
  84. public void DotNormal(ref Vector3 value, out float result)
  85. {
  86. result = ((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z);
  87. }
  88. /// <summary>
  89. /// Transforms a normalized plane by a matrix.
  90. /// </summary>
  91. /// <param name="plane">The normalized plane to transform.</param>
  92. /// <param name="matrix">The transformation matrix.</param>
  93. /// <returns>The transformed plane.</returns>
  94. public static Plane Transform(Plane plane, Matrix matrix)
  95. {
  96. Plane result;
  97. Transform(ref plane, ref matrix, out result);
  98. return result;
  99. }
  100. /// <summary>
  101. /// Transforms a normalized plane by a matrix.
  102. /// </summary>
  103. /// <param name="plane">The normalized plane to transform.</param>
  104. /// <param name="matrix">The transformation matrix.</param>
  105. /// <param name="result">The transformed plane.</param>
  106. public static void Transform(ref Plane plane, ref Matrix matrix, out Plane result)
  107. {
  108. // See "Transforming Normals" in http://www.glprogramming.com/red/appendixf.html
  109. // for an explanation of how this works.
  110. Matrix transformedMatrix;
  111. Matrix.Invert(ref matrix, out transformedMatrix);
  112. Matrix.Transpose(ref transformedMatrix, out transformedMatrix);
  113. var vector = new Vector4(plane.Normal, plane.D);
  114. Vector4 transformedVector;
  115. Vector4.Transform(ref vector, ref transformedMatrix, out transformedVector);
  116. result = new Plane(transformedVector);
  117. }
  118. /// <summary>
  119. /// Transforms a normalized plane by a quaternion rotation.
  120. /// </summary>
  121. /// <param name="plane">The normalized plane to transform.</param>
  122. /// <param name="rotation">The quaternion rotation.</param>
  123. /// <returns>The transformed plane.</returns>
  124. public static Plane Transform(Plane plane, Quaternion rotation)
  125. {
  126. Plane result;
  127. Transform(ref plane, ref rotation, out result);
  128. return result;
  129. }
  130. /// <summary>
  131. /// Transforms a normalized plane by a quaternion rotation.
  132. /// </summary>
  133. /// <param name="plane">The normalized plane to transform.</param>
  134. /// <param name="rotation">The quaternion rotation.</param>
  135. /// <param name="result">The transformed plane.</param>
  136. public static void Transform(ref Plane plane, ref Quaternion rotation, out Plane result)
  137. {
  138. Vector3.Transform(ref plane.Normal, ref rotation, out result.Normal);
  139. result.D = plane.D;
  140. }
  141. public void Normalize()
  142. {
  143. float factor;
  144. Vector3 normal = Normal;
  145. Normal = Vector3.Normalize(Normal);
  146. factor = (float)Math.Sqrt(Normal.X * Normal.X + Normal.Y * Normal.Y + Normal.Z * Normal.Z) /
  147. (float)Math.Sqrt(normal.X * normal.X + normal.Y * normal.Y + normal.Z * normal.Z);
  148. D = D * factor;
  149. }
  150. public static Plane Normalize(Plane value)
  151. {
  152. Plane ret;
  153. Normalize(ref value, out ret);
  154. return ret;
  155. }
  156. public static void Normalize(ref Plane value, out Plane result)
  157. {
  158. float factor;
  159. result.Normal = Vector3.Normalize(value.Normal);
  160. factor = (float)Math.Sqrt(result.Normal.X * result.Normal.X + result.Normal.Y * result.Normal.Y + result.Normal.Z * result.Normal.Z) /
  161. (float)Math.Sqrt(value.Normal.X * value.Normal.X + value.Normal.Y * value.Normal.Y + value.Normal.Z * value.Normal.Z);
  162. result.D = value.D * factor;
  163. }
  164. public static bool operator !=(Plane plane1, Plane plane2)
  165. {
  166. return !plane1.Equals(plane2);
  167. }
  168. public static bool operator ==(Plane plane1, Plane plane2)
  169. {
  170. return plane1.Equals(plane2);
  171. }
  172. public override bool Equals(object other)
  173. {
  174. return (other is Plane) ? this.Equals((Plane)other) : false;
  175. }
  176. public bool Equals(Plane other)
  177. {
  178. return ((Normal == other.Normal) && (D == other.D));
  179. }
  180. public override int GetHashCode()
  181. {
  182. return Normal.GetHashCode() ^ D.GetHashCode();
  183. }
  184. public PlaneIntersectionType Intersects(BoundingBox box)
  185. {
  186. return box.Intersects(this);
  187. }
  188. public void Intersects(ref BoundingBox box, out PlaneIntersectionType result)
  189. {
  190. box.Intersects (ref this, out result);
  191. }
  192. public PlaneIntersectionType Intersects(BoundingFrustum frustum)
  193. {
  194. return frustum.Intersects(this);
  195. }
  196. public PlaneIntersectionType Intersects(BoundingSphere sphere)
  197. {
  198. return sphere.Intersects(this);
  199. }
  200. public void Intersects(ref BoundingSphere sphere, out PlaneIntersectionType result)
  201. {
  202. sphere.Intersects(ref this, out result);
  203. }
  204. internal PlaneIntersectionType Intersects(ref Vector3 point)
  205. {
  206. float distance;
  207. DotCoordinate(ref point, out distance);
  208. if (distance > 0)
  209. return PlaneIntersectionType.Front;
  210. if (distance < 0)
  211. return PlaneIntersectionType.Back;
  212. return PlaneIntersectionType.Intersecting;
  213. }
  214. internal string DebugDisplayString
  215. {
  216. get
  217. {
  218. return string.Concat(
  219. this.Normal.DebugDisplayString, " ",
  220. this.D.ToString()
  221. );
  222. }
  223. }
  224. public override string ToString()
  225. {
  226. return "{Normal:" + Normal + " D:" + D + "}";
  227. }
  228. #endregion
  229. }
  230. }