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

javascript - Reactjs: How to fetch data to loaded before the component is mounted? - Stack Overflow

programmeradmin0浏览0评论

Something weird is happening, I've been reading the React docs and they talk about the lifecycle and how you can do somestuff before your ponent is rendered. I am trying, but everything I try is failing, always the ponent makes the render first and after calls ponenWillMount, ..didMount, etc.. and after the call of those functions, the render happens again.

I need to load the data first in order to fill the state because I don't want initial state to be null, I want it with data since the initial rendering.

I am using Flux and Alt, here is the

action

@createActions(flux)
class GetDealersActions {

  constructor () {
    this.generateActions('dealerDataSuccess', 'dealerDataFail');
  }

  getDealers (data) {
    const that = this;
    that.dispatch();
    axios.get(`${API_ENDPOINT}/get-dealers/get-dealers`)
      .then(function success (response) {
        console.log('success GetDealersActions');
        that.actions.dealerDataSuccess({...response.data});
      })
  }
}

then the store

@createStore(flux)
class GetDealersStore {

  constructor () {
    this.state = {
      dealerData : null,
    };
  }

  @bind(GetDealersActions.dealerDataSuccess)
  dealerDataSuccess (data) {
    this.setState({
      dealerData : data,
    });
  }

}

and the ponent

@connectToStores
export default class Dealers extends Component {

  static propTypes = {
    title : React.PropTypes.func,
  }

  static contextTypes = {
    router : React.PropTypes.func,
  }

  constructor (props) {
    super(props);
    this.state = {
      modal : false,
      dealerData : this.props.dealerData,
    }
  }

  ponentWillMount () {
    GetDealersActions.getDealers();
    this.setState({
      dealerData : this.props.dealerData.dealersData,
    })
  }

  static getStores () {
    return [ GetDealersStore ];
  }

  static getPropsFromStores () {
    return {
      ...GetDealersStore.getState(),
    }
  }

  render () {

    return (<div>
          <div style={Styles.mainCont}>
            {!!this.props.dealerData ?
              this.props.dealerData.dealersData.map((dealer) => {
                return (<div>HERE I AM RENDERING WHAT I NEED</div>);
              }) : <p>Loading . . .</p>
            }
          </div>
      </div>
    );
  }

}

as you can see in the ponent part I have this

  constructor (props) {
    super(props);
    this.state = {
      modal : false,
      dealerData : this.props.dealerData,
    }
  }

  ponentWillMount () {
    GetDealersActions.getDealers();
    this.setState({
      dealerData : this.props.dealerData.dealersData,
    })
  }

which is telling me that dealerData is undefined or can not read property of null.

All I need is to know a technique where I can fetch the data before the initial renders occurs. So I can filled out the state and the start working with that data.

Something weird is happening, I've been reading the React docs and they talk about the lifecycle and how you can do somestuff before your ponent is rendered. I am trying, but everything I try is failing, always the ponent makes the render first and after calls ponenWillMount, ..didMount, etc.. and after the call of those functions, the render happens again.

I need to load the data first in order to fill the state because I don't want initial state to be null, I want it with data since the initial rendering.

I am using Flux and Alt, here is the

action

@createActions(flux)
class GetDealersActions {

  constructor () {
    this.generateActions('dealerDataSuccess', 'dealerDataFail');
  }

  getDealers (data) {
    const that = this;
    that.dispatch();
    axios.get(`${API_ENDPOINT}/get-dealers/get-dealers`)
      .then(function success (response) {
        console.log('success GetDealersActions');
        that.actions.dealerDataSuccess({...response.data});
      })
  }
}

then the store

@createStore(flux)
class GetDealersStore {

  constructor () {
    this.state = {
      dealerData : null,
    };
  }

  @bind(GetDealersActions.dealerDataSuccess)
  dealerDataSuccess (data) {
    this.setState({
      dealerData : data,
    });
  }

}

and the ponent

@connectToStores
export default class Dealers extends Component {

  static propTypes = {
    title : React.PropTypes.func,
  }

  static contextTypes = {
    router : React.PropTypes.func,
  }

  constructor (props) {
    super(props);
    this.state = {
      modal : false,
      dealerData : this.props.dealerData,
    }
  }

  ponentWillMount () {
    GetDealersActions.getDealers();
    this.setState({
      dealerData : this.props.dealerData.dealersData,
    })
  }

  static getStores () {
    return [ GetDealersStore ];
  }

  static getPropsFromStores () {
    return {
      ...GetDealersStore.getState(),
    }
  }

  render () {

    return (<div>
          <div style={Styles.mainCont}>
            {!!this.props.dealerData ?
              this.props.dealerData.dealersData.map((dealer) => {
                return (<div>HERE I AM RENDERING WHAT I NEED</div>);
              }) : <p>Loading . . .</p>
            }
          </div>
      </div>
    );
  }

}

as you can see in the ponent part I have this

  constructor (props) {
    super(props);
    this.state = {
      modal : false,
      dealerData : this.props.dealerData,
    }
  }

  ponentWillMount () {
    GetDealersActions.getDealers();
    this.setState({
      dealerData : this.props.dealerData.dealersData,
    })
  }

which is telling me that dealerData is undefined or can not read property of null.

All I need is to know a technique where I can fetch the data before the initial renders occurs. So I can filled out the state and the start working with that data.

Share Improve this question edited Oct 12, 2015 at 22:26 Felix Kling 818k181 gold badges1.1k silver badges1.2k bronze badges asked Oct 12, 2015 at 22:24 NonNon 8,61920 gold badges80 silver badges130 bronze badges 4
  • A related question: stackoverflow./q/30329418/1233251 – E_net4 Commented Oct 12, 2015 at 22:30
  • @E_net4 there is my issue: Invoked once, both on the client and server, immediately before the initial rendering occurs. If you call setState within this method, render() will see the updated state and will be executed only once despite the state change. that got me confused. – Non Commented Oct 12, 2015 at 22:32
  • @E_net4 in my code I already add an conditional statement and everything as in that question and not working. See my code, try to help me, please. – Non Commented Oct 12, 2015 at 22:34
  • You probably have a type error that prevents the program from running properly. – E_net4 Commented Oct 12, 2015 at 22:42
Add a ment  | 

2 Answers 2

Reset to default 4

React does guarantee that state assignments in ponentWillMount will take place before the first render. As you well stated in the ments:

Invoked once, both on the client and server, immediately before the initial rendering occurs. If you call setState within this method, render() will see the updated state and will be executed only once despite the state change.

However, the asynchronous actions requested there will not immediately update your store. Calling GetDealersActions.getDealers(); will issue that the store is updated with new content, but the response will only arrive later in the event queue. This means that this.props.dealersData does not change during the function and setState will attempt to read property "dealersData" of an undefined property. Regardless, the requested content cannot be visible at the first render.

My advice is the same as the one in a similar question. Preventing the ponent from rendering that content until it bees available, as you did, is an appropriate thing to do in a program. Alternatively, render a loader while your "dealersData" hasn't arrived.


For solving your particular problem, remove that setState from ponentWillMount. All should work well if your parent ponent is properly listening for changes and propagating them to the children's props.

ponentWillMount () {
  GetDealersActions.getDealers();
}

The best answer I use to receive data from server and display it

constructor(props){
        super(props);
        this.state = {
            items2 : [{}],
            isLoading: true
        }

    }

ponentWillMount (){
 axios({
            method: 'get',
            responseType: 'json',
            url: '....',

        })
            .then(response => {
                self.setState({
                    items2: response ,
                    isLoading: false
                });
                console.log("Asmaa Almadhoun *** : " + self.state.items2);
            })
            .catch(error => {
                console.log("Error *** : " + error);
            });
    })}

 render() {
   return(
       { this.state.isLoading &&
                    <i className="fa fa-spinner fa-spin"></i>

                }
                { !this.state.isLoading &&
            //external ponent passing Server data to its classes
                     <TestDynamic  items={this.state.items2}/> 
                }
         ) }
发布评论

评论列表(0)

  1. 暂无评论