JintNewExpression.cs 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374
  1. using System;
  2. using Esprima.Ast;
  3. using Jint.Native;
  4. namespace Jint.Runtime.Interpreter.Expressions
  5. {
  6. internal sealed class JintNewExpression : JintExpression
  7. {
  8. private JintExpression _calleeExpression;
  9. private JintExpression[] _jintArguments = Array.Empty<JintExpression>();
  10. private bool _hasSpreads;
  11. public JintNewExpression(NewExpression expression) : base(expression)
  12. {
  13. _initialized = false;
  14. }
  15. protected override void Initialize(EvaluationContext context)
  16. {
  17. var engine = context.Engine;
  18. var expression = (NewExpression) _expression;
  19. _calleeExpression = Build(engine, expression.Callee);
  20. if (expression.Arguments.Count <= 0)
  21. {
  22. return;
  23. }
  24. _jintArguments = new JintExpression[expression.Arguments.Count];
  25. for (var i = 0; i < _jintArguments.Length; i++)
  26. {
  27. var argument = expression.Arguments[i];
  28. _jintArguments[i] = Build(engine, argument);
  29. _hasSpreads |= argument.Type == Nodes.SpreadElement;
  30. }
  31. }
  32. protected override ExpressionResult EvaluateInternal(EvaluationContext context)
  33. {
  34. var engine = context.Engine;
  35. // todo: optimize by defining a common abstract class or interface
  36. var jsValue = _calleeExpression.GetValue(context).Value;
  37. JsValue[] arguments;
  38. if (_jintArguments.Length == 0)
  39. {
  40. arguments = Array.Empty<JsValue>();
  41. }
  42. else if (_hasSpreads)
  43. {
  44. arguments = BuildArgumentsWithSpreads(context, _jintArguments);
  45. }
  46. else
  47. {
  48. arguments = engine._jsValueArrayPool.RentArray(_jintArguments.Length);
  49. BuildArguments(context, _jintArguments, arguments);
  50. }
  51. if (!jsValue.IsConstructor)
  52. {
  53. ExceptionHelper.ThrowTypeError(engine.Realm, _calleeExpression.SourceText + " is not a constructor");
  54. }
  55. // construct the new instance using the Function's constructor method
  56. var instance = engine.Construct(jsValue, arguments, jsValue, _calleeExpression);
  57. engine._jsValueArrayPool.ReturnArray(arguments);
  58. return NormalCompletion(instance);
  59. }
  60. }
  61. }