rj-action-library/Runtime/Godot/Nodes.cs

530 lines
11 KiB
C#

using Godot;
using System.Collections.Generic;
using System;
using System.Threading.Tasks;
namespace Rokojori
{
public static class Nodes
{
public static T Find<T>( Node root, NodePathLocatorType type = NodePathLocatorType.DirectChildren, int parentOffset = 0 ) where T:Node
{
var it = root;
while ( parentOffset > 0 && it != null )
{
it = it.GetParent();
parentOffset --;
}
if ( it == null )
{
return default(T);
}
switch ( type )
{
case NodePathLocatorType.DirectChildren:
{
return GetDirectChild<T>( it );
}
case NodePathLocatorType.Siblings:
{
return GetSibling<T>( it );
}
case NodePathLocatorType.DirectChildrenAndSiblings:
{
var child = GetDirectChild<T>( it );
if ( child != null )
{
return child;
}
return GetSibling<T>( it );
}
case NodePathLocatorType.AnyChildren:
{
return GetAnyChild<T>( it );
}
case NodePathLocatorType.SiblingsAndAnyChildren:
{
var sibling = GetSibling<T>( it );
if ( sibling != null )
{
return sibling;
}
return GetAnyChild<T>( it );;
}
}
return default(T);
}
public static void ForEachInScene<T>( Action<T> callback ) where T:class
{
var root = Root.Get().GetWindow();
ForEach<T>( root, callback );
}
public static List<T> AllInScene<T>( Func<T,bool> filter = null) where T:class
{
var list = new List<T>();
ForEachInScene<T>(
t =>
{
if ( filter == null || filter( t ) )
{
list.Add( t );
}
}
);
return list;
}
public static List<T> AllIn<T>( Node root, Func<T,bool> filter = null, bool includeRoot = true) where T:class
{
var list = new List<T>();
ForEach<T>( root,
t =>
{
if ( ! includeRoot && t == root )
{
return;
}
if ( filter == null || filter( t ) )
{
list.Add( t );
}
}
);
return list;
}
public static void ForEach<T>( Node root, Action<T> callback ) where T:class
{
var walker = nodesWalker;
walker.Iterate( root,
( n )=>
{
var t = n as T;
if ( t == null )
{
return;
}
callback( t );
}
,false );
}
public static T GetSibling<T>( Node node ) where T:Node
{
if ( node == null )
{
return null;
}
var parent = node.GetParent();
if ( parent == null )
{
return null;
}
return GetDirectChild<T>( parent );
}
public static T CreateChildIn<T>( Node parent, string name = null ) where T:Node,new()
{
var t = new T();
parent.AddChild( t );
t.Owner = parent.Owner;
if ( name != null )
{
t.Name = name;
}
return t;
}
public static async Task RequestNextFrame( this Node node )
{
await node.ToSignal( RenderingServer.Singleton, RenderingServerInstance.SignalName.FramePostDraw );
return;
}
public static T CreateChild<T>( this Node parent, string name = null ) where T:Node,new()
{
return CreateChildIn<T>( parent, name );
}
public static Node CopyNode( Node node, Node parent )
{
var copy = node.Duplicate();
parent.AddChild( copy );
copy.Owner = parent.Owner;
return copy;
}
public static void LogInfo( this Node node, params object[] messages )
{
RJLog.Log( node, messages );
}
public static void LogError( this Node node, params object[] messages )
{
RJLog.Error( node, messages );
}
public static Node DeepCopyTo( this Node node, Node parent )
{
return CopyNodeHierarchy( node, parent );
}
public static Node CopyNodeHierarchy( Node from, Node parent )
{
var copy = CopyNode( from, parent );
for ( int i = 0; i < from.GetChildCount(); i++ )
{
var child = from.GetChild( i );
CopyNodeHierarchy( child, copy );
}
return copy;
}
public static T EnsureValid<T>( T node ) where T:Node
{
if ( GodotObject.IsInstanceValid( node ) )
{
if ( node.IsQueuedForDeletion() )
{
return null;
}
return node;
}
return null;
}
public static void RemoveAndDelete( Node node )
{
if ( ! GodotObject.IsInstanceValid( node ) )
{
return;
}
if ( node.IsQueuedForDeletion() )
{
return;
}
var parent = node.GetParent();
if ( parent != null )
{
parent.RemoveChild( node );
}
node.QueueFree();
}
public static void RemoveAndDeleteAll<N>( List<N> nodes ) where N:Node
{
nodes.ForEach( n => RemoveAndDelete( n ) );
}
public static void RemoveAndDeleteChildrenOfType<T>( Node parent, bool includeInternal = false ) where T:Node
{
if ( parent == null )
{
return;
}
var numChildren = parent.GetChildCount( includeInternal );
for ( int i = numChildren - 1; i >= 0; i-- )
{
var node = parent.GetChild( i, includeInternal );
var t = node as T;
if ( t == null )
{
continue;
}
parent.RemoveChild( node );
node.QueueFree();
}
}
public static void DestroyChildren( this Node parent, bool includeInternal = false )
{
RemoveAndDeleteChildren( parent, includeInternal );
}
public static void RemoveAndDeleteChildren( Node parent, bool includeInternal = false )
{
if ( parent == null )
{
return;
}
var numChildren = parent.GetChildCount( includeInternal );
for ( int i = numChildren - 1; i >= 0; i-- )
{
var node = parent.GetChild( i, includeInternal );
parent.RemoveChild( node );
node.QueueFree();
}
}
public static T GetDirectChild<T>( Node parent ) where T:Node
{
if ( parent == null )
{
return null;
}
var numChildren = parent.GetChildCount();
for ( int i = 0; i < numChildren; i++ )
{
var node = parent.GetChild( i );
if ( node is T )
{
return (T) node;
}
}
return null;
}
public static List<T> GetDirectChildren<T>( Node parent ) where T:Node
{
if ( parent == null )
{
return null;
}
var list = new List<T>();
var numChildren = parent.GetChildCount();
for ( int i = 0; i < numChildren; i++ )
{
var node = parent.GetChild( i );
var script = node.GetScript();
// RJLog.Log( "Node is", typeof(T), node.Name, node.GetType(), script.GetType(), ">>", ( node is T ) );
if ( ! ( node is T ) )
{
continue;
}
list.Add( node as T );
}
return list;
}
public static void OnAllDirectChildren<T>( this Node parent, System.Action<T> action ) where T:Node
{
ForEachDirectChild<T>( parent, action );
}
public static int NumDirectChildrenOf<T>( this Node parent ) where T:Node
{
if ( parent == null )
{
return 0;
}
var numChildren = parent.GetChildCount();
var typeIndex = 0;
for ( int i = 0; i < numChildren; i++ )
{
var node = parent.GetChild( i );
if ( node is T )
{
typeIndex ++;
}
}
return typeIndex;
}
public static T GetNthDirectChild<T>( this Node parent, int index ) where T:Node
{
if ( parent == null )
{
return null;
}
var numChildren = parent.GetChildCount();
var typeIndex = 0;
for ( int i = 0; i < numChildren; i++ )
{
var node = parent.GetChild( i );
if ( node is T t )
{
if ( typeIndex == index )
{
return t;
}
typeIndex ++;
}
}
return null;
}
public static void ForEachDirectChild<T>( Node parent, System.Action<T> action ) where T:Node
{
if ( parent == null || action == null )
{
return;
}
var numChildren = parent.GetChildCount();
for ( int i = 0; i < numChildren; i++ )
{
var node = parent.GetChild( i );
if ( ! ( node is T ) )
{
continue;
}
action( node as T );
}
}
public static List<U> MapDirectChildren<T,U>( Node parent, System.Func<T,U> mapper ) where T:Node
{
var list = new List<U>();
ForEachDirectChild<T>( parent, c => list.Add( mapper( c ) ) );
return list;
}
public static int TypeIndex<T>( Node parent, T child ) where T:Node
{
var counter = 0;
var numChildren = parent.GetChildCount();
for ( int i = 0; i < numChildren; i++ )
{
var node = parent.GetChild( i );
if ( node is T )
{
if ( node == child )
{
return counter;
}
counter++;
}
}
return -1;
}
static NodesWalker nodesWalker = new NodesWalker();
public static T GetAnyChild<T>( Node parent ) where T:Node
{
var result = nodesWalker.Find( parent,
( n )=>
{
var castedNode = n as T;
// RJLog.Log( "Testing", n.UniqueNameInOwner, castedNode != null, n.GetType() );
return castedNode != null;
},
true );
return (T) result;
}
/*
public static void Enable( Node n, bool affectProcess = true, bool affectPhysicsProcess = true, bool affectInput = true )
{
SetState.SetStateOfNode( NodeStateType.Enabled, n, affectProcess, affectPhysicsProcess, affectInput );
}
public static void Disable( Node n, bool affectProcess = true, bool affectPhysicsProcess = true, bool affectInput = true )
{
SetState.SetStateOfNode( NodeStateType.Disabled, n, affectProcess, affectPhysicsProcess, affectInput );
}
*/
public static void Iterate( Node[] nodes, System.Action<Node> callback )
{
for ( int i = 0; i < nodes.Length; i++ )
{
nodesWalker.Iterate( nodes[ i ], callback );
}
}
}
}