Source: documents/collector.js

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
"use strict";
/**
 @fileOverview An object and array collector
 @module ink/collector
 */

var probe = require( "ink-probe" );
var sys = require( "lodash" );
var dcl = require( "dcl" );

/**
 * A collector
 * @constructor
 */
var CollectorBase = dcl( Destroyable, {
    declaredClass : "CollectorBase",
    constructor   : function ( obj ) {
        var that = this;
        if ( obj && !sys.isObject( obj ) ) {
            throw new TypeError( "Collectors require an initial object or array passed to the constructor" );
        }
        /**
         * The collection that being managed
         * @type {object|array}
         */
        this.heap = obj || {};
        // mixin the probe
        probe.mixTo( this, this.heap );
        /**
         * Get the size of the collection
         * @name length
         * @type {number}
         * @memberOf module:documents/collector~CollectorBase#
         */
        Object.defineProperty( this, "length", {
                get : function () {
                    return sys.size( that.heap );
                }
            }
        );
        /**
         * Creates an array of shuffled array values, using a version of the Fisher-Yates shuffle.
         * See http://en.wikipedia.org/wiki/Fisher-Yates_shuffle.
         * @function
         * @memberOf module:documents/collector~CollectorBase#
         * @returns {array}
         */
        this.shuffle = sys.bind( sys.shuffle, this, this.heap );

    },
    /**
     * Adds an item to the collection
     * @param {*} key The key to use for the item being added.
     * @param {*} item The item to add to the collection. The item is not iterated so that you could add bundled items to the collection
     */
    add           : function ( key, item ) {
        this.heap[key] = item;
    },
    /**
     * Iterate over each item in the collection, or a subset that matches a query. This supports two signatures:
     * `.each(query, function)` and `.each(function)`. If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {object=} query A query to evaluate
     * @param {function(val, key)} iterator Function to execute against each item in the collection
     * @param {object=} thisobj The value of `this`
     */
    each          : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            sys.each( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            sys.each( this.heap, query, thisobj );
        }
    },
    /**
     * Returns the collection as an array. If it is already an array, it just returns that.
     * @return {array}
     */
    toArray       : function () {
        return sys.toArray( this.heap );
    },
    /**
     * Supports conversion to a JSON string or for passing over the wire
     * @return {object}
     * @returns {Object|array}
     */
    toJSON        : function () {
        return this.heap;
    },
    /**
     * Maps the contents to an array by iterating over it and transforming it. You supply the iterator. Supports two signatures:
     * `.map(query, function)` and `.map(function)`. If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {object=} query A query to evaluate
     * @param {function(val, key)} iterator Function to execute against each item in the collection
     * @param {object=} thisobj The value of `this`
     */
    map           : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.map( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            return sys.map( this.heap, query, thisobj );
        }
    },
    /**
     * Reduces a collection to a value which is the accumulated result of running each element in the collection through the
     * callback, where each successive callback execution consumes the return value of the previous execution. If accumulator
     * is not passed, the first element of the collection will be used as the initial accumulator value.
     * are iterated over.
     * @param {object=} query A query to evaluate
     * @param {function(result, val, key)} iterator The function that will be executed in each item in the collection
     * @param {*=} accumulator Initial value of the accumulator.
     * @param {object=} thisobj The value of `this`
     * @return {*}
     */
    reduce        : function ( query, iterator, accumulator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.reduce( this.find( query ), iterator, accumulator, thisobj );
        } else {
            thisobj = accumulator || this;
            return  sys.reduce( this.heap, query, iterator, thisobj );
        }
    },
    /**
     * Creates an object composed of keys returned from running each element
     * of the collection through the given callback. The corresponding value of each key
     * is the number of times the key was returned by the callback.
     * @param {object=} query A query to evaluate. If you pass in a query, only the items that match the query
     * are iterated over.
     * @param  {function(value, key, collection)} iterator
     * @param {object=} thisobj The value of `this`
     * @return {object}
     */
    countBy       : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.countBy( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            return sys.countBy( this.heap, query, thisobj );
        }
    },
    /**
     * Creates an object composed of keys returned from running each element of the collection through the callback.
     * The corresponding value of each key is an array of elements passed to callback that returned the key.
     * The callback is invoked with three arguments: (value, index|key, collection).
     * @param {object=} query A query to evaluate . If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {function(value, key, collection)} iterator
     * @param {object=} thisobj The value of `this`
     * @return {object}
     */
    groupBy       : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.groupBy( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            return sys.groupBy( this.heap, query, thisobj );
        }
    },
    /**
     * Reduce the collection to a single value. Supports two signatures:
     * `.pluck(query, function)` and `.pluck(function)`
     * @param {object=} query The query to evaluate. If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {string} property The property that will be 'plucked' from the contents of the collection
     * @return {*}
     */
    pluck         : function ( query, property ) {
        if ( arguments.length === 2 ) {
            return sys.map( this.find( query ), function ( record ) {
                return probe.get( record, property );
            } );
        } else {
            return sys.map( this.heap, function ( record ) {
                return probe.get( record, query );
            } );
        }
    },
    /**
     * Returns a sorted copy of the collection.
     * @param {object=} query The query to evaluate. If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {function(value, key)} iterator
     * @param {object=} thisobj The value of `this`
     * @return {array}
     */
    sortBy        : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.sortBy( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            return sys.sortBy( this.heap, query, thisobj );
        }
    },
    /**
     * Retrieves the maximum value of an array. If callback is passed,
     * it will be executed for each value in the array to generate the criterion by which the value is ranked.
     * @param {object=} query A query to evaluate . If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {function(value, key, collection)} iterator
     * @param {object=} thisobj The value of `this`
     * @return {number}
     */
    max           : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.max( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            return sys.max( this.heap, query, thisobj );
        }
    },
    /**
     * Retrieves the minimum value of an array. If callback is passed,
     * it will be executed for each value in the array to generate the criterion by which the value is ranked.
     * @param {object=} query A query to evaluate . If you pass in a query, only the items that match the query
     * are iterated over.
     * @param {function(value, key, collection)} iterator
     * @param {object=} thisobj The value of `this`
     * @return {number}
     */
    min           : function ( query, iterator, thisobj ) {
        if ( sys.isPlainObject( query ) ) {
            thisobj = thisobj || this;
            return sys.min( this.find( query ), iterator, thisobj );
        } else {
            thisobj = iterator || this;
            return sys.min( this.heap, query, thisobj );
        }
    },
    /**
     * Destructor called when the object is destroyed.
     */
    destroy       : function () {
        this.heap = null;
    }
} );

/**
 * An object based collector
 * @extends module:documents/collector~CollectorBase
 * @constructor
 */
var OCollector = dcl( CollectorBase, {
    /**
     * Get a record by key
     * @param {*} key The key of the record to get
     * @return {*}
     */
    key : function ( key ) {
        return this.heap[key];
    }
} );

//noinspection JSCommentMatchesSignature
/**
 An array based collector
 @extends module:documents/collector~CollectorBase
 @constructor
 */
var ACollector = dcl( CollectorBase, {
        constructor : function ( obj ) {
            if ( obj && !sys.isArray( obj ) ) {
                throw new TypeError( "Collectors require an array passed to the constructor" );
            }
            this.heap = obj || [];
            /**
             * Creates an array of array elements not present in the other arrays using strict equality for comparisons, i.e. ===.
             * @returns {array}
             */
            this.difference = sys.bind( sys.difference, this, this.heap );
            /**
             * This method gets all but the first values of array
             * @param {number=} n The numer of items to return
             * @returns {*}
             */
            this.tail = sys.bind( sys.tail, this, this.heap );
            /**
             * Gets the first n values of the array
             * @param {number=} n The numer of items to return
             * @returns {*}
             */
            this.head = sys.bind( sys.head, this, this.heap );
        },
        /**
         * Adds to the top of the collection
         * @param {*} item The item to add to the collection. Only one item at a time can be added
         */
        add         : function ( item ) {
            this.heap.unshift( item );
        },
        /**
         * Add to the bottom of the list
         * @param {*} item The item to add to the collection.  Only one item at a time can be added
         */
        append      : function ( item ) {
            this.heap.push( item );
        },
        /**
         * Add an item to the top of the list. This is identical to `add`, but is provided for stack semantics
         * @param {*} item The item to add to the collection. Only one item at a time can be added
         */
        push        : function ( item ) {
            this.add( item );
        },
        /**
         * Modifies the collection with all falsey values of array removed. The values false, null, 0, "", undefined and NaN are all falsey.
         */
        compact     : function () {
            this.heap = sys.compact( this.heap );
        },
        /**
         * Creates an array of elements from the specified indexes, or keys, of the collection. Indexes may be specified as
         * individual arguments or as arrays of indexes
         * @param {indexes} args The indexes to use
         */
        at          : function () {
            var arr = sys.toArray( arguments );
            arr.unshift( this.heap );
            return sys.at.apply( this, arr );
        },
        /**
         * Flattens a nested array (the nesting can be to any depth). If isShallow is truthy, array will only be flattened a single level.
         * If callback is passed, each element of array is passed through a callback before flattening.
         * @param {object=} query A query to evaluate . If you pass in a query, only the items that match the query
         * are iterated over.
         * @param {function(value, key, collection)} iterator,
         * @param {object=} thisobj The value of `this`
         * @return {number}
         */
        flatten     : function ( query, iterator, thisobj ) {
            if ( sys.isPlainObject( query ) ) {
                thisobj = thisobj || this;
                return sys.flatten( this.find( query ), iterator, thisobj );
            } else {
                thisobj = iterator || this;
                return sys.flatten( this.heap, query, thisobj );
            }
        },
        /**
         * Gets an items by its index
         * @param {number} key The index to get
         * @return {*}
         */
        index       : function ( index ) {
            return this.heap[ index ];
        }
    }
);

/**
 Collect an object
 @param {array|object} obj What to collect
 @return {ACollector|OCollector}
 */
exports.collect = function ( obj ) {
    if ( sys.isArray( obj ) ) {
        return new ACollector( obj );
    } else {
        return new OCollector( obj );
    }
};

exports.array = function ( obj ) {
    return new ACollector( obj );
};

exports.object = function ( obj ) {
    return new OCollector( obj );
};

/**
 Returns true if all items match the query. Aliases as `all`
 @function

 @param {object} qu The query to execute
 @returns {boolean}
 @name every
 @memberOf module:documents/collector~CollectorBase#
 */


/**
 Returns true if any of the items match the query. Aliases as `any`
 @function

 @param {object} qu The query to execute
 @returns {boolean}
 @memberOf module:documents/collector~CollectorBase#
 @name some
 */


/**
 Returns the set of unique records that match a query

 @param {object} qu The query to execute.
 @return {array}
 @memberOf module:documents/collector~CollectorBase#
 @name unique
 @method
 **/

/**
 Returns true if all items match the query. Aliases as `every`
 @function

 @param {object} qu The query to execute
 @returns {boolean}
 @name all
 @memberOf module:documents/collector~CollectorBase#
 */


/**
 Returns true if any of the items match the query. Aliases as `all`
 @function

 @param {object} qu The query to execute
 @returns {boolean}
 @memberOf module:documents/collector~CollectorBase#
 @name any
 */


/**
 Remove all items in the object/array that match the query

 @param {object} qu The query to execute. See {@link module:ink/probe.queryOperators} for the operators you can use.
 @return {object|array} The array or object as appropriate without the records.
 @memberOf module:documents/collector~CollectorBase#
 @name remove
 @method
 **/

/**
 Returns the first record that matches the query and returns its key or index depending on whether `obj` is an object or array respectively.
 Aliased as `seekKey`.

 @param {object} qu The query to execute.
 @returns {object}
 @memberOf module:documents/collector~CollectorBase#
 @name findOneKey
 @method
 */


/**
 Returns the first record that matches the query. Aliased as `seek`.

 @param {object} qu The query to execute.
 @returns {object}
 @memberOf module:documents/collector~CollectorBase#
 @name findOne
 @method
 */


/**
 Find all records that match a query and returns the keys for those items. This is similar to {@link module:ink/probe.find} but instead of returning
 records, returns the keys. If `obj` is an object it will return the hash key. If 'obj' is an array, it will return the index

 @param {object} qu The query to execute.
 @returns {array}
 @memberOf module:documents/collector~CollectorBase#
 @name findKeys
 @method
 */


/**
 Find all records that match a query

 @param {object} qu The query to execute.
 @returns {array} The results
 @memberOf module:documents/collector~CollectorBase#
 @name find
 @method
 **/

/**
 Updates all records in obj that match the query. See {@link module:ink/probe.updateOperators} for the operators that are supported.

 @param {object} qu The query which will be used to identify the records to updated
 @param {object} setDocument The update operator. See {@link module:ink/probe.updateOperators}
 @memberOf module:documents/collector~CollectorBase#
 @name update
 @method
 */