1

Below is my array object

[{ role : 'Role 1', name: 'A', slid: 'a'}, {role : 'Role 1', name: 'B',slid: 'b'}, {role : 'Role 2', name: 'X', slid: 'x'}, {role : 'Role 2',name: 'Y', slid: 'y'},{role : 'Role 3',name: 'Z',slid: 'z'}]

How to combine the objects in the array by using role and form new array objects in the below format.

[{role : 'Role 1', list: [{name: 'A', slid: 'a'}, {name: 'B', slid: 'b'}]},
 {role : 'Role 2', list: [{name: 'X, slid: 'x'}, {name: 'Y', slid: 'y'}]},
 {role : 'Role 3', list: [{name: 'Z', slid: 'z'}]}]

I have looked into many examples, but could not find a solution.

4 Answers 4

3

You can use reduce to group the array into an object. Use Object.values to convert the object into an array.

var arr = [{"role":"Role 1","name":"A","slid":"a"},{"role":"Role 1","name":"B","slid":"b"},{"role":"Role 2","name":"X","slid":"x"},{"role":"Role 2","name":"Y","slid":"y"},{"role":"Role 3","name":"Z","slid":"z"}];

var result = Object.values(arr.reduce((c, {role,...r}) => {
  c[role] = c[role] || {role,list: []};
  c[role].list.push(r);
  return c;
}, {}));

console.log( result );

Sign up to request clarification or add additional context in comments.

Comments

0

const arr = [{
  role: 'Role 1',
  name: 'A',
  slid: 'a',
}, {
  role: 'Role 1',
  name: 'B',
  slid: 'b',
}, {
  role: 'Role 2',
  name: 'X',
  slid: 'x',
}, {
  role: 'Role 2',
  name: 'Y',
  slid: 'y',
}, {
  role: 'Role 3',
  name: 'Z',
  slid: 'z',
}];

// We go through every array element
const ret = arr.reduce((tmp, x) => {
  // Does we have already the x.role stored in our final array ?
  const elem = tmp.find(y => y.role === x.role);

  // Yes, so we push a new value inside of it
  if (elem) {
    elem.list.push({
      name: x.name,
      slid: x.slid
    });

    return tmp;
  }

  // No, so we create a new array element
  tmp.push({
    role: x.role,
    
    list: [{
      name: x.name,
      slid: x.slid
    }],
  });
  
  return tmp;
}, []);

console.log(ret);

1 Comment

Thanks for a quick response.
0

Just a different approach with Array.map

let input = [{ role : 'Role 1', name: 'A', slid: 'a'}, {role : 'Role 1', name: 'B',slid: 'b'}, {role : 'Role 2', name: 'X', slid: 'x'}, {role : 'Role 2',name: 'Y', slid: 'y'},{role : 'Role 3',name: 'Z',slid: 'z'}]


let res = input.sort((a,b) => a.role.localeCompare(b.role))
               .map(({ role, ...rest }) => ({ role, list: [rest] }))
               .flatMap(({ role, list }, i, t) => 
                     role == (t[i+1] && t[i+1].role)
                       ? (t[i+1].list.unshift(list[0]), [])
                       : { role, list })

console.log(res)

Comments

0

You could use reduce function to solve the issue

const arr = [{ role : 'Role 1', name: 'A', slid: 'a'}, {role : 'Role 1', name: 'B',slid: 'b'}, {role : 'Role 2', name: 'X', slid: 'x'}, {role : 'Role 2',name: 'Y', slid: 'y'},{role : 'Role 3',name: 'Z',slid: 'z'}]

var groupBy = function(array, key) {
  return array.reduce(function(reduceVal, obj) {
    (reduceVal[obj[key]] = reduceVal[obj[key]] || []).push(obj);
    return reduceVal;
  }, {});
};

console.log(groupBy(arr, 'role'));

//{Role 1: Array(2), Role 2: Array(2), Role 3: Array(1)}

Comments

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.