ReactJs:为具有相同组件的选项卡维护单独的状态

标签 reactjs state

我在组件中使用选项卡结构,两个选项卡内部都有相同的组件。

如果单击“添加”按钮,则会将新组件添加到选项卡,现在如果我切换选项卡并转到选项卡 2,其中也包含相同的组件。

我尝试在下面的 fiddle 中复制场景。

https://jsfiddle.net/rzv6Lrjh/89/

var App = React.createClass({
  render: function () {
    return (
      <div>
        <Tabs selected={0}>
          <Pane label="Tab 1">
            <Tickets/>
          </Pane>
          <Pane label="Tab 2">
            <Tickets/>
          </Pane>
        </Tabs>
      </div>
    );
  }
});

最佳答案

好吧,你只需要让 React 了解你的 2 个 Ticket 组件是不同的。

您可以通过向该组件添加虚拟属性来简单地实现此目的

<Tickets key="1/>

第一个选项卡和

<Tickets key="2/>

第二个。然后你将得到如你所期望的 2 次渲染;)


将门票作为 Prop 传递的完整答案

var Tabs = React.createClass({
    displayName: 'Tabs',
    propTypes: {
    selected: React.PropTypes.number,
    children: React.PropTypes.oneOfType([
      React.PropTypes.array,
      React.PropTypes.element
    ]).isRequired
  },
  getDefaultProps: function () {
    return {
        selected: 0
    };
  },
  getInitialState: function () {
    return {
        selected: this.props.selected
    };
  },
  shouldComponentUpdate(nextProps, nextState) {
    return this.props !== nextProps || this.state !== nextState;
  },
  handleClick: function (index, event) {
    event.preventDefault();
    this.setState({
        selected: index
    });
  },
  _renderTitles: function () {
    function labels(child, index) {
        var activeClass = (this.state.selected === index ? 'active' : '');
        return (
        <li key={index}>
            <a href="#" 
            className={activeClass}
            onClick={this.handleClick.bind(this, index)}>
            {child.props.label}
          </a>
        </li>
      );
    }
    return (
        <ul className="tabs__labels">
        {this.props.children.map(labels.bind(this))}
      </ul>
    );
  },
  _renderContent: function () {
    return (
        <div className="tabs__content">
            {this.props.children[this.state.selected]}
      </div>
    );
  },
    render: function () {
    return (
        <div className="tabs">
        {this._renderTitles()}
        {this._renderContent()}
      </div>
    );
  }
});

var Pane = React.createClass({
    displayName: 'Pane',
  propTypes: {
    label: React.PropTypes.string.isRequired,
    children: React.PropTypes.element.isRequired
  },
    render: function () {
    return (
        <div>
        {this.props.children}
      </div>
    );
  }
});

var CreateTicket = React.createClass({
  getInitialState: function() {
    return{};
  },
  render: function() {
    return(
      <button type="button" onClick={this.props.createTicket} className="add-another-ticket">
        +Add Ticket
      </button>
    );
  }
});

var Tickets = React.createClass({
  render: function() {
    return (
      <div>
        {this.props.tickets}
        <CreateTicket createTicket={this.props.onCreateTicket} />
      </div>
    );
  }
});

var IndividualTicketInput = React.createClass({
  getInitialState: function() {
    return { ticket: {name: '', quantity: '', price: null} };
  },
  render: function() {
    return (
      <ul>
        <li>
          <label>Ticket Name</label>
          <input className="ticket-name" type="text" placeholder="E.g. General Admission" value={this.state.ticket.name} />
        </li>
        <li>
          <label>Quantity Available</label>
          <input className="quantity" type="number" placeholder="100" value={this.state.ticket.quantity} />
        </li>
          <li>
            <label>Price</label>
            <input className="price" type="number" placeholder="25.00" value={this.state.ticket.price} />
          </li>
        <li>
          <button type="button" className="delete-ticket" onClick={this.deleteTicket}><i className="fa fa-trash-o delete-ticket"></i></button>
        </li>
      </ul>
    );
  }
});

var App = React.createClass({
    getInitialState: function() {
        return {
            tickets1: [(<IndividualTicketInput/>)],
            tickets2: [(<IndividualTicketInput/>)]
        }
    },
    onCreateTicket1: function() {
        this.state.tickets1.push(<IndividualTicketInput/>);
    },
    onCreateTicket2: function() {
        this.state.tickets2.push(<IndividualTicketInput/>);
    },
    render: function() {
        return (
        <div>
            <Tabs selected={0}>
                <Pane label="Tab 1">
                    <Tickets tickets={this.state.tickets1} onCreateTicket={this.onCreateTicket1.bind(this)}/>
                </Pane>
                <Pane label="Tab 2">
                    <Tickets tickets={this.state.tickets2} onCreateTicket={this.onCreateTicket2.bind(this)}/>
                </Pane>
            </Tabs>
        </div>
        );
    }
});

ReactDOM.render(<App />, document.querySelector('.container'));

关于ReactJs:为具有相同组件的选项卡维护单独的状态,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44513204/

相关文章:

reactjs - 在 React 中,将获取的 Firebase 数据存储在本地或全局状态中?

reactjs - 当 React Hooks 中的 redux 状态发生变化时, Prop 不会更新

javascript - Reactjs 按键和值过滤状态

javascript - 从withHandlers中的withReducer中检索重构本地调度函数

python - 使用 Flask 为使用 create-react-app 创建的前端提供服务

reactjs - 保存到数据库后 React Redux 存储状态更新过程

javascript - 如何在 React JS 中检查损坏的图像

node.js - 自动刷新 firebase 身份验证而不重新加载窗口?

reactjs - 创建 react "development"构建

javascript - 如何将项目正确添加到 reactjs 中的状态(空数组)?