using Godot; using System.Collections; using System.Collections.Generic; using Godot.Collections; namespace Rokojori { [GlobalClass] public partial class MultiRayCaster:Caster { [Export] public float rayLength = 10; [Export] public int maxHits = 100; [Export] public bool sortByPointerPriority; List collisions = new List(); int numCollisions = 0; [Export] public Vector3 to; public override int NumColliders() { return numCollisions; } public override Node GetCollider( int index ) { return collisions[ index ].collider; } public override Vector3 GetCollisionNormal( int index ) { return collisions[ index ].normal; } public override Vector3 GetCollisionPosition( int index ) { return collisions[ index ].position; } public override Shape3D GetCollisionShape( int index ) { return collisions[ index ].shape; } PhysicsRayQueryParameters3D rayParameters = new PhysicsRayQueryParameters3D(); public override void _Process( double delta ) { Action.Trigger( beforeProcess ); ResolveCollisions(); SortCollisions(); Action.Trigger( afterProcess ); } ValueSorter singleSorter; MultiValueSorter multiSorter; CollisionData GetCollisionData( int i ) { while ( collisions.Count <= i ) { collisions.Add( new CollisionData() ); } return collisions[ i ]; } void ResolveCollisions() { var physics = GetWorld3D().DirectSpaceState; var excludes = new Array(); var from = GlobalPosition; var to = from + Math3D.GetGlobalForward( this ) * rayLength; this.to = to; rayParameters.From = from; rayParameters.To = to; numCollisions = 0; for ( int i = 0; i < maxHits; i++ ) { rayParameters.Exclude = excludes; var collisionData = GetCollisionData( numCollisions ); collisionData.Get( rayParameters, physics ); if ( ! collisionData.hasCollision ) { return; } excludes.Add( collisionData.rid ); if ( IsSelected( collisionData ) ) { numCollisions ++; } } } bool IsSelected( CollisionData cd ) { if ( includeSelector != null && ! includeSelector.Selects( cd.collider ) ) { return false; } if ( excludeSelector != null && excludeSelector.Selects( cd.collider ) ) { return false; } return true; } void SortCollisions() { if ( ! sortByPointerPriority ) { if ( singleSorter == null ) { singleSorter = ValueSorter.Create( cd => GetDistance( cd ) ); } singleSorter.Sort( 0, numCollisions, collisions ); } else { if ( multiSorter == null ) { multiSorter = MultiValueSorter.Create( cd => GetPointablePriority( cd ), cd => GetDistance( cd ) ); } multiSorter.Sort( 0, numCollisions, collisions ); } } float GetPointablePriority( CollisionData cd ) { float priority = 0; var pointable = Nodes.Find( cd.collider ); if ( pointable != null ) { priority = pointable.pointingPriority; } return priority; } float GetDistance( CollisionData cd ) { return ( cd.position - GlobalPosition ).Length(); } } }