188 lines
7.7 KiB
C#
188 lines
7.7 KiB
C#
|
using System.Collections;
|
|||
|
using System.Linq;
|
|||
|
using System.Reflection;
|
|||
|
using NodeCanvas.Framework;
|
|||
|
using NodeCanvas.Framework.Internal;
|
|||
|
using ParadoxNotion;
|
|||
|
using ParadoxNotion.Design;
|
|||
|
using ParadoxNotion.Serialization;
|
|||
|
using UnityEngine;
|
|||
|
|
|||
|
|
|||
|
namespace NodeCanvas.Tasks.Actions
|
|||
|
{
|
|||
|
|
|||
|
[Name("Execute Function (Desktop Only)", 10)]
|
|||
|
[Category("✫ Reflected/Faster Versions (Desktop Platforms Only)")]
|
|||
|
[Description("This version works in destop/JIT platform only.\n\nExecute a function on a script, of up to 6 parameters and save the return if any. If function is an IEnumerator it will execute as a coroutine.")]
|
|||
|
public class ExecuteFunction : ActionTask, IReflectedWrapper
|
|||
|
{
|
|||
|
|
|||
|
[SerializeField]
|
|||
|
protected ReflectedWrapper functionWrapper;
|
|||
|
|
|||
|
private bool routineRunning;
|
|||
|
|
|||
|
ISerializedReflectedInfo IReflectedWrapper.GetSerializedInfo() { return functionWrapper?.GetSerializedMethod(); }
|
|||
|
|
|||
|
private MethodInfo targetMethod { get { return functionWrapper != null ? functionWrapper.GetMethod() : null; } }
|
|||
|
|
|||
|
public override System.Type agentType {
|
|||
|
get
|
|||
|
{
|
|||
|
if ( targetMethod == null ) { return typeof(Transform); }
|
|||
|
return targetMethod.IsStatic ? null : targetMethod.RTReflectedOrDeclaredType();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
protected override string info {
|
|||
|
get
|
|||
|
{
|
|||
|
if ( functionWrapper == null ) { return "No Method Selected"; }
|
|||
|
if ( targetMethod == null ) { return functionWrapper.AsString().FormatError(); }
|
|||
|
|
|||
|
var variables = functionWrapper.GetVariables();
|
|||
|
var returnInfo = "";
|
|||
|
var paramInfo = "";
|
|||
|
if ( targetMethod.ReturnType == typeof(void) ) {
|
|||
|
for ( var i = 0; i < variables.Length; i++ ) {
|
|||
|
paramInfo += ( i != 0 ? ", " : "" ) + variables[i].ToString();
|
|||
|
}
|
|||
|
} else {
|
|||
|
returnInfo = targetMethod.ReturnType == typeof(void) || targetMethod.ReturnType == typeof(IEnumerator) || variables[0].isNone ? "" : variables[0] + " = ";
|
|||
|
for ( var i = 1; i < variables.Length; i++ ) {
|
|||
|
paramInfo += ( i != 1 ? ", " : "" ) + variables[i].ToString();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
var mInfo = targetMethod.IsStatic ? targetMethod.RTReflectedOrDeclaredType().FriendlyName() : agentInfo;
|
|||
|
return string.Format("{0}{1}.{2}({3})", returnInfo, mInfo, targetMethod.Name, paramInfo);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public override void OnValidate(ITaskSystem ownerSystem) {
|
|||
|
if ( functionWrapper != null && functionWrapper.HasChanged() ) {
|
|||
|
SetMethod(functionWrapper.GetMethod());
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//store the method info on init
|
|||
|
protected override string OnInit() {
|
|||
|
if ( functionWrapper == null ) { return "No Method selected"; }
|
|||
|
if ( targetMethod == null ) { return string.Format("Missing Method '{0}'", functionWrapper.AsString()); }
|
|||
|
|
|||
|
try {
|
|||
|
functionWrapper.Init(targetMethod.IsStatic ? null : agent);
|
|||
|
return null;
|
|||
|
}
|
|||
|
catch { return "ExecuteFunction Error"; }
|
|||
|
}
|
|||
|
|
|||
|
//do it by calling delegate or invoking method
|
|||
|
protected override void OnExecute() {
|
|||
|
|
|||
|
if ( targetMethod == null ) {
|
|||
|
EndAction(false);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if ( targetMethod.ReturnType == typeof(IEnumerator) ) {
|
|||
|
StartCoroutine(InternalCoroutine((IEnumerator)( (ReflectedFunctionWrapper)functionWrapper ).Call()));
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if ( targetMethod.ReturnType == typeof(void) ) {
|
|||
|
( (ReflectedActionWrapper)functionWrapper ).Call();
|
|||
|
} else {
|
|||
|
( (ReflectedFunctionWrapper)functionWrapper ).Call();
|
|||
|
}
|
|||
|
|
|||
|
EndAction(true);
|
|||
|
}
|
|||
|
|
|||
|
protected override void OnStop() {
|
|||
|
routineRunning = false;
|
|||
|
}
|
|||
|
|
|||
|
IEnumerator InternalCoroutine(IEnumerator routine) {
|
|||
|
routineRunning = true;
|
|||
|
while ( routineRunning && routine.MoveNext() ) {
|
|||
|
if ( routineRunning == false ) {
|
|||
|
yield break;
|
|||
|
}
|
|||
|
yield return routine.Current;
|
|||
|
}
|
|||
|
|
|||
|
if ( routineRunning ) {
|
|||
|
EndAction();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void SetMethod(MethodInfo method) {
|
|||
|
if ( method != null ) {
|
|||
|
UndoUtility.RecordObject(ownerSystem.contextObject, "Set Reflection Member");
|
|||
|
functionWrapper = ReflectedWrapper.Create(method, blackboard);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
///----------------------------------------------------------------------------------------------
|
|||
|
///---------------------------------------UNITY EDITOR-------------------------------------------
|
|||
|
#if UNITY_EDITOR
|
|||
|
|
|||
|
protected override void OnTaskInspectorGUI() {
|
|||
|
|
|||
|
if ( !Application.isPlaying && GUILayout.Button("Select Method") ) {
|
|||
|
var menu = new UnityEditor.GenericMenu();
|
|||
|
if ( agent != null ) {
|
|||
|
foreach ( var comp in agent.GetComponents(typeof(Component)).Where(c => !c.hideFlags.HasFlag(HideFlags.HideInInspector)) ) {
|
|||
|
menu = EditorUtils.GetInstanceMethodSelectionMenu(comp.GetType(), typeof(object), typeof(object), SetMethod, 6, false, false, menu);
|
|||
|
}
|
|||
|
menu.AddSeparator("/");
|
|||
|
}
|
|||
|
|
|||
|
foreach ( var t in TypePrefs.GetPreferedTypesList(typeof(object)) ) {
|
|||
|
menu = EditorUtils.GetStaticMethodSelectionMenu(t, typeof(object), typeof(object), SetMethod, 6, false, false, menu);
|
|||
|
if ( typeof(UnityEngine.Component).IsAssignableFrom(t) ) {
|
|||
|
menu = EditorUtils.GetInstanceMethodSelectionMenu(t, typeof(object), typeof(object), SetMethod, 6, false, false, menu);
|
|||
|
}
|
|||
|
}
|
|||
|
menu.ShowAsBrowser("Select Method", this.GetType());
|
|||
|
Event.current.Use();
|
|||
|
}
|
|||
|
|
|||
|
var m = targetMethod;
|
|||
|
if ( m != null ) {
|
|||
|
GUILayout.BeginVertical("box");
|
|||
|
UnityEditor.EditorGUILayout.LabelField("Type", targetMethod.RTReflectedOrDeclaredType().FriendlyName());
|
|||
|
UnityEditor.EditorGUILayout.LabelField("Method", m.Name);
|
|||
|
UnityEditor.EditorGUILayout.LabelField("Returns", m.ReturnType.FriendlyName());
|
|||
|
UnityEditor.EditorGUILayout.HelpBox(XMLDocs.GetMemberSummary(targetMethod), UnityEditor.MessageType.None);
|
|||
|
|
|||
|
if ( m.ReturnType == typeof(IEnumerator) ) {
|
|||
|
GUILayout.Label("<b>This will execute as a Coroutine!</b>");
|
|||
|
}
|
|||
|
|
|||
|
GUILayout.EndVertical();
|
|||
|
|
|||
|
var paramNames = m.GetParameters().Select(p => p.Name.SplitCamelCase()).ToArray();
|
|||
|
var variables = functionWrapper.GetVariables();
|
|||
|
if ( m.ReturnType == typeof(void) ) {
|
|||
|
for ( var i = 0; i < paramNames.Length; i++ ) {
|
|||
|
NodeCanvas.Editor.BBParameterEditor.ParameterField(paramNames[i], variables[i]);
|
|||
|
}
|
|||
|
} else {
|
|||
|
for ( var i = 0; i < paramNames.Length; i++ ) {
|
|||
|
NodeCanvas.Editor.BBParameterEditor.ParameterField(paramNames[i], variables[i + 1]);
|
|||
|
}
|
|||
|
|
|||
|
if ( m.ReturnType != typeof(IEnumerator) ) {
|
|||
|
NodeCanvas.Editor.BBParameterEditor.ParameterField("Save Return Value", variables[0], true);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#endif
|
|||
|
}
|
|||
|
}
|