rj-action-library/Runtime/Math/Geometry/Plane3.cs

131 lines
2.8 KiB
C#

using Godot;
namespace Rokojori
{
public class Plane3
{
public Vector3 normal = Vector3.Up;
public float constant = 0;
public Plane3(){}
public static Plane3 CreateFromNode3D( Node3D n )
{
var p = new Plane3();
p.SetFromNormalAndCoplanarPoint( n.GlobalUp(), n.GlobalPosition );
return p;
}
public void Set( Vector3 normal, float constant = 0 )
{
this.normal = normal;
this.constant = constant;
}
public Vector3 ConstrainToPlane( Vector3 p, float distance = 10000 )
{
var line3 = new Line3();
line3.Set( p - normal * distance, p + normal * distance );
var intersection = IntersectLine( line3 );
if ( intersection == null )
{
return p;
}
return (Vector3) intersection;
}
public Vector3 ClosestPointTo( Vector3 p )
{
return ConstrainToPlane( p );
}
public float DistanceTo( Vector3 p, float normalIntersectionDistance = 5000 )
{
var line = Line3.Create( p - normal * normalIntersectionDistance, p + normal * normalIntersectionDistance );
var isc = IntersectLine( line );
if ( isc == null )
{
return float.PositiveInfinity;
}
var xp = (Vector3)isc;
return ( p - xp ).Length();
}
public Ray3 GetIntersectionRay( Plane3 other )
{
var lineDirection = normal.Cross( other.normal );
var det = lineDirection.LengthSquared();
if ( det == 0 )
{
return null;
}
var r = new Ray3();
r.offset = ( ( lineDirection.Cross( other.normal ) * constant ) +
( normal.Cross( lineDirection ) * other.constant ) ) / det;
r.direction = lineDirection;
return r;
}
public Vector3? IntersectLine( Line3 line )
{
var direction = line.direction;
var denominator = Math3D.Dot( normal, direction );
if ( denominator == 0 )
{
return null;
}
var t = - ( Math3D.Dot( line.start, normal ) + constant ) / denominator;
if ( t < 0 || t > 1 )
{
return null;
}
return direction * t + line.start;
}
public static Plane3 GetZeroUp()
{
var plane = new Plane3();
plane.Set( Vector3.Up, 0 );
return plane;
}
public void SetFromNormalAndCoplanarPoint( Vector3 normal, Vector3 point )
{
this.normal = normal;
this.constant = - Math3D.Dot( point, this.normal );
}
public void SetFromCoplanarPoints( Vector3 a, Vector3 b, Vector3 c )
{
var normal = Math3D.ComputeNormal( a, b, c );
SetFromNormalAndCoplanarPoint( normal, a );
}
public static Plane3 FromCoplanarPoints( Vector3 a, Vector3 b, Vector3 c )
{
var p = new Plane3();
p.SetFromCoplanarPoints( a, b, c );
return p;
}
}
}