开发者问题收集

React:如何为复选框数组设置动态状态

2017-08-04
3003

我有一个 ServiceCard 组件,它接受一个对象。此对象可能有一个权限键,其中包含权限数组,每个权限都带有 disabled:truedisabled:false

如何为这种用例设置基本状态?

Service 对象的权限数组是什么样的:

在此处输入图像描述

(刚刚意识到我拼错了 turret :p)

构造函数和权限映射函数

constructor(props) {
  super(props);
  this.onChangeHandler = this.onChangeHandler.bind(this);
  this.updateService = this.updateService.bind(this);

  // How to define this.state for dynamic Rights Array?

  this.state = {
    serviceName: props.name,
    rights: props.rights
  }

  props.rights.map(right => {
    console.log(' map right', right)
    console.log(' this.state', this.state)
  })
}


// Later down in the render...
<ul>
  {
    rights.map(right =>
      <li key={ right.name }>
        <input type="checkbox"
               checked={ !right.disabled }
               onChange={ () => this.onChangeHandler(name, right) }/>
        <p>{ right.name }</p>
      </li>
    )
  }
</ul>

完整组件代码

import React, { Component } from 'react'

export default class ServiceCard extends React.Component {
  constructor(props) {
    super(props);
    this.onChangeHandler = this.onChangeHandler.bind(this);
    this.updateService = this.updateService.bind(this);
    this.deleteService = this.deleteService.bind(this);
  }

  onChangeHandler(name, right) {
    console.log('onChangeHandler...')
    console.log(' name', name)
    console.log(' right', right)

  }

  deleteService(serviceName) {
    this.props.deleteService(serviceName);
  }

  render () {
    const name = this.props.name;
    // console.log('this.props', this.props)
    const rights = this.props.rights;

    return (
      <li className="service-card-item">
        <section className="card">
          <div className="icon-connectdevelop service-icon"></div>
          <div className="service-details">
            <h3>{ name }</h3>
            <section className="rights-section">
              <ul>
                {
                  rights.map(right =>
                    <li key={ right.name }>
                      <input type="checkbox" checked={ !right.disabled } onChange={ () => this.onChangeHandler(name, right) }/>
                      <p>{ right.name }</p>
                    </li>
                  )
                }
              </ul>
              <div className="icon-cancel-circled" title="Delete Service" onClick={ () => this.deleteService(name) }></div>
            </section>
          </div>
        </section>
      </li>
    )
  }
}
1个回答

正如 Cloud Tseng 提供的链接中所述,我最初使用权限数组设置了我的状态:

this.state = {
  serviceName: props.name,
  rights: props.rights
}

然后,只需获取复选框 changeHandled 的权限的名称,使用 Ramda 进行一些函数映射,返回一个新的权限数组,然后将权限状态设置为新更新的数组:

onChangeHandler 具有 Ramda 功能性感

onChangeHandler(name, right) {
  const serviceRights = this.state.rights;

  const findRight = R.curry((updatedRight, propsRight) => {
    if (updatedRight.name === propsRight.name) {
      propsRight.disabled = !propsRight.disabled;
    }
    return propsRight;
  });

  const updatedRights = R.map(findRight(right), serviceRights);
  console.log('  updatedRights', updatedRights)

  this.setState({
    rights: updatedRights
  }, console.log('  >>> State update completed:', this.state));
}

之前

在此处输入图像描述

之后

在此处输入图片描述

完整修复代码

import React, { Component } from 'react'
import * as R from 'ramda'

export default class ServiceCard extends React.Component {
  constructor(props) {
    super(props);
    console.log('ServiceCard', props)
    this.onChangeHandler = this.onChangeHandler.bind(this);
    this.updateService = this.updateService.bind(this);
    this.deleteService = this.deleteService.bind(this);

    this.state = {
      serviceName: props.name,
      rights: props.rights
    }

    props.rights.map(right => {
      console.log(' map right', right)
      console.log(' this.state', this.state)
    });
  }

  onChangeHandler(name, right) {
    console.log('onChangeHandler...')

    const serviceRights = this.state.rights;

    const findRight = R.curry((updatedRight, propsRight) => {
      if (updatedRight.name === propsRight.name) {
        propsRight.disabled = !propsRight.disabled;
      }
      return propsRight;
    });

    const updatedRights = R.map(findRight(right), serviceRights);
    console.log('  updatedRights', updatedRights)

    this.setState({
      rights: updatedRights
    }, console.log('  >>> State update completed:', this.state));
  }

  updateService() {

  }

  deleteService(serviceName) {
    this.props.deleteService(serviceName);
  }

  render () {
    const name = this.props.name;
    // console.log('this.props', this.props)
    const rights = this.state.rights;

    return (
      <li className="service-card-item">
        <section className="card">
          <div className="icon-connectdevelop service-icon"></div>
          <div className="service-details">
            <h3>{ name }</h3>
            <section className="rights-section">
              <ul>
                {
                  rights.map(right =>
                    <li key={ right.name }>
                      {/* <input type="checkbox" checked={ !right.disabled } onChange={ () => this.onChangeHandler(name, right) }/> */}
                      <input type="checkbox" checked={ !right.disabled } onChange={ () => this.onChangeHandler(name, right) }/>
                      <p>{ right.name }</p>
                    </li>
                  )
                }
              </ul>
              <div className="icon-cancel-circled" title="Delete Service" onClick={ () => this.deleteService(name) }></div>
            </section>
          </div>
        </section>
      </li>
    )
  }
}
Leon Gaban
2017-08-04