javascript - react onClick 方法

标签 javascript reactjs accordion

我正在尝试使用 React 制作一个 Accordion 应用程序,我有来自 API 的数据,并且我有该应用程序的基本轮廓,但我不确定如何处理 Accordion 上的点击。

这是我的代码:

这是我设置数据的地方

App.js

class App extends Component {
  constructor(){
    super();
    this.state = {
      myData: {}
    };
  }

  componentDidMount() {
    axios.get(linkToApi)
      .then(responseData => {
        this.setState({ mydata: responseData.data });
      })
      .catch(error => {
        console.log("Porblem getting data", error);
      });
  }

  render() {
    return (
      <div className="App">
        <Accordion data={this.state.myData} />
      </div>
    );
  }
}

export default App;

Accordion .js

const Accordion = props => {
   let accordionElements = Object.keys(props.data).map(function(keyName, keyIndex) {
     return <AccordionElement
            {...props.data[keyName]}
            key={props.data[keyName].id}
          />;
 })


return (
   <ul className="accordion">
     {accordionElements}
   </ul>
 );
 }
 export default Accordion;

AccordionElement.js

const AccordionElement = props => {

 const handleOnClick = (e) => {
   e.preventDefault();
   //
 }

 return (
   <li style={listItemStyle} onClick={handleOnClick}>
     <h1 data={props}>{ props.name }</h1>
     <ul style={descriptionStyle}>
       <li>Description: { props.description }</li>
     </ul>
   </li>
 );

}

export default AccordionElement;

我希望能够在单击标题时显示或隐藏其下方的说明。我不太确定我会如何解决这个问题,有什么想法吗?

最佳答案

我会使用状态来管理描述在给定时间是否可见。所以像这样:

class AccordionElement extends React.Component {

  constructor(props) {
    super(props);
    this.state = {
      open: false
    }
  }

  const handleOnClick = (e) => {
    e.preventDefault();
    this.setState({ open: !this.state.open });
  }

  renderDescription() {
    const description = (
      <ul style={descriptionStyle}>
        <li>Description: { props.description }</li>
      </ul>
    );
    return this.state.open ? description : null;
}

  return (
    <li style={listItemStyle} onClick={handleOnClick}>
      <h1 data={props}>{ props.name }</h1>
      {this.renderDescription()} 
    </li>
  );

}

export default AccordionElement;

编辑:我已经解决了一些问题;现在应该可以按预期工作,将 AccordionAccordionElement 保留为功能组件。代码如下:

class App extends React.Component {
  constructor(){
    super();
    this.state = {
      myData: []
    };
    this.setOpenStatus = this.setOpenStatus.bind(this);
  }

  componentDidMount() {
    // this would be inside the API call
        const myData = [
        {
        id: 1,
        name: 'Item 1',
        description: 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Ut varius metus tellus, eu tincidunt est viverra vitae. Fusce et mollis libero.'
      }, {
        id: 2,
        name: 'Item 2',
        description: 'Ut interdum ut justo ac euismod. Phasellus vitae pellentesque lectus, et cursus erat. Suspendisse eget risus gravida tellus rutrum gravida et vitae felis.'
      }, {
        id: 3,
        name: 'Item 2',
        description: ' Cras euismod massa eu mi consequat mollis. Sed aliquam tellus sed sem dictum feugiat. Nullam pretium purus sed ipsum pharetra luctus.'
      }
    ];
    myData.forEach(item => {
        item.open = false;
    });
    this.setState({ myData });
  }

  setOpenStatus(id) {
    console.log(this.state);
    const myData = this.state.myData;
    myData.forEach(item => {
        if (item.id == id) {
        item.open = !item.open;
      } else {
        item.open = false;
      }
    });
    this.setState({ myData });
  }

  render() {
    return (
      <div className="App">
        <Accordion data={this.state.myData} setOpenStatus={this.setOpenStatus} />
      </div>
    );
  }
}

const Accordion = props => {
   let accordionElements = [];
   const {data} = props;
   if (data && data.length > 0) {
     data.forEach(item => {
       console.log('props', item);
       accordionElements.push(<AccordionElement
              {...item}
              setOpenStatus={props.setOpenStatus}
              key={item.id}
            />);
     });
   }


return (
   <ul className="accordion">
     {accordionElements}
   </ul>
 );
 }

const AccordionElement = props => {

 const handleOnClick = (e) => {
   e.preventDefault();
     props.setOpenStatus(props.id);
 }

 const renderDescription = () => {
    return props.open ? props.description : null;
 }

 return (
   <li onClick={handleOnClick}>
     <h1>{ props.name }</h1>
     <span>{ renderDescription() }</span>
   </li>
 );

}

setOpenStatus(id) 函数作为 props 传递给 AccordionElement,然后在点击处理程序上调用。这会在 myData 状态数组中找到 ID 并切换 open bool 值,然后将其他 open bool 值更改为 false,因此只有单击的那个可以为 true。 jsfiddle链接。

关于javascript - react onClick 方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46958402/

相关文章:

javascript - 单击填充图像映射并设置变量

javascript - Electron/Node -使用require和readFile的代码片段使javascript失败

javascript - 在 react native 中对齐导航屏幕内容

jquery - Accordion 效果重叠问题

javascript - 如何使用 'toHaveClass' 来匹配 Jest 中的子字符串?

javascript - 如何使用 JSON 在 AJAX 中发送数组

javascript - 为什么异步函数中的连续 setState 调用不进行批处理?

javascript - React createRef() 与回调引用。使用一个比另一个有什么优势吗?

javascript - 当您单击图像时,使文本出现(向上滑动)在图像下方

css - Zurb 基础 Accordion CSS 断点