es.number.constructor.js 3.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. 'use strict';
  2. var DESCRIPTORS = require('../internals/descriptors');
  3. var global = require('../internals/global');
  4. var isForced = require('../internals/is-forced');
  5. var redefine = require('../internals/redefine');
  6. var has = require('../internals/has');
  7. var classof = require('../internals/classof-raw');
  8. var inheritIfRequired = require('../internals/inherit-if-required');
  9. var toPrimitive = require('../internals/to-primitive');
  10. var fails = require('../internals/fails');
  11. var create = require('../internals/object-create');
  12. var getOwnPropertyNames = require('../internals/object-get-own-property-names').f;
  13. var getOwnPropertyDescriptor = require('../internals/object-get-own-property-descriptor').f;
  14. var defineProperty = require('../internals/object-define-property').f;
  15. var trim = require('../internals/string-trim').trim;
  16. var NUMBER = 'Number';
  17. var NativeNumber = global[NUMBER];
  18. var NumberPrototype = NativeNumber.prototype;
  19. // Opera ~12 has broken Object#toString
  20. var BROKEN_CLASSOF = classof(create(NumberPrototype)) == NUMBER;
  21. // `ToNumber` abstract operation
  22. // https://tc39.es/ecma262/#sec-tonumber
  23. var toNumber = function (argument) {
  24. var it = toPrimitive(argument, false);
  25. var first, third, radix, maxCode, digits, length, index, code;
  26. if (typeof it == 'string' && it.length > 2) {
  27. it = trim(it);
  28. first = it.charCodeAt(0);
  29. if (first === 43 || first === 45) {
  30. third = it.charCodeAt(2);
  31. if (third === 88 || third === 120) return NaN; // Number('+0x1') should be NaN, old V8 fix
  32. } else if (first === 48) {
  33. switch (it.charCodeAt(1)) {
  34. case 66: case 98: radix = 2; maxCode = 49; break; // fast equal of /^0b[01]+$/i
  35. case 79: case 111: radix = 8; maxCode = 55; break; // fast equal of /^0o[0-7]+$/i
  36. default: return +it;
  37. }
  38. digits = it.slice(2);
  39. length = digits.length;
  40. for (index = 0; index < length; index++) {
  41. code = digits.charCodeAt(index);
  42. // parseInt parses a string to a first unavailable symbol
  43. // but ToNumber should return NaN if a string contains unavailable symbols
  44. if (code < 48 || code > maxCode) return NaN;
  45. } return parseInt(digits, radix);
  46. }
  47. } return +it;
  48. };
  49. // `Number` constructor
  50. // https://tc39.es/ecma262/#sec-number-constructor
  51. if (isForced(NUMBER, !NativeNumber(' 0o1') || !NativeNumber('0b1') || NativeNumber('+0x1'))) {
  52. var NumberWrapper = function Number(value) {
  53. var it = arguments.length < 1 ? 0 : value;
  54. var dummy = this;
  55. return dummy instanceof NumberWrapper
  56. // check on 1..constructor(foo) case
  57. && (BROKEN_CLASSOF ? fails(function () { NumberPrototype.valueOf.call(dummy); }) : classof(dummy) != NUMBER)
  58. ? inheritIfRequired(new NativeNumber(toNumber(it)), dummy, NumberWrapper) : toNumber(it);
  59. };
  60. for (var keys = DESCRIPTORS ? getOwnPropertyNames(NativeNumber) : (
  61. // ES3:
  62. 'MAX_VALUE,MIN_VALUE,NaN,NEGATIVE_INFINITY,POSITIVE_INFINITY,' +
  63. // ES2015 (in case, if modules with ES2015 Number statics required before):
  64. 'EPSILON,isFinite,isInteger,isNaN,isSafeInteger,MAX_SAFE_INTEGER,' +
  65. 'MIN_SAFE_INTEGER,parseFloat,parseInt,isInteger,' +
  66. // ESNext
  67. 'fromString,range'
  68. ).split(','), j = 0, key; keys.length > j; j++) {
  69. if (has(NativeNumber, key = keys[j]) && !has(NumberWrapper, key)) {
  70. defineProperty(NumberWrapper, key, getOwnPropertyDescriptor(NativeNumber, key));
  71. }
  72. }
  73. NumberWrapper.prototype = NumberPrototype;
  74. NumberPrototype.constructor = NumberWrapper;
  75. redefine(global, NUMBER, NumberWrapper);
  76. }