Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Splitting dots into separate objects javascript

I have an object like this:

var data = {"prop.health": 1, "prop.cost":1, "prop.time":1}

I want to change it into an object like this:

{
  "prop": {
    "health": 1, 
    "cost":1, 
    "time":1
  }
}

Here's my code:

  _.each(data, function (value, key) {
    var split = key.split('.')
    if (split.length > 1) {
      data[split[0]] = data[split[0]] || {}
      data[split[0]][split[1]] = value
      delete data[key]
    }
  })

But this only works for 1 level of nesting. How would you write it to ensure it works for as deeply nested properties as you need?

like image 624
Harry Avatar asked Nov 12 '15 00:11

Harry


2 Answers

You can use a combination of _.transform and _.set, for example

data = _.transform(data, function(transformed, val, key) {
    _.set(transformed, key, val);
});

Results in

{"prop":{"health":1,"cost":1,"time":1}}
like image 91
Phil Avatar answered Nov 09 '22 15:11

Phil


Without a library it would be something like this:

(function(){
    var data = {"prop.health": 1, "prop.cost":1, "prop.time":1, "prop.test.fun" : 1, "prop.test.sun" : 1};
    var obj = {};  //will hold the object all parsed out
    Object.keys(data).forEach( function (key) {  //loop through the keys in the object
        var val = data[key];  //grab the value of this key
        var step = obj;  //reference the object that holds the values
        key.split(".").forEach(function(part, index, arr){   //split the parts and loop
            if(index===arr.length-1){  //If we are at the last index, than we set the value
                step[part] = val;
            } else if(step[part]===undefined) {  //If we have not seen this key before, create an object
                step[part] = {};
            }
            step = step[part];  //Step up the object we are referencing 
        });
    } );
    console.log(obj);
}());

Or the double reduce loop

(function(){
    var data = {"prop.health": 1, "prop.cost":1, "prop.time":1, "prop.test.fun" : 1, "prop.test.sun" : 1};
    var result = Object.keys(data).reduce( function (obj, key) {  //loop through the keys in the object
        var val = data[key];  //grab the value of this key
        key.split(".").reduce(function(step, part, index, arr){   //split the parts and loop
            if(index===arr.length-1){  //If we are at the last index, than we set the value
                step[part] = val;
            } else if(step[part]===undefined) {  //If we have not seen this key before, create an object
                step[part] = {};
            }
            return step[part];  //Step up the object we are referencing
        }, obj);
        return obj;
    }, {});
    console.log(result);
}());
like image 1
epascarello Avatar answered Nov 09 '22 17:11

epascarello