define-iterator.js 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. 'use strict';
  2. var $ = require('../internals/export');
  3. var createIteratorConstructor = require('../internals/create-iterator-constructor');
  4. var getPrototypeOf = require('../internals/object-get-prototype-of');
  5. var setPrototypeOf = require('../internals/object-set-prototype-of');
  6. var setToStringTag = require('../internals/set-to-string-tag');
  7. var createNonEnumerableProperty = require('../internals/create-non-enumerable-property');
  8. var redefine = require('../internals/redefine');
  9. var wellKnownSymbol = require('../internals/well-known-symbol');
  10. var IS_PURE = require('../internals/is-pure');
  11. var Iterators = require('../internals/iterators');
  12. var IteratorsCore = require('../internals/iterators-core');
  13. var IteratorPrototype = IteratorsCore.IteratorPrototype;
  14. var BUGGY_SAFARI_ITERATORS = IteratorsCore.BUGGY_SAFARI_ITERATORS;
  15. var ITERATOR = wellKnownSymbol('iterator');
  16. var KEYS = 'keys';
  17. var VALUES = 'values';
  18. var ENTRIES = 'entries';
  19. var returnThis = function () { return this; };
  20. module.exports = function (Iterable, NAME, IteratorConstructor, next, DEFAULT, IS_SET, FORCED) {
  21. createIteratorConstructor(IteratorConstructor, NAME, next);
  22. var getIterationMethod = function (KIND) {
  23. if (KIND === DEFAULT && defaultIterator) return defaultIterator;
  24. if (!BUGGY_SAFARI_ITERATORS && KIND in IterablePrototype) return IterablePrototype[KIND];
  25. switch (KIND) {
  26. case KEYS: return function keys() { return new IteratorConstructor(this, KIND); };
  27. case VALUES: return function values() { return new IteratorConstructor(this, KIND); };
  28. case ENTRIES: return function entries() { return new IteratorConstructor(this, KIND); };
  29. } return function () { return new IteratorConstructor(this); };
  30. };
  31. var TO_STRING_TAG = NAME + ' Iterator';
  32. var INCORRECT_VALUES_NAME = false;
  33. var IterablePrototype = Iterable.prototype;
  34. var nativeIterator = IterablePrototype[ITERATOR]
  35. || IterablePrototype['@@iterator']
  36. || DEFAULT && IterablePrototype[DEFAULT];
  37. var defaultIterator = !BUGGY_SAFARI_ITERATORS && nativeIterator || getIterationMethod(DEFAULT);
  38. var anyNativeIterator = NAME == 'Array' ? IterablePrototype.entries || nativeIterator : nativeIterator;
  39. var CurrentIteratorPrototype, methods, KEY;
  40. // fix native
  41. if (anyNativeIterator) {
  42. CurrentIteratorPrototype = getPrototypeOf(anyNativeIterator.call(new Iterable()));
  43. if (IteratorPrototype !== Object.prototype && CurrentIteratorPrototype.next) {
  44. if (!IS_PURE && getPrototypeOf(CurrentIteratorPrototype) !== IteratorPrototype) {
  45. if (setPrototypeOf) {
  46. setPrototypeOf(CurrentIteratorPrototype, IteratorPrototype);
  47. } else if (typeof CurrentIteratorPrototype[ITERATOR] != 'function') {
  48. createNonEnumerableProperty(CurrentIteratorPrototype, ITERATOR, returnThis);
  49. }
  50. }
  51. // Set @@toStringTag to native iterators
  52. setToStringTag(CurrentIteratorPrototype, TO_STRING_TAG, true, true);
  53. if (IS_PURE) Iterators[TO_STRING_TAG] = returnThis;
  54. }
  55. }
  56. // fix Array.prototype.{ values, @@iterator }.name in V8 / FF
  57. if (DEFAULT == VALUES && nativeIterator && nativeIterator.name !== VALUES) {
  58. INCORRECT_VALUES_NAME = true;
  59. defaultIterator = function values() { return nativeIterator.call(this); };
  60. }
  61. // define iterator
  62. if ((!IS_PURE || FORCED) && IterablePrototype[ITERATOR] !== defaultIterator) {
  63. createNonEnumerableProperty(IterablePrototype, ITERATOR, defaultIterator);
  64. }
  65. Iterators[NAME] = defaultIterator;
  66. // export additional methods
  67. if (DEFAULT) {
  68. methods = {
  69. values: getIterationMethod(VALUES),
  70. keys: IS_SET ? defaultIterator : getIterationMethod(KEYS),
  71. entries: getIterationMethod(ENTRIES)
  72. };
  73. if (FORCED) for (KEY in methods) {
  74. if (BUGGY_SAFARI_ITERATORS || INCORRECT_VALUES_NAME || !(KEY in IterablePrototype)) {
  75. redefine(IterablePrototype, KEY, methods[KEY]);
  76. }
  77. } else $({ target: NAME, proto: true, forced: BUGGY_SAFARI_ITERATORS || INCORRECT_VALUES_NAME }, methods);
  78. }
  79. return methods;
  80. };