Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Underscore.js groupBy multiple values

Tags:

Using Underscore.js, I'm trying to group a list of items multiple times, ie

Group by SIZE then for each SIZE, group by CATEGORY...

http://jsfiddle.net/rickysullivan/WTtXP/1/

Ideally, I'd like to have a function or extend _.groupBy() so that you can throw an array at it with the paramaters to group by.

var multiGroup = ['size', 'category']; 

Probably could just make a mixin...

_.mixin({     groupByMulti: function(obj, val, arr) {         var result = {};         var iterator = typeof val == 'function' ? val : function(obj) {                 return obj[val];             };         _.each(arr, function(arrvalue, arrIndex) {             _.each(obj, function(value, objIndex) {                 var key = iterator(value, objIndex);                 var arrresults = obj[objIndex][arrvalue];                 if (_.has(value, arrvalue))                     (result[arrIndex] || (result[arrIndex] = [])).push(value); 

My head hurts, but I think some more pushing needs to go here...

            });         })         return result;     } });  properties = _.groupByMulti(properties, function(item) {      var testVal = item["size"];      if (parseFloat(testVal)) {         testVal = parseFloat(item["size"])     }      return testVal  }, multiGroup); 
like image 612
rickysullivan Avatar asked Apr 05 '12 03:04

rickysullivan


2 Answers

A simple recursive implementation:

_.mixin({   /*    * @mixin    *    * Splits a collection into sets, grouped by the result of running each value    * through iteratee. If iteratee is a string instead of a function, groups by    * the property named by iteratee on each of the values.    *    * @param {array|object} list - The collection to iterate over.    * @param {(string|function)[]} values - The iteratees to transform keys.    * @param {object=} context - The values are bound to the context object.    *     * @returns {Object} - Returns the composed aggregate object.    */   groupByMulti: function(list, values, context) {     if (!values.length) {       return list;     }     var byFirst = _.groupBy(list, values[0], context),         rest    = values.slice(1);     for (var prop in byFirst) {       byFirst[prop] = _.groupByMulti(byFirst[prop], rest, context);     }     return byFirst;   } }); 

Demo in your jsfiddle

like image 200
Bergi Avatar answered Nov 08 '22 02:11

Bergi


I think @Bergi's answer can be streamlined a bit by utilizing Lo-Dash's mapValues (for mapping functions over object values). It allows us to group the entries in an array by multiple keys in a nested fashion:

_ = require('lodash');  var _.nest = function (collection, keys) {   if (!keys.length) {     return collection;   }   else {     return _(collection).groupBy(keys[0]).mapValues(function(values) {        return nest(values, keys.slice(1));     }).value();   } }; 

I renamed the method to nest because it ends up serving much the same role served by D3's nest operator. See this gist for details and this fiddle for demonstrated usage with your example.

lodash nest groupby

like image 30
joyrexus Avatar answered Nov 08 '22 02:11

joyrexus