I've got an object that looks like this:
state: {
"1": {
"show": false,
"description": "one",
"children": {
"1": { "show": false, "description": "one" },
"2": { "show": false, "description": "one" }
}
},
"2": {
"show": false,
"description": "one",
"children": {
"1": { "show": false, "description": "one" },
"2": { "show": false, "description": "one" }
}
}
}
I've got a for loop that change the children "show" property to the opposite boolean. So I try to update the value with this but doesn't worked.
for (var childKey in state[appClassId].children) {
newState = {
...state,
[appClassId]: {
children: {
[childKey]: { ...state[appClassId].children[childKey], show: !state[appClassId].children[childKey].show}
}
}
"appClassId" variable is a variable that I get from the action.
How can I update every key in the child property for instance state.1.children.1.show
I've got an object that looks like this:
state: {
"1": {
"show": false,
"description": "one",
"children": {
"1": { "show": false, "description": "one" },
"2": { "show": false, "description": "one" }
}
},
"2": {
"show": false,
"description": "one",
"children": {
"1": { "show": false, "description": "one" },
"2": { "show": false, "description": "one" }
}
}
}
I've got a for loop that change the children "show" property to the opposite boolean. So I try to update the value with this but doesn't worked.
for (var childKey in state[appClassId].children) {
newState = {
...state,
[appClassId]: {
children: {
[childKey]: { ...state[appClassId].children[childKey], show: !state[appClassId].children[childKey].show}
}
}
"appClassId" variable is a variable that I get from the action.
How can I update every key in the child property for instance state.1.children.1.show
Share Improve this question asked Jan 27, 2017 at 18:33 Diego UnanueDiego Unanue 6,8469 gold badges48 silver badges69 bronze badges 3- 2 Could you look at Normalizing State Shape from the redux documentation? I think it will help you to normalize your state and be able to update it easily! – cdaiga Commented Jan 27, 2017 at 18:39
- It's normalized take a look at the example it gives in your link. – Diego Unanue Commented Jan 27, 2017 at 18:59
- 2 Actually, as author of that "Structuring Reducers" section, I think the "Immutable Update Patterns" section is more relevant: redux.js/docs/recipes/reducers/ImmutableUpdatePatterns.html – markerikson Commented Jan 27, 2017 at 19:17
4 Answers
Reset to default 4With the help of @markerikson author of:
http://redux.js/docs/recipes/reducers/ImmutableUpdatePatterns.html
I was able to update that deep level of nested data:
The object to update are the all objects under "children" property of this object:
state: {
"1": {
"show": false,
"description": "one",
"children": {
"1": { "show": false, "description": "one" },
"2": { "show": false, "description": "one" }
}
},
"2": {
"show": false,
"description": "one",
"children": {
"1": { "show": false, "description": "one" },
"2": { "show": false, "description": "one" }
}
}
}
And the code to update it is:
let newState = {};
newState = { ...state, newState }
for (var childKey in newState[appClassId].children) {
newState = {
...newState,
[appClassId]: {
...newState[appClassId],
children: {
...newState[appClassId].children,
[childKey]: {
...newState[appClassId].children[childKey],
show: !newState[appClassId].children[childKey].show
}
}
}
}
}
I would remend leaning on a utility library like lodash.js
in addition to the spread operator if you're doing anything more plex than assigning a value or two.
Assuming that the number of appClassId
s inside state
and children
inside each appClass
are pletely dynamic:
import { reduce } from 'lodash'
const newState = reduce(state, (modifiedState, appClass, appClassId) => {
const { children } = appClass
// toggle show for each child (non-mutating)
const toggledChildren = reduce(children, (newChildren, child, childId) => {
return {
...newChildren,
[childId]: { ...child, show: !child.show }
}
}, {})
// persist modified children within the appClass
return {
...modifiedState,
[appClassId]: {
...appClass,
children: toggledChildren
}
}
}, {})
Hope this helps!
try this:
const originalChildren = state[appClassId].children;
let updatedChildren = {};
for (var child in originalChildren) {
if (originalChildren.hasOwnProperty(child)) {
updatedChildren = {
...updatedChildren,
[child]: { ...originalChildren[child], show: !originalChildren[child].show }
};
}
}
const newState = {
...state,
[appClassId]: {
...state[appClassId],
children: { ...originalChildren, ...updatedChildren }
}
}
Another example,
let test = [
{ id: 1, name: 'Phone',
Devices: [
{ id: 1, name: 'BlackBerry',GroupId: 1 },
{ id: 2, name: 'Iphone', GroupId: 1 }
]
}, { id: 2, name: 'Speaker',
Devices: [
{ id: 3, name: 'Senheiser', GroupId: 2 },
{ id: 4, name: 'Simbadda', GroupId: 2 }
]
}
];
const GroupId = 1;
const device = {
id: 5,
name: 'Android',
GroupId: GroupId
}
let Group = [...test];
const idx = test.findIndex(payload => {
return payload.id === GroupId
});
console.log(Group[idx].Devices = [...Group[idx].Devices, device]);
Hope it will help