123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 |
- // MIT License - Copyright (C) The Mono.Xna Team
- // This file is subject to the terms and conditions defined in
- // file 'LICENSE.txt', which is part of this source code package.
- using System;
- using System.Diagnostics;
- using System.Runtime.Serialization;
- namespace CommonLang.Geometry
- {
-
-
- public struct Ray : IEquatable<Ray>
- {
- #region Public Fields
-
- public Vector3 Direction;
-
-
- public Vector3 Position;
- #endregion
- #region Public Constructors
- public Ray(Vector3 position, Vector3 direction)
- {
- this.Position = position;
- this.Direction = direction;
- }
- #endregion
- #region Public Methods
- public override bool Equals(object obj)
- {
- return (obj is Ray) ? this.Equals((Ray)obj) : false;
- }
-
- public bool Equals(Ray other)
- {
- return this.Position.Equals(other.Position) && this.Direction.Equals(other.Direction);
- }
-
- public override int GetHashCode()
- {
- return Position.GetHashCode() ^ Direction.GetHashCode();
- }
- // adapted from http://www.scratchapixel.com/lessons/3d-basic-lessons/lesson-7-intersecting-simple-shapes/ray-box-intersection/
- public float? Intersects(BoundingBox box)
- {
- const float Epsilon = 1e-6f;
- float? tMin = null, tMax = null;
- if (Math.Abs(Direction.X) < Epsilon)
- {
- if (Position.X < box.Min.X || Position.X > box.Max.X)
- return null;
- }
- else
- {
- tMin = (box.Min.X - Position.X) / Direction.X;
- tMax = (box.Max.X - Position.X) / Direction.X;
- if (tMin > tMax)
- {
- var temp = tMin;
- tMin = tMax;
- tMax = temp;
- }
- }
- if (Math.Abs(Direction.Y) < Epsilon)
- {
- if (Position.Y < box.Min.Y || Position.Y > box.Max.Y)
- return null;
- }
- else
- {
- var tMinY = (box.Min.Y - Position.Y) / Direction.Y;
- var tMaxY = (box.Max.Y - Position.Y) / Direction.Y;
- if (tMinY > tMaxY)
- {
- var temp = tMinY;
- tMinY = tMaxY;
- tMaxY = temp;
- }
- if ((tMin.HasValue && tMin > tMaxY) || (tMax.HasValue && tMinY > tMax))
- return null;
- if (!tMin.HasValue || tMinY > tMin) tMin = tMinY;
- if (!tMax.HasValue || tMaxY < tMax) tMax = tMaxY;
- }
- if (Math.Abs(Direction.Z) < Epsilon)
- {
- if (Position.Z < box.Min.Z || Position.Z > box.Max.Z)
- return null;
- }
- else
- {
- var tMinZ = (box.Min.Z - Position.Z) / Direction.Z;
- var tMaxZ = (box.Max.Z - Position.Z) / Direction.Z;
- if (tMinZ > tMaxZ)
- {
- var temp = tMinZ;
- tMinZ = tMaxZ;
- tMaxZ = temp;
- }
- if ((tMin.HasValue && tMin > tMaxZ) || (tMax.HasValue && tMinZ > tMax))
- return null;
- if (!tMin.HasValue || tMinZ > tMin) tMin = tMinZ;
- if (!tMax.HasValue || tMaxZ < tMax) tMax = tMaxZ;
- }
- // having a positive tMin and a negative tMax means the ray is inside the box
- // we expect the intesection distance to be 0 in that case
- if ((tMin.HasValue && tMin < 0) && tMax > 0) return 0;
- // a negative tMin means that the intersection point is behind the ray's origin
- // we discard these as not hitting the AABB
- if (tMin < 0) return null;
- return tMin;
- }
- public void Intersects(ref BoundingBox box, out float? result)
- {
- result = Intersects(box);
- }
- /*
- public float? Intersects(BoundingFrustum frustum)
- {
- if (frustum == null)
- {
- throw new ArgumentNullException("frustum");
- }
-
- return frustum.Intersects(this);
- }
- */
- public float? Intersects(BoundingSphere sphere)
- {
- float? result;
- Intersects(ref sphere, out result);
- return result;
- }
- public float? Intersects(Plane plane)
- {
- float? result;
- Intersects(ref plane, out result);
- return result;
- }
- public void Intersects(ref Plane plane, out float? result)
- {
- var den = Vector3.Dot(Direction, plane.Normal);
- if (Math.Abs(den) < 0.00001f)
- {
- result = null;
- return;
- }
- result = (-plane.D - Vector3.Dot(plane.Normal, Position)) / den;
- if (result < 0.0f)
- {
- if (result < -0.00001f)
- {
- result = null;
- return;
- }
- result = 0.0f;
- }
- }
- public void Intersects(ref BoundingSphere sphere, out float? result)
- {
- // Find the vector between where the ray starts the the sphere's centre
- Vector3 difference = sphere.Center - this.Position;
- float differenceLengthSquared = difference.LengthSquared();
- float sphereRadiusSquared = sphere.Radius * sphere.Radius;
- float distanceAlongRay;
- // If the distance between the ray start and the sphere's centre is less than
- // the radius of the sphere, it means we've intersected. N.B. checking the LengthSquared is faster.
- if (differenceLengthSquared < sphereRadiusSquared)
- {
- result = 0.0f;
- return;
- }
- Vector3.Dot(ref this.Direction, ref difference, out distanceAlongRay);
- // If the ray is pointing away from the sphere then we don't ever intersect
- if (distanceAlongRay < 0)
- {
- result = null;
- return;
- }
- // Next we kinda use Pythagoras to check if we are within the bounds of the sphere
- // if x = radius of sphere
- // if y = distance between ray position and sphere centre
- // if z = the distance we've travelled along the ray
- // if x^2 + z^2 - y^2 < 0, we do not intersect
- float dist = sphereRadiusSquared + distanceAlongRay * distanceAlongRay - differenceLengthSquared;
- result = (dist < 0) ? null : distanceAlongRay - (float?)Math.Sqrt(dist);
- }
- public static bool operator !=(Ray a, Ray b)
- {
- return !a.Equals(b);
- }
- public static bool operator ==(Ray a, Ray b)
- {
- return a.Equals(b);
- }
- internal string DebugDisplayString
- {
- get
- {
- return string.Concat(
- "Pos( ", this.Position.DebugDisplayString, " ) \r\n",
- "Dir( ", this.Direction.DebugDisplayString, " )"
- );
- }
- }
- public override string ToString()
- {
- return "{{Position:" + Position.ToString() + " Direction:" + Direction.ToString() + "}}";
- }
-
- #endregion
- }
- }
|