12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- using System;
- using System.Globalization;
- using System.Linq;
- using Jint.Native;
- using Jint.Native.Function;
- namespace Jint.Runtime.Interop
- {
- /// <summary>
- /// Represents a FunctionInstance wrapper around a CLR method. This is used by user to pass
- /// custom methods to the engine.
- /// </summary>
- public sealed class DelegateWrapper : FunctionInstance
- {
- private readonly Delegate _d;
- public DelegateWrapper(Engine engine, Delegate d) : base(engine, null, null, false)
- {
- _d = d;
- }
- public override JsValue Call(JsValue thisObject, JsValue[] jsArguments)
- {
- var parameterInfos = _d.Method.GetParameters();
- bool delegateContainsParamsArgument = parameterInfos.Any(p => Attribute.IsDefined(p, typeof(ParamArrayAttribute)));
- int delegateArgumentsCount = parameterInfos.Length;
- int delegateNonParamsArgumentsCount = delegateContainsParamsArgument ? delegateArgumentsCount - 1 : delegateArgumentsCount;
- int jsArgumentsCount = jsArguments.Length;
- int jsArgumentsWithoutParamsCount = Math.Min(jsArgumentsCount, delegateNonParamsArgumentsCount);
- var parameters = new object[delegateArgumentsCount];
- // convert non params parameter to expected types
- for (var i = 0; i < jsArgumentsWithoutParamsCount; i++)
- {
- var parameterType = parameterInfos[i].ParameterType;
- if (parameterType == typeof (JsValue))
- {
- parameters[i] = jsArguments[i];
- }
- else
- {
- parameters[i] = Engine.ClrTypeConverter.Convert(
- jsArguments[i].ToObject(),
- parameterType,
- CultureInfo.InvariantCulture);
- }
- }
- // assign null to parameters not provided
- for (var i = jsArgumentsWithoutParamsCount; i < delegateNonParamsArgumentsCount; i++)
- {
- if (parameterInfos[i].ParameterType.IsValueType)
- {
- parameters[i] = Activator.CreateInstance(parameterInfos[i].ParameterType);
- }
- else
- {
- parameters[i] = null;
- }
- }
- // assign params to array and converts each objet to expected type
- if(delegateContainsParamsArgument)
- {
- int paramsArgumentIndex = delegateArgumentsCount - 1;
- int paramsCount = Math.Max(0, jsArgumentsCount - delegateNonParamsArgumentsCount);
- object[] paramsParameter = new object[paramsCount];
- var paramsParameterType = parameterInfos[paramsArgumentIndex].ParameterType.GetElementType();
- for (var i = paramsArgumentIndex; i < jsArgumentsCount; i++)
- {
- var paramsIndex = i - paramsArgumentIndex;
- if (paramsParameterType == typeof(JsValue))
- {
- paramsParameter[paramsIndex] = jsArguments[i];
- }
- else
- {
- paramsParameter[paramsIndex] = Engine.ClrTypeConverter.Convert(
- jsArguments[i].ToObject(),
- paramsParameterType,
- CultureInfo.InvariantCulture);
- }
- }
- parameters[paramsArgumentIndex] = paramsParameter;
- }
- return JsValue.FromObject(Engine, _d.DynamicInvoke(parameters));
- }
- }
- }
|