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

javascript - componentDidMount not running after calling browserHistory.push("url") - Stack Overflow

programmeradmin2浏览0评论

I have an issue where I will send a user to a react-router route after login, based on the following:

        ...
        //check login
        browserHistory.push(self.props.destination_url);

I was expecting ponentDidMount to run, since this ponent had not been on the screen since I loaded the app, but it will not. If I click a link to it (react-router link) in the nav bar, ponentDidMount does run however.

I just need to make an API call when this ponent es on the screen because of a browserHistory.push(self.props.destination_url); route change. I've tried things like

<Router createElement={ (ponent, props) =>
{
  const { location } = props
  const key = `${location.pathname}${location.search}`
  props = { ...props, key }
  return React.createElement(ponent, props)
} }/>

here Component does not remount when route parameters change and it isn't working.

Here / shows "on mount", "on unmount", "on state change", or "on props changes". I'm not seeing any of those apply here. Is there a lifecycle method that will run after this browserHistory push transition?

I've been trying random lifecycle methods and ponentWillUpdate does run after browserHistory.push but it runs hundreds of times, pletely slowing the app down. I'd assume something I did inside it caused the nearly infinite loop:

ponentWillUpdate() {
    console.log('it ran ponentWillUpdate');
    if (this.props.email) {

        console.log('firing off /api/userInfo');
        let self = this;
        axios.post('/api/userInfo', {email: this.props.email})
          .then(function (response) {
              let result = response.data.result;
              console.log('after calling /api/userInfo');
              console.log(response);
              console.log(result);
              if (result) {
                  self.setState({restaurant_profile: result});
              }
          })
          .catch(function (error) {
              console.log("Something went wrong trying to check for a user's restaurant profile");
              console.log(error);
          });
    }
}

On the server/client you now see the POST run hundreds of times:

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

...

This will work for the student's demo, but not longterm. Looking for a lifecycle method that will only run once, and that changing state is safe and won't cause infinite loop

My r dependencies look like

"react": "^15.6.1",
"react-dom": "^15.6.1",
"react-redux": "^5.0.6",
"react-router": "^3.0.5",
"react-router-dom": "^4.2.2",
"react-transform-hmr": "^1.0.4",
"redux": "^3.7.2",

These routes are looking like

import React from "react";
import ReactDOM from "react-dom";
import { Provider } from "react-redux";
import { createStore, applyMiddleware } from "redux";
import { Router, Route, Link, IndexRoute, browserHistory } from "react-router";

import reducers from "./reducers";
import { loadConfig, getConfig } from "./config";
import Nav from "./Nav";
import LoginPage from "./containers/LoginPage";
import MapShowRestaurants from "./ponents/MapShowRestaurants";
import RestaurantRegistration from "./containers/RestaurantRegistration";


const createStoreWithMiddleware = applyMiddleware()(createStore);


getConfig.then((config) => {
    loadConfig(config);

    ReactDOM.render(
        (
            <Provider store={createStoreWithMiddleware(reducers)}>
                <Router history={browserHistory}>
                    <Route path="/" ponent={Nav}>
                        <IndexRoute ponent={MapShowRestaurants} />
                        <Route path="/login" ponent={LoginPage} />
                        <Route path="/registerRestaurant" ponent={RestaurantRegistration} />
                    </Route>
                </Router>
            </Provider>
        ), document.querySelector('.container'));
})
.catch((err) => {
    console.log(err);
})

I have an issue where I will send a user to a react-router route after login, based on the following:

        ...
        //check login
        browserHistory.push(self.props.destination_url);

I was expecting ponentDidMount to run, since this ponent had not been on the screen since I loaded the app, but it will not. If I click a link to it (react-router link) in the nav bar, ponentDidMount does run however.

I just need to make an API call when this ponent es on the screen because of a browserHistory.push(self.props.destination_url); route change. I've tried things like

<Router createElement={ (ponent, props) =>
{
  const { location } = props
  const key = `${location.pathname}${location.search}`
  props = { ...props, key }
  return React.createElement(ponent, props)
} }/>

here Component does not remount when route parameters change and it isn't working.

Here http://busypeoples.github.io/post/react-ponent-lifecycle/ shows "on mount", "on unmount", "on state change", or "on props changes". I'm not seeing any of those apply here. Is there a lifecycle method that will run after this browserHistory push transition?

I've been trying random lifecycle methods and ponentWillUpdate does run after browserHistory.push but it runs hundreds of times, pletely slowing the app down. I'd assume something I did inside it caused the nearly infinite loop:

ponentWillUpdate() {
    console.log('it ran ponentWillUpdate');
    if (this.props.email) {

        console.log('firing off /api/userInfo');
        let self = this;
        axios.post('/api/userInfo', {email: this.props.email})
          .then(function (response) {
              let result = response.data.result;
              console.log('after calling /api/userInfo');
              console.log(response);
              console.log(result);
              if (result) {
                  self.setState({restaurant_profile: result});
              }
          })
          .catch(function (error) {
              console.log("Something went wrong trying to check for a user's restaurant profile");
              console.log(error);
          });
    }
}

On the server/client you now see the POST run hundreds of times:

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

Executing (default): SELECT `id`, `email`, `password`, `RestaurantId` FROM `Users` AS `User` WHERE `User`.`email` = '[email protected]' LIMIT 1;

...

This will work for the student's demo, but not longterm. Looking for a lifecycle method that will only run once, and that changing state is safe and won't cause infinite loop

My r dependencies look like

"react": "^15.6.1",
"react-dom": "^15.6.1",
"react-redux": "^5.0.6",
"react-router": "^3.0.5",
"react-router-dom": "^4.2.2",
"react-transform-hmr": "^1.0.4",
"redux": "^3.7.2",

These routes are looking like

import React from "react";
import ReactDOM from "react-dom";
import { Provider } from "react-redux";
import { createStore, applyMiddleware } from "redux";
import { Router, Route, Link, IndexRoute, browserHistory } from "react-router";

import reducers from "./reducers";
import { loadConfig, getConfig } from "./config";
import Nav from "./Nav";
import LoginPage from "./containers/LoginPage";
import MapShowRestaurants from "./ponents/MapShowRestaurants";
import RestaurantRegistration from "./containers/RestaurantRegistration";


const createStoreWithMiddleware = applyMiddleware()(createStore);


getConfig.then((config) => {
    loadConfig(config);

    ReactDOM.render(
        (
            <Provider store={createStoreWithMiddleware(reducers)}>
                <Router history={browserHistory}>
                    <Route path="/" ponent={Nav}>
                        <IndexRoute ponent={MapShowRestaurants} />
                        <Route path="/login" ponent={LoginPage} />
                        <Route path="/registerRestaurant" ponent={RestaurantRegistration} />
                    </Route>
                </Router>
            </Provider>
        ), document.querySelector('.container'));
})
.catch((err) => {
    console.log(err);
})
Share Improve this question edited May 18, 2019 at 17:34 halfer 20.4k19 gold badges109 silver badges202 bronze badges asked Sep 28, 2017 at 1:57 codyc4321codyc4321 9,69224 gold badges100 silver badges173 bronze badges 10
  • I know that ponentDidMount only gets called once which is why it's not working as you expect (because (i believe) react-router does an initial render before it's actually called) I could be wrong - hence the ment and not an answer. I'll see if i can't find the docs for that. Also, as you've discovered, ponentWillUpdate() isn't the right place for that either... you might try creating your own method and then call it in the browserPush callback? Just a thought. – rimraf Commented Oct 1, 2017 at 1:31
  • how would I call it in browserHistory.push? From what I understand you just pass it the URL you want to go to and it pushes it into the history and reroutes you there – codyc4321 Commented Oct 1, 2017 at 1:51
  • I thought you could add a callback... when looking for that info i found this: github./ReactTraining/react-router/issues/3554 – rimraf Commented Oct 1, 2017 at 2:21
  • 1 @codyc4321 can you provide us two things? First and important, which version of react router are you using, and second, your defined routes. Thanks – rdarioduarte Commented Oct 2, 2017 at 1:22
  • 1 @codyc4321 are you sure that "ponentDidMount" is not getting called? I tried to replicate your scenario as best as I could in this sandbox: codesandbox.io/s/ov84yrzyy5 it works as I would expect it to, ponentDidMount is getting called every time I "navigate" to that ponent, it does not get called when I am already ON that ponent. – dubes Commented Oct 6, 2017 at 13:13
 |  Show 5 more ments

2 Answers 2

Reset to default 5 +50

don't put API calls in a ponent lifecycle.

create an action

function getUserInfo(payload) {
  // do axios
  return { 
      type: 'GET_USER_INFO',
      payload: result_of_your_api_call,
      error: false/true
  };
}

create reducer for this action and mutate state in reducer

after that, you need to matStateToProps and mapDispatchToProps

connect(mapStateToProps, mapDispatchToProps)(YourMagicComponent)

that connects your ponent and your redux

After that, you get a redux state in your ponent. For more information please read this http://redux.js/docs/basics/ExampleTodoList.html

Remember 1. when ponent initialized it hasn't any data in props except defaultProps. 2. ponentDidMount doesn't know anything outside ponent (available only default props and props defined before ponent mounted) 3. if you need to do something in ponentWillUpdate you need to define rules for update that's ponent

ponentWillUpdate(nextProps) {
  this.setState(key: nextProps.key)
}

shouldComponentUpdate(nextProps, nextState) {
  // fix for your update
  if (this.state.key !== nextProps.key) {
     return true;
  }
}

@codyc4321 try self.props.router.push('/url')

发布评论

评论列表(0)

  1. 暂无评论