yzt
2023-05-26 de4278af2fd46705a40bac58ec01122db6b7f3d7
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
/* This file is automatically rebuilt by the Cesium build process. */
define(['exports'], (function (exports) { 'use strict';
 
  /**
   * @function
   *
   * @param {*} value The object.
   * @returns {Boolean} Returns true if the object is defined, returns false otherwise.
   *
   * @example
   * if (Cesium.defined(positions)) {
   *      doSomething();
   * } else {
   *      doSomethingElse();
   * }
   */
  function defined(value) {
    return value !== undefined && value !== null;
  }
 
  /**
   * Returns the first parameter if not undefined, otherwise the second parameter.
   * Useful for setting a default value for a parameter.
   *
   * @function
   *
   * @param {*} a
   * @param {*} b
   * @returns {*} Returns the first parameter if not undefined, otherwise the second parameter.
   *
   * @example
   * param = Cesium.defaultValue(param, 'default');
   */
  function defaultValue(a, b) {
    if (a !== undefined && a !== null) {
      return a;
    }
    return b;
  }
 
  /**
   * A frozen empty object that can be used as the default value for options passed as
   * an object literal.
   * @type {Object}
   * @memberof defaultValue
   */
  defaultValue.EMPTY_OBJECT = Object.freeze({});
 
  /* This file is automatically rebuilt by the Cesium build process. */
  var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};
 
  function createCommonjsModule(fn, basedir, module) {
    return module = {
        path: basedir,
        exports: {},
        require: function (path, base) {
            return commonjsRequire(path, (base === undefined || base === null) ? module.path : base);
        }
    }, fn(module, module.exports), module.exports;
  }
 
  function commonjsRequire () {
    throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs');
  }
 
  /* This file is automatically rebuilt by the Cesium build process. */
 
  var when = createCommonjsModule(function (module, exports) {
  /** @license MIT License (c) copyright B Cavalier & J Hann */
 
  /**
   * A lightweight CommonJS Promises/A and when() implementation
   * when is part of the cujo.js family of libraries (http://cujojs.com/)
   *
   * Licensed under the MIT License at:
   * http://www.opensource.org/licenses/mit-license.php
   *
   * @version 1.7.1
   */
 
  (function(define) {define(function () {
    var reduceArray, slice, undef;
 
    //
    // Public API
    //
 
    when.defer     = defer;     // Create a deferred
    when.resolve   = resolve;   // Create a resolved promise
    when.reject    = reject;    // Create a rejected promise
 
    when.join      = join;      // Join 2 or more promises
 
    when.all       = all;       // Resolve a list of promises
    when.map       = map;       // Array.map() for promises
    when.reduce    = reduce;    // Array.reduce() for promises
 
    when.any       = any;       // One-winner race
    when.some      = some;      // Multi-winner race
 
    when.chain     = chain;     // Make a promise trigger another resolver
 
    when.isPromise = isPromise; // Determine if a thing is a promise
 
    /**
     * Register an observer for a promise or immediate value.
     *
     * @param {*} promiseOrValue
     * @param {function?} [onFulfilled] callback to be called when promiseOrValue is
     *   successfully fulfilled.  If promiseOrValue is an immediate value, callback
     *   will be invoked immediately.
     * @param {function?} [onRejected] callback to be called when promiseOrValue is
     *   rejected.
     * @param {function?} [onProgress] callback to be called when progress updates
     *   are issued for promiseOrValue.
     * @returns {Promise} a new {@link Promise} that will complete with the return
     *   value of callback or errback or the completion value of promiseOrValue if
     *   callback and/or errback is not supplied.
     */
    function when(promiseOrValue, onFulfilled, onRejected, onProgress) {
        // Get a trusted promise for the input promiseOrValue, and then
        // register promise handlers
        return resolve(promiseOrValue).then(onFulfilled, onRejected, onProgress);
    }
 
    /**
     * Returns promiseOrValue if promiseOrValue is a {@link Promise}, a new Promise if
     * promiseOrValue is a foreign promise, or a new, already-fulfilled {@link Promise}
     * whose value is promiseOrValue if promiseOrValue is an immediate value.
     *
     * @param {*} promiseOrValue
     * @returns Guaranteed to return a trusted Promise.  If promiseOrValue is a when.js {@link Promise}
     *   returns promiseOrValue, otherwise, returns a new, already-resolved, when.js {@link Promise}
     *   whose resolution value is:
     *   * the resolution value of promiseOrValue if it's a foreign promise, or
     *   * promiseOrValue if it's a value
     */
    function resolve(promiseOrValue) {
        var promise, deferred;
 
        if(promiseOrValue instanceof Promise) {
            // It's a when.js promise, so we trust it
            promise = promiseOrValue;
 
        } else {
            // It's not a when.js promise. See if it's a foreign promise or a value.
            if(isPromise(promiseOrValue)) {
                // It's a thenable, but we don't know where it came from, so don't trust
                // its implementation entirely.  Introduce a trusted middleman when.js promise
                deferred = defer();
 
                // IMPORTANT: This is the only place when.js should ever call .then() on an
                // untrusted promise. Don't expose the return value to the untrusted promise
                promiseOrValue.then(
                    function(value)  { deferred.resolve(value); },
                    function(reason) { deferred.reject(reason); },
                    function(update) { deferred.progress(update); }
                );
 
                promise = deferred.promise;
 
            } else {
                // It's a value, not a promise.  Create a resolved promise for it.
                promise = fulfilled(promiseOrValue);
            }
        }
 
        return promise;
    }
 
    /**
     * Returns a rejected promise for the supplied promiseOrValue.  The returned
     * promise will be rejected with:
     * - promiseOrValue, if it is a value, or
     * - if promiseOrValue is a promise
     *   - promiseOrValue's value after it is fulfilled
     *   - promiseOrValue's reason after it is rejected
     * @param {*} promiseOrValue the rejected value of the returned {@link Promise}
     * @return {Promise} rejected {@link Promise}
     */
    function reject(promiseOrValue) {
        return when(promiseOrValue, rejected);
    }
 
    /**
     * Trusted Promise constructor.  A Promise created from this constructor is
     * a trusted when.js promise.  Any other duck-typed promise is considered
     * untrusted.
     * @constructor
     * @name Promise
     */
    function Promise(then) {
        this.then = then;
    }
 
    Promise.prototype = {
        /**
         * Register a callback that will be called when a promise is
         * fulfilled or rejected.  Optionally also register a progress handler.
         * Shortcut for .then(onFulfilledOrRejected, onFulfilledOrRejected, onProgress)
         * @param {function?} [onFulfilledOrRejected]
         * @param {function?} [onProgress]
         * @return {Promise}
         */
        always: function(onFulfilledOrRejected, onProgress) {
            return this.then(onFulfilledOrRejected, onFulfilledOrRejected, onProgress);
        },
 
        /**
         * Register a rejection handler.  Shortcut for .then(undefined, onRejected)
         * @param {function?} onRejected
         * @return {Promise}
         */
        otherwise: function(onRejected) {
            return this.then(undef, onRejected);
        },
 
        /**
         * Shortcut for .then(function() { return value; })
         * @param  {*} value
         * @return {Promise} a promise that:
         *  - is fulfilled if value is not a promise, or
         *  - if value is a promise, will fulfill with its value, or reject
         *    with its reason.
         */
        yield: function(value) {
            return this.then(function() {
                return value;
            });
        },
 
        /**
         * Assumes that this promise will fulfill with an array, and arranges
         * for the onFulfilled to be called with the array as its argument list
         * i.e. onFulfilled.spread(undefined, array).
         * @param {function} onFulfilled function to receive spread arguments
         * @return {Promise}
         */
        spread: function(onFulfilled) {
            return this.then(function(array) {
                // array may contain promises, so resolve its contents.
                return all(array, function(array) {
                    return onFulfilled.apply(undef, array);
                });
            });
        }
    };
 
    /**
     * Create an already-resolved promise for the supplied value
     * @private
     *
     * @param {*} value
     * @return {Promise} fulfilled promise
     */
    function fulfilled(value) {
        var p = new Promise(function(onFulfilled) {
            // TODO: Promises/A+ check typeof onFulfilled
            try {
                return resolve(onFulfilled ? onFulfilled(value) : value);
            } catch(e) {
                return rejected(e);
            }
        });
 
        return p;
    }
 
    /**
     * Create an already-rejected {@link Promise} with the supplied
     * rejection reason.
     * @private
     *
     * @param {*} reason
     * @return {Promise} rejected promise
     */
    function rejected(reason) {
        var p = new Promise(function(_, onRejected) {
            // TODO: Promises/A+ check typeof onRejected
            try {
                return onRejected ? resolve(onRejected(reason)) : rejected(reason);
            } catch(e) {
                return rejected(e);
            }
        });
 
        return p;
    }
 
    /**
     * Creates a new, Deferred with fully isolated resolver and promise parts,
     * either or both of which may be given out safely to consumers.
     * The Deferred itself has the full API: resolve, reject, progress, and
     * then. The resolver has resolve, reject, and progress.  The promise
     * only has then.
     *
     * @return {Deferred}
     */
    function defer() {
        var deferred, promise, handlers, progressHandlers,
            _then, _progress, _resolve;
 
        /**
         * The promise for the new deferred
         * @type {Promise}
         */
        promise = new Promise(then);
 
        /**
         * The full Deferred object, with {@link Promise} and {@link Resolver} parts
         * @class Deferred
         * @name Deferred
         */
        deferred = {
            then:     then, // DEPRECATED: use deferred.promise.then
            resolve:  promiseResolve,
            reject:   promiseReject,
            // TODO: Consider renaming progress() to notify()
            progress: promiseProgress,
 
            promise:  promise,
 
            resolver: {
                resolve:  promiseResolve,
                reject:   promiseReject,
                progress: promiseProgress
            }
        };
 
        handlers = [];
        progressHandlers = [];
 
        /**
         * Pre-resolution then() that adds the supplied callback, errback, and progback
         * functions to the registered listeners
         * @private
         *
         * @param {function?} [onFulfilled] resolution handler
         * @param {function?} [onRejected] rejection handler
         * @param {function?} [onProgress] progress handler
         */
        _then = function(onFulfilled, onRejected, onProgress) {
            // TODO: Promises/A+ check typeof onFulfilled, onRejected, onProgress
            var deferred, progressHandler;
 
            deferred = defer();
 
            progressHandler = typeof onProgress === 'function'
                ? function(update) {
                    try {
                        // Allow progress handler to transform progress event
                        deferred.progress(onProgress(update));
                    } catch(e) {
                        // Use caught value as progress
                        deferred.progress(e);
                    }
                }
                : function(update) { deferred.progress(update); };
 
            handlers.push(function(promise) {
                promise.then(onFulfilled, onRejected)
                    .then(deferred.resolve, deferred.reject, progressHandler);
            });
 
            progressHandlers.push(progressHandler);
 
            return deferred.promise;
        };
 
        /**
         * Issue a progress event, notifying all progress listeners
         * @private
         * @param {*} update progress event payload to pass to all listeners
         */
        _progress = function(update) {
            processQueue(progressHandlers, update);
            return update;
        };
 
        /**
         * Transition from pre-resolution state to post-resolution state, notifying
         * all listeners of the resolution or rejection
         * @private
         * @param {*} value the value of this deferred
         */
        _resolve = function(value) {
            value = resolve(value);
 
            // Replace _then with one that directly notifies with the result.
            _then = value.then;
            // Replace _resolve so that this Deferred can only be resolved once
            _resolve = resolve;
            // Make _progress a noop, to disallow progress for the resolved promise.
            _progress = noop;
 
            // Notify handlers
            processQueue(handlers, value);
 
            // Free progressHandlers array since we'll never issue progress events
            progressHandlers = handlers = undef;
 
            return value;
        };
 
        return deferred;
 
        /**
         * Wrapper to allow _then to be replaced safely
         * @param {function?} [onFulfilled] resolution handler
         * @param {function?} [onRejected] rejection handler
         * @param {function?} [onProgress] progress handler
         * @return {Promise} new promise
         */
        function then(onFulfilled, onRejected, onProgress) {
            // TODO: Promises/A+ check typeof onFulfilled, onRejected, onProgress
            return _then(onFulfilled, onRejected, onProgress);
        }
 
        /**
         * Wrapper to allow _resolve to be replaced
         */
        function promiseResolve(val) {
            return _resolve(val);
        }
 
        /**
         * Wrapper to allow _reject to be replaced
         */
        function promiseReject(err) {
            return _resolve(rejected(err));
        }
 
        /**
         * Wrapper to allow _progress to be replaced
         */
        function promiseProgress(update) {
            return _progress(update);
        }
    }
 
    /**
     * Determines if promiseOrValue is a promise or not.  Uses the feature
     * test from http://wiki.commonjs.org/wiki/Promises/A to determine if
     * promiseOrValue is a promise.
     *
     * @param {*} promiseOrValue anything
     * @returns {boolean} true if promiseOrValue is a {@link Promise}
     */
    function isPromise(promiseOrValue) {
        return promiseOrValue && typeof promiseOrValue.then === 'function';
    }
 
    /**
     * Initiates a competitive race, returning a promise that will resolve when
     * howMany of the supplied promisesOrValues have resolved, or will reject when
     * it becomes impossible for howMany to resolve, for example, when
     * (promisesOrValues.length - howMany) + 1 input promises reject.
     *
     * @param {Array} promisesOrValues array of anything, may contain a mix
     *      of promises and values
     * @param howMany {number} number of promisesOrValues to resolve
     * @param {function?} [onFulfilled] resolution handler
     * @param {function?} [onRejected] rejection handler
     * @param {function?} [onProgress] progress handler
     * @returns {Promise} promise that will resolve to an array of howMany values that
     * resolved first, or will reject with an array of (promisesOrValues.length - howMany) + 1
     * rejection reasons.
     */
    function some(promisesOrValues, howMany, onFulfilled, onRejected, onProgress) {
 
        checkCallbacks(2, arguments);
 
        return when(promisesOrValues, function(promisesOrValues) {
 
            var toResolve, toReject, values, reasons, deferred, fulfillOne, rejectOne, progress, len, i;
 
            len = promisesOrValues.length >>> 0;
 
            toResolve = Math.max(0, Math.min(howMany, len));
            values = [];
 
            toReject = (len - toResolve) + 1;
            reasons = [];
 
            deferred = defer();
 
            // No items in the input, resolve immediately
            if (!toResolve) {
                deferred.resolve(values);
 
            } else {
                progress = deferred.progress;
 
                rejectOne = function(reason) {
                    reasons.push(reason);
                    if(!--toReject) {
                        fulfillOne = rejectOne = noop;
                        deferred.reject(reasons);
                    }
                };
 
                fulfillOne = function(val) {
                    // This orders the values based on promise resolution order
                    // Another strategy would be to use the original position of
                    // the corresponding promise.
                    values.push(val);
 
                    if (!--toResolve) {
                        fulfillOne = rejectOne = noop;
                        deferred.resolve(values);
                    }
                };
 
                for(i = 0; i < len; ++i) {
                    if(i in promisesOrValues) {
                        when(promisesOrValues[i], fulfiller, rejecter, progress);
                    }
                }
            }
 
            return deferred.then(onFulfilled, onRejected, onProgress);
 
            function rejecter(reason) {
                rejectOne(reason);
            }
 
            function fulfiller(val) {
                fulfillOne(val);
            }
 
        });
    }
 
    /**
     * Initiates a competitive race, returning a promise that will resolve when
     * any one of the supplied promisesOrValues has resolved or will reject when
     * *all* promisesOrValues have rejected.
     *
     * @param {Array|Promise} promisesOrValues array of anything, may contain a mix
     *      of {@link Promise}s and values
     * @param {function?} [onFulfilled] resolution handler
     * @param {function?} [onRejected] rejection handler
     * @param {function?} [onProgress] progress handler
     * @returns {Promise} promise that will resolve to the value that resolved first, or
     * will reject with an array of all rejected inputs.
     */
    function any(promisesOrValues, onFulfilled, onRejected, onProgress) {
 
        function unwrapSingleResult(val) {
            return onFulfilled ? onFulfilled(val[0]) : val[0];
        }
 
        return some(promisesOrValues, 1, unwrapSingleResult, onRejected, onProgress);
    }
 
    /**
     * Return a promise that will resolve only once all the supplied promisesOrValues
     * have resolved. The resolution value of the returned promise will be an array
     * containing the resolution values of each of the promisesOrValues.
     * @memberOf when
     *
     * @param {Array|Promise} promisesOrValues array of anything, may contain a mix
     *      of {@link Promise}s and values
     * @param {function?} [onFulfilled] resolution handler
     * @param {function?} [onRejected] rejection handler
     * @param {function?} [onProgress] progress handler
     * @returns {Promise}
     */
    function all(promisesOrValues, onFulfilled, onRejected, onProgress) {
        checkCallbacks(1, arguments);
        return map(promisesOrValues, identity).then(onFulfilled, onRejected, onProgress);
    }
 
    /**
     * Joins multiple promises into a single returned promise.
     * @return {Promise} a promise that will fulfill when *all* the input promises
     * have fulfilled, or will reject when *any one* of the input promises rejects.
     */
    function join(/* ...promises */) {
        return map(arguments, identity);
    }
 
    /**
     * Traditional map function, similar to `Array.prototype.map()`, but allows
     * input to contain {@link Promise}s and/or values, and mapFunc may return
     * either a value or a {@link Promise}
     *
     * @param {Array|Promise} promise array of anything, may contain a mix
     *      of {@link Promise}s and values
     * @param {function} mapFunc mapping function mapFunc(value) which may return
     *      either a {@link Promise} or value
     * @returns {Promise} a {@link Promise} that will resolve to an array containing
     *      the mapped output values.
     */
    function map(promise, mapFunc) {
        return when(promise, function(array) {
            var results, len, toResolve, resolve, i, d;
 
            // Since we know the resulting length, we can preallocate the results
            // array to avoid array expansions.
            toResolve = len = array.length >>> 0;
            results = [];
            d = defer();
 
            if(!toResolve) {
                d.resolve(results);
            } else {
 
                resolve = function resolveOne(item, i) {
                    when(item, mapFunc).then(function(mapped) {
                        results[i] = mapped;
 
                        if(!--toResolve) {
                            d.resolve(results);
                        }
                    }, d.reject);
                };
 
                // Since mapFunc may be async, get all invocations of it into flight
                for(i = 0; i < len; i++) {
                    if(i in array) {
                        resolve(array[i], i);
                    } else {
                        --toResolve;
                    }
                }
 
            }
 
            return d.promise;
 
        });
    }
 
    /**
     * Traditional reduce function, similar to `Array.prototype.reduce()`, but
     * input may contain promises and/or values, and reduceFunc
     * may return either a value or a promise, *and* initialValue may
     * be a promise for the starting value.
     *
     * @param {Array|Promise} promise array or promise for an array of anything,
     *      may contain a mix of promises and values.
     * @param {function} reduceFunc reduce function reduce(currentValue, nextValue, index, total),
     *      where total is the total number of items being reduced, and will be the same
     *      in each call to reduceFunc.
     * @returns {Promise} that will resolve to the final reduced value
     */
    function reduce(promise, reduceFunc /*, initialValue */) {
        var args = slice.call(arguments, 1);
 
        return when(promise, function(array) {
            var total;
 
            total = array.length;
 
            // Wrap the supplied reduceFunc with one that handles promises and then
            // delegates to the supplied.
            args[0] = function (current, val, i) {
                return when(current, function (c) {
                    return when(val, function (value) {
                        return reduceFunc(c, value, i, total);
                    });
                });
            };
 
            return reduceArray.apply(array, args);
        });
    }
 
    /**
     * Ensure that resolution of promiseOrValue will trigger resolver with the
     * value or reason of promiseOrValue, or instead with resolveValue if it is provided.
     *
     * @param promiseOrValue
     * @param {Object} resolver
     * @param {function} resolver.resolve
     * @param {function} resolver.reject
     * @param {*} [resolveValue]
     * @returns {Promise}
     */
    function chain(promiseOrValue, resolver, resolveValue) {
        var useResolveValue = arguments.length > 2;
 
        return when(promiseOrValue,
            function(val) {
                val = useResolveValue ? resolveValue : val;
                resolver.resolve(val);
                return val;
            },
            function(reason) {
                resolver.reject(reason);
                return rejected(reason);
            },
            resolver.progress
        );
    }
 
    //
    // Utility functions
    //
 
    /**
     * Apply all functions in queue to value
     * @param {Array} queue array of functions to execute
     * @param {*} value argument passed to each function
     */
    function processQueue(queue, value) {
        var handler, i = 0;
 
        while (handler = queue[i++]) {
            handler(value);
        }
    }
 
    /**
     * Helper that checks arrayOfCallbacks to ensure that each element is either
     * a function, or null or undefined.
     * @private
     * @param {number} start index at which to start checking items in arrayOfCallbacks
     * @param {Array} arrayOfCallbacks array to check
     * @throws {Error} if any element of arrayOfCallbacks is something other than
     * a functions, null, or undefined.
     */
    function checkCallbacks(start, arrayOfCallbacks) {
        // TODO: Promises/A+ update type checking and docs
        var arg, i = arrayOfCallbacks.length;
 
        while(i > start) {
            arg = arrayOfCallbacks[--i];
 
            if (arg != null && typeof arg != 'function') {
                throw new Error('arg '+i+' must be a function');
            }
        }
    }
 
    /**
     * No-Op function used in method replacement
     * @private
     */
    function noop() {}
 
    slice = [].slice;
 
    // ES5 reduce implementation if native not available
    // See: http://es5.github.com/#x15.4.4.21 as there are many
    // specifics and edge cases.
    reduceArray = [].reduce ||
        function(reduceFunc /*, initialValue */) {
            /*jshint maxcomplexity: 7*/
 
            // ES5 dictates that reduce.length === 1
 
            // This implementation deviates from ES5 spec in the following ways:
            // 1. It does not check if reduceFunc is a Callable
 
            var arr, args, reduced, len, i;
 
            i = 0;
            // This generates a jshint warning, despite being valid
            // "Missing 'new' prefix when invoking a constructor."
            // See https://github.com/jshint/jshint/issues/392
            arr = Object(this);
            len = arr.length >>> 0;
            args = arguments;
 
            // If no initialValue, use first item of array (we know length !== 0 here)
            // and adjust i to start at second item
            if(args.length <= 1) {
                // Skip to the first real element in the array
                for(;;) {
                    if(i in arr) {
                        reduced = arr[i++];
                        break;
                    }
 
                    // If we reached the end of the array without finding any real
                    // elements, it's a TypeError
                    if(++i >= len) {
                        throw new TypeError();
                    }
                }
            } else {
                // If initialValue provided, use it
                reduced = args[1];
            }
 
            // Do the actual reduce
            for(;i < len; ++i) {
                // Skip holes
                if(i in arr) {
                    reduced = reduceFunc(reduced, arr[i], i, arr);
                }
            }
 
            return reduced;
        };
 
    function identity(x) {
        return x;
    }
 
    return when;
  });
  })(function (factory) { (module.exports = factory())
        ;
    }
    // Boilerplate for AMD, Node, and browser global
  );
  });
 
  exports.commonjsGlobal = commonjsGlobal;
  exports.createCommonjsModule = createCommonjsModule;
  exports.defaultValue = defaultValue;
  exports.defined = defined;
  exports.when = when;
 
}));