123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 |
- // 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
- {
- internal class PlaneHelper
- {
- /// <summary>
- /// Returns a value indicating what side (positive/negative) of a plane a point is
- /// </summary>
- /// <param name="point">The point to check with</param>
- /// <param name="plane">The plane to check against</param>
- /// <returns>Greater than zero if on the positive side, less than zero if on the negative size, 0 otherwise</returns>
- public static float ClassifyPoint(ref Vector3 point, ref Plane plane)
- {
- return point.X * plane.Normal.X + point.Y * plane.Normal.Y + point.Z * plane.Normal.Z + plane.D;
- }
- /// <summary>
- /// Returns the perpendicular distance from a point to a plane
- /// </summary>
- /// <param name="point">The point to check</param>
- /// <param name="plane">The place to check</param>
- /// <returns>The perpendicular distance from the point to the plane</returns>
- public static float PerpendicularDistance(ref Vector3 point, ref Plane plane)
- {
- // dist = (ax + by + cz + d) / sqrt(a*a + b*b + c*c)
- return (float)Math.Abs((plane.Normal.X * point.X + plane.Normal.Y * point.Y + plane.Normal.Z * point.Z)
- / Math.Sqrt(plane.Normal.X * plane.Normal.X + plane.Normal.Y * plane.Normal.Y + plane.Normal.Z * plane.Normal.Z));
- }
- }
-
-
-
- public struct Plane : IEquatable<Plane>
- {
- #region Public Fields
-
- public float D;
-
- public Vector3 Normal;
- #endregion Public Fields
- #region Constructors
- public Plane(Vector4 value)
- : this(new Vector3(value.X, value.Y, value.Z), value.W)
- {
- }
- public Plane(Vector3 normal, float d)
- {
- Normal = normal;
- D = d;
- }
- public Plane(Vector3 a, Vector3 b, Vector3 c)
- {
- Vector3 ab = b - a;
- Vector3 ac = c - a;
- Vector3 cross = Vector3.Cross(ab, ac);
- Normal = Vector3.Normalize(cross);
- D = -(Vector3.Dot(Normal, a));
- }
- public Plane(float a, float b, float c, float d)
- : this(new Vector3(a, b, c), d)
- {
- }
- #endregion Constructors
- #region Public Methods
- public float Dot(Vector4 value)
- {
- return ((((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + (this.D * value.W));
- }
- public void Dot(ref Vector4 value, out float result)
- {
- result = (((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + (this.D * value.W);
- }
- public float DotCoordinate(Vector3 value)
- {
- return ((((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + this.D);
- }
- public void DotCoordinate(ref Vector3 value, out float result)
- {
- result = (((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z)) + this.D;
- }
- public float DotNormal(Vector3 value)
- {
- return (((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z));
- }
- public void DotNormal(ref Vector3 value, out float result)
- {
- result = ((this.Normal.X * value.X) + (this.Normal.Y * value.Y)) + (this.Normal.Z * value.Z);
- }
- /// <summary>
- /// Transforms a normalized plane by a matrix.
- /// </summary>
- /// <param name="plane">The normalized plane to transform.</param>
- /// <param name="matrix">The transformation matrix.</param>
- /// <returns>The transformed plane.</returns>
- public static Plane Transform(Plane plane, Matrix matrix)
- {
- Plane result;
- Transform(ref plane, ref matrix, out result);
- return result;
- }
- /// <summary>
- /// Transforms a normalized plane by a matrix.
- /// </summary>
- /// <param name="plane">The normalized plane to transform.</param>
- /// <param name="matrix">The transformation matrix.</param>
- /// <param name="result">The transformed plane.</param>
- public static void Transform(ref Plane plane, ref Matrix matrix, out Plane result)
- {
- // See "Transforming Normals" in http://www.glprogramming.com/red/appendixf.html
- // for an explanation of how this works.
- Matrix transformedMatrix;
- Matrix.Invert(ref matrix, out transformedMatrix);
- Matrix.Transpose(ref transformedMatrix, out transformedMatrix);
- var vector = new Vector4(plane.Normal, plane.D);
- Vector4 transformedVector;
- Vector4.Transform(ref vector, ref transformedMatrix, out transformedVector);
- result = new Plane(transformedVector);
- }
- /// <summary>
- /// Transforms a normalized plane by a quaternion rotation.
- /// </summary>
- /// <param name="plane">The normalized plane to transform.</param>
- /// <param name="rotation">The quaternion rotation.</param>
- /// <returns>The transformed plane.</returns>
- public static Plane Transform(Plane plane, Quaternion rotation)
- {
- Plane result;
- Transform(ref plane, ref rotation, out result);
- return result;
- }
- /// <summary>
- /// Transforms a normalized plane by a quaternion rotation.
- /// </summary>
- /// <param name="plane">The normalized plane to transform.</param>
- /// <param name="rotation">The quaternion rotation.</param>
- /// <param name="result">The transformed plane.</param>
- public static void Transform(ref Plane plane, ref Quaternion rotation, out Plane result)
- {
- Vector3.Transform(ref plane.Normal, ref rotation, out result.Normal);
- result.D = plane.D;
- }
- public void Normalize()
- {
- float factor;
- Vector3 normal = Normal;
- Normal = Vector3.Normalize(Normal);
- factor = (float)Math.Sqrt(Normal.X * Normal.X + Normal.Y * Normal.Y + Normal.Z * Normal.Z) /
- (float)Math.Sqrt(normal.X * normal.X + normal.Y * normal.Y + normal.Z * normal.Z);
- D = D * factor;
- }
- public static Plane Normalize(Plane value)
- {
- Plane ret;
- Normalize(ref value, out ret);
- return ret;
- }
- public static void Normalize(ref Plane value, out Plane result)
- {
- float factor;
- result.Normal = Vector3.Normalize(value.Normal);
- factor = (float)Math.Sqrt(result.Normal.X * result.Normal.X + result.Normal.Y * result.Normal.Y + result.Normal.Z * result.Normal.Z) /
- (float)Math.Sqrt(value.Normal.X * value.Normal.X + value.Normal.Y * value.Normal.Y + value.Normal.Z * value.Normal.Z);
- result.D = value.D * factor;
- }
- public static bool operator !=(Plane plane1, Plane plane2)
- {
- return !plane1.Equals(plane2);
- }
- public static bool operator ==(Plane plane1, Plane plane2)
- {
- return plane1.Equals(plane2);
- }
- public override bool Equals(object other)
- {
- return (other is Plane) ? this.Equals((Plane)other) : false;
- }
- public bool Equals(Plane other)
- {
- return ((Normal == other.Normal) && (D == other.D));
- }
- public override int GetHashCode()
- {
- return Normal.GetHashCode() ^ D.GetHashCode();
- }
- public PlaneIntersectionType Intersects(BoundingBox box)
- {
- return box.Intersects(this);
- }
- public void Intersects(ref BoundingBox box, out PlaneIntersectionType result)
- {
- box.Intersects (ref this, out result);
- }
- public PlaneIntersectionType Intersects(BoundingFrustum frustum)
- {
- return frustum.Intersects(this);
- }
- public PlaneIntersectionType Intersects(BoundingSphere sphere)
- {
- return sphere.Intersects(this);
- }
- public void Intersects(ref BoundingSphere sphere, out PlaneIntersectionType result)
- {
- sphere.Intersects(ref this, out result);
- }
- internal PlaneIntersectionType Intersects(ref Vector3 point)
- {
- float distance;
- DotCoordinate(ref point, out distance);
- if (distance > 0)
- return PlaneIntersectionType.Front;
- if (distance < 0)
- return PlaneIntersectionType.Back;
- return PlaneIntersectionType.Intersecting;
- }
- internal string DebugDisplayString
- {
- get
- {
- return string.Concat(
- this.Normal.DebugDisplayString, " ",
- this.D.ToString()
- );
- }
- }
- public override string ToString()
- {
- return "{Normal:" + Normal + " D:" + D + "}";
- }
- #endregion
- }
- }
|