I am stuck on a problem trying to convert a flat array of objects into a nested array of objects based the name property.
What is the best way to convert the input
array to resemble the structure of the desiredOutput
array?
var input = [
{
name: 'foo',
url: '/somewhere1',
templateUrl: 'foo.tpl.html',
title: 'title A',
subtitle: 'description A'
},
{
name: 'foo.bar',
url: '/somewhere2',
templateUrl: 'anotherpage.tpl.html',
title: 'title B',
subtitle: 'description B'
},
{
name: 'buzz.fizz',
url: '/another/place',
templateUrl: 'hello.tpl.html',
title: 'title C',
subtitle: 'description C'
},
{
name: 'foo.hello.world',
url: '/',
templateUrl: 'world.tpl.html',
title: 'title D',
subtitle: 'description D'
}
]
var desiredOutput = [
{
name: 'foo',
url: '/somewhere1',
templateUrl: 'foo.tpl.html',
data: {
title: 'title A',
subtitle: 'description A'
},
children: [
{
name: 'bar',
url: '/somewhere2',
templateUrl: 'anotherpage.tpl.html',
data: {
title: 'title B',
subtitle: 'description B'
}
},
{
name: 'hello',
data: {},
children: [
{
name: 'world',
url: '/',
templateUrl: 'world.tpl.html',
data: {
title: 'title D',
subtitle: 'description D'
}
}
]
}
]
},
{
name: 'buzz',
data: {},
children: [
{
name: 'fizz',
url: '/',
templateUrl: 'world.tpl.html',
data: {
title: 'title C',
subtitle: 'description C'
}
}
]
}
]
Note the order of the objects in the input array is not guaranteed. This code will be running in a Node.js environment and I am open to using libraries such as lodash to achieve the desired output.
Any help is greatly appreciated.
Using Lodash (because why on earth would you want to manipulate complex data without a utility library). Here's the fiddle.
It works by reshaping the initial input so as to split the names into arrays and add the data / children properties. Then it reduces the data over
buildTree
which uses a mutating function ( :( ) to insert the current item in the reduce at the given path.The strange
if (!match)
part makes sure that missing segments are added in if they're not explicitly specified in the initial data set with a URL etc.The last two lines that actually do the work should probably be in a little function, and it could do with some JSDoc. It's just a shame I didn't get it completely recursive, I'm relying on array mutation to insert the route object deep within the tree.
Should be simple enough to follow though.