最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Update very deep object with spread operator - Stack Overflow

programmeradmin1浏览0评论

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
Add a ment  | 

4 Answers 4

Reset to default 4

With 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 appClassIds 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

发布评论

评论列表(0)

  1. 暂无评论
ok 不同模板 switch ($forum['model']) { /*case '0': include _include(APP_PATH . 'view/htm/read.htm'); break;*/ default: include _include(theme_load('read', $fid)); break; } } break; case '10': // 主题外链 / thread external link http_location(htmlspecialchars_decode(trim($thread['description']))); break; case '11': // 单页 / single page $attachlist = array(); $imagelist = array(); $thread['filelist'] = array(); $threadlist = NULL; $thread['files'] > 0 and list($attachlist, $imagelist, $thread['filelist']) = well_attach_find_by_tid($tid); $data = data_read_cache($tid); empty($data) and message(-1, lang('data_malformation')); $tidlist = $forum['threads'] ? page_find_by_fid($fid, $page, $pagesize) : NULL; if ($tidlist) { $tidarr = arrlist_values($tidlist, 'tid'); $threadlist = well_thread_find($tidarr, $pagesize); // 按之前tidlist排序 $threadlist = array2_sort_key($threadlist, $tidlist, 'tid'); } $allowpost = forum_access_user($fid, $gid, 'allowpost'); $allowupdate = forum_access_mod($fid, $gid, 'allowupdate'); $allowdelete = forum_access_mod($fid, $gid, 'allowdelete'); $access = array('allowpost' => $allowpost, 'allowupdate' => $allowupdate, 'allowdelete' => $allowdelete); $header['title'] = $thread['subject']; $header['mobile_link'] = $thread['url']; $header['keywords'] = $thread['keyword'] ? $thread['keyword'] : $thread['subject']; $header['description'] = $thread['description'] ? $thread['description'] : $thread['brief']; $_SESSION['fid'] = $fid; if ($ajax) { empty($conf['api_on']) and message(0, lang('closed')); $apilist['header'] = $header; $apilist['extra'] = $extra; $apilist['access'] = $access; $apilist['thread'] = well_thread_safe_info($thread); $apilist['thread_data'] = $data; $apilist['forum'] = $forum; $apilist['imagelist'] = $imagelist; $apilist['filelist'] = $thread['filelist']; $apilist['threadlist'] = $threadlist; message(0, $apilist); } else { include _include(theme_load('single_page', $fid)); } break; default: message(-1, lang('data_malformation')); break; } ?>