Ray.cs 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  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. public struct Ray : IEquatable<Ray>
  10. {
  11. #region Public Fields
  12. public Vector3 Direction;
  13. public Vector3 Position;
  14. #endregion
  15. #region Public Constructors
  16. public Ray(Vector3 position, Vector3 direction)
  17. {
  18. this.Position = position;
  19. this.Direction = direction;
  20. }
  21. #endregion
  22. #region Public Methods
  23. public override bool Equals(object obj)
  24. {
  25. return (obj is Ray) ? this.Equals((Ray)obj) : false;
  26. }
  27. public bool Equals(Ray other)
  28. {
  29. return this.Position.Equals(other.Position) && this.Direction.Equals(other.Direction);
  30. }
  31. public override int GetHashCode()
  32. {
  33. return Position.GetHashCode() ^ Direction.GetHashCode();
  34. }
  35. // adapted from http://www.scratchapixel.com/lessons/3d-basic-lessons/lesson-7-intersecting-simple-shapes/ray-box-intersection/
  36. public float? Intersects(BoundingBox box)
  37. {
  38. const float Epsilon = 1e-6f;
  39. float? tMin = null, tMax = null;
  40. if (Math.Abs(Direction.X) < Epsilon)
  41. {
  42. if (Position.X < box.Min.X || Position.X > box.Max.X)
  43. return null;
  44. }
  45. else
  46. {
  47. tMin = (box.Min.X - Position.X) / Direction.X;
  48. tMax = (box.Max.X - Position.X) / Direction.X;
  49. if (tMin > tMax)
  50. {
  51. var temp = tMin;
  52. tMin = tMax;
  53. tMax = temp;
  54. }
  55. }
  56. if (Math.Abs(Direction.Y) < Epsilon)
  57. {
  58. if (Position.Y < box.Min.Y || Position.Y > box.Max.Y)
  59. return null;
  60. }
  61. else
  62. {
  63. var tMinY = (box.Min.Y - Position.Y) / Direction.Y;
  64. var tMaxY = (box.Max.Y - Position.Y) / Direction.Y;
  65. if (tMinY > tMaxY)
  66. {
  67. var temp = tMinY;
  68. tMinY = tMaxY;
  69. tMaxY = temp;
  70. }
  71. if ((tMin.HasValue && tMin > tMaxY) || (tMax.HasValue && tMinY > tMax))
  72. return null;
  73. if (!tMin.HasValue || tMinY > tMin) tMin = tMinY;
  74. if (!tMax.HasValue || tMaxY < tMax) tMax = tMaxY;
  75. }
  76. if (Math.Abs(Direction.Z) < Epsilon)
  77. {
  78. if (Position.Z < box.Min.Z || Position.Z > box.Max.Z)
  79. return null;
  80. }
  81. else
  82. {
  83. var tMinZ = (box.Min.Z - Position.Z) / Direction.Z;
  84. var tMaxZ = (box.Max.Z - Position.Z) / Direction.Z;
  85. if (tMinZ > tMaxZ)
  86. {
  87. var temp = tMinZ;
  88. tMinZ = tMaxZ;
  89. tMaxZ = temp;
  90. }
  91. if ((tMin.HasValue && tMin > tMaxZ) || (tMax.HasValue && tMinZ > tMax))
  92. return null;
  93. if (!tMin.HasValue || tMinZ > tMin) tMin = tMinZ;
  94. if (!tMax.HasValue || tMaxZ < tMax) tMax = tMaxZ;
  95. }
  96. // having a positive tMin and a negative tMax means the ray is inside the box
  97. // we expect the intesection distance to be 0 in that case
  98. if ((tMin.HasValue && tMin < 0) && tMax > 0) return 0;
  99. // a negative tMin means that the intersection point is behind the ray's origin
  100. // we discard these as not hitting the AABB
  101. if (tMin < 0) return null;
  102. return tMin;
  103. }
  104. public void Intersects(ref BoundingBox box, out float? result)
  105. {
  106. result = Intersects(box);
  107. }
  108. /*
  109. public float? Intersects(BoundingFrustum frustum)
  110. {
  111. if (frustum == null)
  112. {
  113. throw new ArgumentNullException("frustum");
  114. }
  115. return frustum.Intersects(this);
  116. }
  117. */
  118. public float? Intersects(BoundingSphere sphere)
  119. {
  120. float? result;
  121. Intersects(ref sphere, out result);
  122. return result;
  123. }
  124. public float? Intersects(Plane plane)
  125. {
  126. float? result;
  127. Intersects(ref plane, out result);
  128. return result;
  129. }
  130. public void Intersects(ref Plane plane, out float? result)
  131. {
  132. var den = Vector3.Dot(Direction, plane.Normal);
  133. if (Math.Abs(den) < 0.00001f)
  134. {
  135. result = null;
  136. return;
  137. }
  138. result = (-plane.D - Vector3.Dot(plane.Normal, Position)) / den;
  139. if (result < 0.0f)
  140. {
  141. if (result < -0.00001f)
  142. {
  143. result = null;
  144. return;
  145. }
  146. result = 0.0f;
  147. }
  148. }
  149. public void Intersects(ref BoundingSphere sphere, out float? result)
  150. {
  151. // Find the vector between where the ray starts the the sphere's centre
  152. Vector3 difference = sphere.Center - this.Position;
  153. float differenceLengthSquared = difference.LengthSquared();
  154. float sphereRadiusSquared = sphere.Radius * sphere.Radius;
  155. float distanceAlongRay;
  156. // If the distance between the ray start and the sphere's centre is less than
  157. // the radius of the sphere, it means we've intersected. N.B. checking the LengthSquared is faster.
  158. if (differenceLengthSquared < sphereRadiusSquared)
  159. {
  160. result = 0.0f;
  161. return;
  162. }
  163. Vector3.Dot(ref this.Direction, ref difference, out distanceAlongRay);
  164. // If the ray is pointing away from the sphere then we don't ever intersect
  165. if (distanceAlongRay < 0)
  166. {
  167. result = null;
  168. return;
  169. }
  170. // Next we kinda use Pythagoras to check if we are within the bounds of the sphere
  171. // if x = radius of sphere
  172. // if y = distance between ray position and sphere centre
  173. // if z = the distance we've travelled along the ray
  174. // if x^2 + z^2 - y^2 < 0, we do not intersect
  175. float dist = sphereRadiusSquared + distanceAlongRay * distanceAlongRay - differenceLengthSquared;
  176. result = (dist < 0) ? null : distanceAlongRay - (float?)Math.Sqrt(dist);
  177. }
  178. public static bool operator !=(Ray a, Ray b)
  179. {
  180. return !a.Equals(b);
  181. }
  182. public static bool operator ==(Ray a, Ray b)
  183. {
  184. return a.Equals(b);
  185. }
  186. internal string DebugDisplayString
  187. {
  188. get
  189. {
  190. return string.Concat(
  191. "Pos( ", this.Position.DebugDisplayString, " ) \r\n",
  192. "Dir( ", this.Direction.DebugDisplayString, " )"
  193. );
  194. }
  195. }
  196. public override string ToString()
  197. {
  198. return "{{Position:" + Position.ToString() + " Direction:" + Direction.ToString() + "}}";
  199. }
  200. #endregion
  201. }
  202. }