rj-action-library/Runtime/Procedural/Parametric/Spline/Spline.cs

300 lines
6.2 KiB
C#

using Godot;
using Rokojori;
using System.Collections.Generic;
namespace Rokojori
{
public enum SplineAutoOrientationMode
{
Tangent,
Next_Neighbor
}
[Tool]
[GlobalClass, Icon("res://addons/rokojori_action_library/Icons/Spline.svg") ]
public partial class Spline : Node3D
#if TOOLS
, GizmoDrawer
#endif
{
[Export]
public bool closed = false;
[Export]
public int editorResolution = 20;
[Export]
public bool autoOrienation = false;
[Export]
public SplineAutoOrientationMode autoOrientationMode = SplineAutoOrientationMode.Tangent;
[Export( PropertyHint.Range, "-1,1")]
public float autoOrientationTangentAdjustment = 0f;
[Export]
public Vector3 autoOrientationUp = Vector3.Up;
[Export]
public bool updateAlways = false;
[Export]
public int xzPathBakeResolution = 50;
SplineCurve splineCurve;
Vector3 min;
Vector3 max;
public Box3 GetBounds()
{
GetCurve();
return Box3.Create( min, max );
}
public SplineCurve GetCurve()
{
if ( splineCurve == null )
{
var splinePoints = Nodes.GetDirectChildren<SplinePoint>( this );
splineCurve = new SplineCurveCreator().Create( splinePoints, closed );
min = splineCurve.MinPointPosition();
max = splineCurve.MaxPointPosition();
}
return splineCurve;
}
SplineCurve splineCurveXZ;
public SplineCurve GetCurveXZ()
{
if ( splineCurveXZ == null )
{
var splinePoints = Nodes.GetDirectChildren<SplinePoint>( this );
splineCurveXZ = new SplineCurveCreator().Create( splinePoints, closed ).CloneForXZ( 0 );
}
return splineCurveXZ;
}
Path2 xzPath;
Convex2Group convex2Group;
public Path2 GetXZPath2()
{
if ( xzPath == null )
{
xzPath = Path2.AsXZFrom( GetCurve(), closed, xzPathBakeResolution );
}
return xzPath;
}
public Convex2Group GetConvex2Group()
{
if ( convex2Group == null )
{
convex2Group = Convex2Group.FromPath( GetXZPath2() );
}
return convex2Group;
}
public void ClearCurveCache()
{
splineCurve = null;
splineCurveXZ = null;
xzPath = null;
convex2Group = null;
}
#if TOOLS
public void DrawGizmo( EditorNode3DGizmoPlugin gizmoPlugin, EditorNode3DGizmo gizmo )
{
ClearCurveCache();
var curve = GetCurve();
gizmo.Clear();
if ( curve.points.Count <= 1 )
{
return;
}
var linePoints = new List<Vector3>();
int resolution = editorResolution <= 0 ? 20 : editorResolution;
renderedResolution = editorResolution;
var lastPoint = ToLocal( curve.PositionAt( 0 ) );
for ( int i = 1; i < resolution; i++ )
{
var t = i / (float) (resolution - 1 );
var point = ToLocal( curve.PositionAt( t ) );
linePoints.Add( lastPoint );
linePoints.Add( point );
lastPoint = point;
}
for ( int i = 0; i < curve.points.Count; i++ )
{
var p = curve.points[ i ];
linePoints.Add( ToLocal( p.position ) );
linePoints.Add( ToLocal( p.tangentBefore.position ) );
linePoints.Add( ToLocal( p.position ) );
linePoints.Add( ToLocal( p.tangentNext.position ) );
}
var material = gizmoPlugin.GetMaterial( "main", gizmo );
gizmo.AddLines( linePoints.ToArray(), material, false );
}
List<Vector3> cachedPositions = new List<Vector3>();
int renderedResolution = -100;
public override void _Process( double delta )
{
var changed = updateAlways;
var index = 0;
if ( renderedResolution != editorResolution )
{
changed = true;
}
if ( autoOrienation )
{
AutoOrientate();
}
Nodes.ForEachDirectChild<SplinePoint>( this,
sp =>
{
if ( changed )
{
return;
}
if ( index >= cachedPositions.Count )
{
changed = true;
return;
}
if ( cachedPositions[ index ] != sp.GlobalPosition )
{
changed = true;
}
index ++;
}
);
if ( ! changed )
{
return;
}
cachedPositions = Nodes.MapDirectChildren<SplinePoint,Vector3>( this, sp => sp.GlobalPosition );
// RJLog.Log( "Updating gizmos" );
UpdateGizmos();
}
#endif
void AutoOrientate()
{
var list = Nodes.GetDirectChildren<SplinePoint>( this );
if ( list.Count <= 1 )
{
return;
}
var up = autoOrientationUp.Normalized();
if ( SplineAutoOrientationMode.Next_Neighbor == autoOrientationMode )
{
for ( int i = 0; i < list.Count; i++ )
{
var point = list[ i ];
if ( i == ( list.Count - 1 ) )
{
if ( closed )
{
var first = list[ 0 ];
var firstDirection = ( first.GlobalPosition - point.GlobalPosition ).Normalized();
point.LookTowards( firstDirection, up);
}
continue;
}
var next = list[ i + 1 ];
var direction = ( next.GlobalPosition - point.GlobalPosition ).Normalized();
point.LookTowards( direction, up );
}
}
else if ( SplineAutoOrientationMode.Tangent == autoOrientationMode )
{
for ( int i = 0; i < list.Count; i++ )
{
var point = list[ i ];
var tangentForward = Vector3.Zero;
if ( i == ( list.Count - 1 ) && ! closed )
{
tangentForward = - SplineCurveCreator.GetTangentDirectionSmoothed( 1, autoOrientationTangentAdjustment, list, i, true, false );
}
else
{
tangentForward = SplineCurveCreator.GetTangentDirectionSmoothed( 1, autoOrientationTangentAdjustment, list, i, false, closed );
}
if ( tangentForward.Length() == 0 )
{
continue;
}
point.LookTowards( tangentForward, up );
}
}
}
}
}