using System;
using Unity.VisualScripting.Dependencies.NCalc;
using UnityEngine;
using NCalc = Unity.VisualScripting.Dependencies.NCalc.Expression;
namespace Unity.VisualScripting
{
///
/// Evaluates a mathematical or logical formula with multiple arguments.
///
public sealed class Formula : MultiInputUnit
{
[SerializeAs(nameof(Formula))]
private string _formula;
private NCalc ncalc;
///
/// A mathematical or logical expression tree.
///
[DoNotSerialize]
[Inspectable, UnitHeaderInspectable]
[InspectorTextArea]
public string formula
{
get => _formula;
set
{
_formula = value;
InitializeNCalc();
}
}
///
/// Whether input arguments should only be fetched once and then reused.
///
[Serialize]
[Inspectable(order = int.MaxValue)]
[InspectorExpandTooltip]
public bool cacheArguments { get; set; }
///
/// The result of the calculation or evaluation.
///
[DoNotSerialize]
[PortLabelHidden]
public ValueOutput result { get; private set; }
protected override int minInputCount => 0;
protected override void Definition()
{
base.Definition();
result = ValueOutput(nameof(result), Evaluate);
InputsAllowNull();
foreach (var input in multiInputs)
{
Requirement(input, result);
}
InitializeNCalc();
}
private void InitializeNCalc()
{
if (string.IsNullOrEmpty(formula))
{
ncalc = null;
return;
}
ncalc = new NCalc(formula);
ncalc.Options = EvaluateOptions.IgnoreCase;
ncalc.EvaluateParameter += EvaluateTreeParameter;
ncalc.EvaluateFunction += EvaluateTreeFunction;
}
private object Evaluate(Flow flow)
{
if (ncalc == null)
{
throw new InvalidOperationException("No formula provided.");
}
ncalc.UpdateUnityTimeParameters();
return ncalc.Evaluate(flow);
}
private void EvaluateTreeFunction(Flow flow, string name, FunctionArgs args)
{
if (name == "v2" || name == "V2")
{
if (args.Parameters.Length != 2)
{
throw new ArgumentException($"v2() takes at exactly 2 arguments. {args.Parameters.Length} provided.");
}
args.Result = new Vector2
(
ConversionUtility.Convert(args.Parameters[0].Evaluate(flow)),
ConversionUtility.Convert(args.Parameters[1].Evaluate(flow))
);
}
else if (name == "v3" || name == "V3")
{
if (args.Parameters.Length != 3)
{
throw new ArgumentException($"v3() takes at exactly 3 arguments. {args.Parameters.Length} provided.");
}
args.Result = new Vector3
(
ConversionUtility.Convert(args.Parameters[0].Evaluate(flow)),
ConversionUtility.Convert(args.Parameters[1].Evaluate(flow)),
ConversionUtility.Convert(args.Parameters[2].Evaluate(flow))
);
}
else if (name == "v4" || name == "V4")
{
if (args.Parameters.Length != 4)
{
throw new ArgumentException($"v4() takes at exactly 4 arguments. {args.Parameters.Length} provided.");
}
args.Result = new Vector4
(
ConversionUtility.Convert(args.Parameters[0].Evaluate(flow)),
ConversionUtility.Convert(args.Parameters[1].Evaluate(flow)),
ConversionUtility.Convert(args.Parameters[2].Evaluate(flow)),
ConversionUtility.Convert(args.Parameters[3].Evaluate(flow))
);
}
}
public object GetParameterValue(Flow flow, string name)
{
if (name.Length == 1)
{
var character = name[0];
if (char.IsLetter(character))
{
character = char.ToLower(character);
var index = GetArgumentIndex(character);
if (index < multiInputs.Count)
{
var input = multiInputs[index];
if (cacheArguments && !flow.IsLocal(input))
{
flow.SetValue(input, flow.GetValue(input));
}
return flow.GetValue(input);
}
}
}
else
{
if (Variables.Graph(flow.stack).IsDefined(name))
{
return Variables.Graph(flow.stack).Get(name);
}
var self = flow.stack.self;
if (self != null)
{
if (Variables.Object(self).IsDefined(name))
{
return Variables.Object(self).Get(name);
}
}
var scene = flow.stack.scene;
if (scene != null)
{
if (Variables.Scene(scene).IsDefined(name))
{
return Variables.Scene(scene).Get(name);
}
}
if (Variables.Application.IsDefined(name))
{
return Variables.Application.Get(name);
}
if (Variables.Saved.IsDefined(name))
{
return Variables.Saved.Get(name);
}
}
throw new InvalidOperationException($"Unknown expression tree parameter: '{name}'.\nSupported parameter names are alphabetical indices and variable names.");
}
private void EvaluateTreeParameter(Flow flow, string name, ParameterArgs args)
{
// [param.fieldOrProperty]
// [param.parmeterLessMethod()]
if (name.Contains("."))
{
var parts = name.Split('.');
if (parts.Length == 2)
{
var parameterName = parts[0];
var memberName = parts[1].TrimEnd("()");
var variableValue = GetParameterValue(flow, parameterName);
var manipulator = new Member(variableValue.GetType(), memberName, Type.EmptyTypes);
var target = variableValue;
if (manipulator.isInvocable)
{
args.Result = manipulator.Invoke(target);
}
else if (manipulator.isGettable)
{
args.Result = manipulator.Get(target);
}
else
{
throw new InvalidOperationException($"Cannot get or invoke expression tree parameter: [{parameterName}.{memberName}]");
}
}
else
{
throw new InvalidOperationException($"Cannot parse expression tree parameter: [{name}]");
}
}
else
{
args.Result = GetParameterValue(flow, name);
}
}
public static string GetArgumentName(int index)
{
if (index > ('z' - 'a'))
{
throw new NotImplementedException("Argument indices above 26 are not yet supported.");
}
return ((char)('a' + index)).ToString();
}
public static int GetArgumentIndex(char name)
{
if (name < 'a' || name > 'z')
{
throw new NotImplementedException("Unalphabetical argument names are not yet supported.");
}
return name - 'a';
}
}
}