NumberConstructor.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. using Jint.Collections;
  2. using Jint.Native.Function;
  3. using Jint.Native.Global;
  4. using Jint.Native.Object;
  5. using Jint.Runtime;
  6. using Jint.Runtime.Descriptors;
  7. using Jint.Runtime.Interop;
  8. namespace Jint.Native.Number
  9. {
  10. public sealed class NumberConstructor : FunctionInstance, IConstructor
  11. {
  12. private static readonly JsString _functionName = new JsString("Number");
  13. private const long MinSafeInteger = -9007199254740991;
  14. internal const long MaxSafeInteger = 9007199254740991;
  15. public NumberConstructor(Engine engine)
  16. : base(engine, _functionName)
  17. {
  18. }
  19. public static NumberConstructor CreateNumberConstructor(Engine engine)
  20. {
  21. var obj = new NumberConstructor(engine)
  22. {
  23. _prototype = engine.Function.PrototypeObject
  24. };
  25. // The value of the [[Prototype]] internal property of the Number constructor is the Function prototype object
  26. obj.PrototypeObject = NumberPrototype.CreatePrototypeObject(engine, obj);
  27. obj._length = new PropertyDescriptor(JsNumber.One, PropertyFlag.Configurable);
  28. // The initial value of Number.prototype is the Number prototype object
  29. obj._prototypeDescriptor = new PropertyDescriptor(obj.PrototypeObject, PropertyFlag.AllForbidden);
  30. return obj;
  31. }
  32. protected override void Initialize()
  33. {
  34. var properties = new PropertyDictionary(15, checkExistingKeys: false)
  35. {
  36. ["MAX_VALUE"] = new PropertyDescriptor(new PropertyDescriptor(double.MaxValue, PropertyFlag.AllForbidden)),
  37. ["MIN_VALUE"] = new PropertyDescriptor(new PropertyDescriptor(double.Epsilon, PropertyFlag.AllForbidden)),
  38. ["NaN"] = new PropertyDescriptor(new PropertyDescriptor(double.NaN, PropertyFlag.AllForbidden)),
  39. ["NEGATIVE_INFINITY"] = new PropertyDescriptor(new PropertyDescriptor(double.NegativeInfinity, PropertyFlag.AllForbidden)),
  40. ["POSITIVE_INFINITY"] = new PropertyDescriptor(new PropertyDescriptor(double.PositiveInfinity, PropertyFlag.AllForbidden)),
  41. ["EPSILON"] = new PropertyDescriptor(new PropertyDescriptor(JsNumber.JavaScriptEpsilon, PropertyFlag.AllForbidden)),
  42. ["MIN_SAFE_INTEGER"] = new PropertyDescriptor(new PropertyDescriptor(MinSafeInteger, PropertyFlag.AllForbidden)),
  43. ["MAX_SAFE_INTEGER"] = new PropertyDescriptor(new PropertyDescriptor(MaxSafeInteger, PropertyFlag.AllForbidden)),
  44. ["isFinite"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "isFinite", IsFinite, 1, PropertyFlag.Configurable), true, false, true),
  45. ["isInteger"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "isInteger", IsInteger, 1, PropertyFlag.Configurable), true, false, true),
  46. ["isNaN"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "isNaN", IsNaN, 1, PropertyFlag.Configurable), true, false, true),
  47. ["isSafeInteger"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "isSafeInteger", IsSafeInteger, 1, PropertyFlag.Configurable), true, false, true),
  48. ["parseFloat"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "parseFloat", GlobalObject.ParseFloat, 0, PropertyFlag.Configurable), true, false, true),
  49. ["parseInt"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "parseInt", GlobalObject.ParseInt, 0, PropertyFlag.Configurable), true, false, true)
  50. };
  51. SetProperties(properties);
  52. }
  53. private static JsValue IsFinite(JsValue thisObj, JsValue[] arguments)
  54. {
  55. if (!(arguments.At(0) is JsNumber num))
  56. {
  57. return false;
  58. }
  59. return double.IsInfinity(num._value) || double.IsNaN(num._value) ? JsBoolean.False : JsBoolean.True;
  60. }
  61. private static JsValue IsInteger(JsValue thisObj, JsValue[] arguments)
  62. {
  63. if (!(arguments.At(0) is JsNumber num))
  64. {
  65. return false;
  66. }
  67. if (double.IsInfinity(num._value) || double.IsNaN(num._value))
  68. {
  69. return JsBoolean.False;
  70. }
  71. var integer = TypeConverter.ToInteger(num);
  72. return integer == num._value;
  73. }
  74. private static JsValue IsNaN(JsValue thisObj, JsValue[] arguments)
  75. {
  76. if (!(arguments.At(0) is JsNumber num))
  77. {
  78. return false;
  79. }
  80. return double.IsNaN(num._value);
  81. }
  82. private static JsValue IsSafeInteger(JsValue thisObj, JsValue[] arguments)
  83. {
  84. if (!(arguments.At(0) is JsNumber num))
  85. {
  86. return false;
  87. }
  88. if (double.IsInfinity(num._value) || double.IsNaN(num._value))
  89. {
  90. return JsBoolean.False;
  91. }
  92. var integer = TypeConverter.ToInteger(num);
  93. if (integer != num._value)
  94. {
  95. return false;
  96. }
  97. return System.Math.Abs(integer) <= MaxSafeInteger;
  98. }
  99. public override JsValue Call(JsValue thisObject, JsValue[] arguments)
  100. {
  101. if (arguments.Length == 0)
  102. {
  103. return 0d;
  104. }
  105. return TypeConverter.ToNumber(arguments[0]);
  106. }
  107. /// <summary>
  108. /// http://www.ecma-international.org/ecma-262/5.1/#sec-15.7.2.1
  109. /// </summary>
  110. /// <param name="arguments"></param>
  111. /// <returns></returns>
  112. public ObjectInstance Construct(JsValue[] arguments, JsValue newTarget)
  113. {
  114. return Construct(arguments.Length > 0 ? TypeConverter.ToNumber(arguments[0]) : 0);
  115. }
  116. public NumberPrototype PrototypeObject { get; private set; }
  117. public NumberInstance Construct(double value)
  118. {
  119. return Construct(JsNumber.Create(value));
  120. }
  121. public NumberInstance Construct(JsNumber value)
  122. {
  123. var instance = new NumberInstance(Engine)
  124. {
  125. _prototype = PrototypeObject,
  126. NumberData = value
  127. };
  128. return instance;
  129. }
  130. }
  131. }