MethodInfoFunctionInstance.cs 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. using System.Reflection;
  7. using Jint.Native;
  8. using Jint.Native.Array;
  9. using Jint.Native.Function;
  10. namespace Jint.Runtime.Interop
  11. {
  12. public sealed class MethodInfoFunctionInstance : FunctionInstance
  13. {
  14. private readonly MethodInfo[] _methods;
  15. public MethodInfoFunctionInstance(Engine engine, MethodInfo[] methods)
  16. : base(engine, null, null, false)
  17. {
  18. _methods = methods;
  19. Prototype = engine.Function.PrototypeObject;
  20. }
  21. public override JsValue Call(JsValue thisObject, JsValue[] arguments)
  22. {
  23. return Invoke(_methods, thisObject, arguments);
  24. }
  25. public JsValue Invoke(MethodInfo[] methodInfos, JsValue thisObject, JsValue[] jsArguments)
  26. {
  27. var arguments = ProcessParamsArrays(jsArguments, methodInfos);
  28. var methods = TypeConverter.FindBestMatch(Engine, methodInfos, arguments).ToList();
  29. var converter = Engine.ClrTypeConverter;
  30. foreach (var method in methods)
  31. {
  32. var parameters = new object[arguments.Length];
  33. var argumentsMatch = true;
  34. for (var i = 0; i < arguments.Length; i++)
  35. {
  36. var parameterType = method.GetParameters()[i].ParameterType;
  37. if (parameterType == typeof(JsValue))
  38. {
  39. parameters[i] = arguments[i];
  40. }
  41. else
  42. {
  43. if (!converter.TryConvert(arguments[i].ToObject(), parameterType, CultureInfo.InvariantCulture, out parameters[i]))
  44. {
  45. argumentsMatch = false;
  46. break;
  47. }
  48. var lambdaExpression = parameters[i] as LambdaExpression;
  49. if (lambdaExpression != null)
  50. {
  51. parameters[i] = lambdaExpression.Compile();
  52. }
  53. }
  54. }
  55. if (!argumentsMatch)
  56. {
  57. continue;
  58. }
  59. // todo: cache method info
  60. return JsValue.FromObject(Engine, method.Invoke(thisObject.ToObject(), parameters.ToArray()));
  61. }
  62. throw new JavaScriptException(Engine.TypeError, "No public methods with the specified arguments were found.");
  63. }
  64. private JsValue[] ProcessParamsArrays(JsValue[] jsArguments, IEnumerable<MethodInfo> methodInfos)
  65. {
  66. foreach (var methodInfo in methodInfos)
  67. {
  68. var parameters = methodInfo.GetParameters();
  69. if (!parameters.Any(p => Attribute.IsDefined(p, typeof(ParamArrayAttribute))))
  70. continue;
  71. var nonParamsArgumentsCount = parameters.Length - 1;
  72. if (jsArguments.Length < nonParamsArgumentsCount)
  73. continue;
  74. var newArgumentsCollection = jsArguments.Take(nonParamsArgumentsCount).ToList();
  75. var argsToTransform = jsArguments.Skip(nonParamsArgumentsCount).ToList();
  76. if (argsToTransform.Count == 1 && argsToTransform.FirstOrDefault().IsArray())
  77. continue;
  78. var arrayInstance = ArrayConstructor.CreateArrayConstructor(Engine).Construct(argsToTransform.ToArray());
  79. newArgumentsCollection.Add(new JsValue(arrayInstance));
  80. return newArgumentsCollection.ToArray();
  81. }
  82. return jsArguments;
  83. }
  84. }
  85. }