Получение TypeError: Object (...) не является функцией при изменении маршрутов в React - PullRequest
0 голосов
/ 28 июня 2018

Не уверен, почему я получаю следующую ошибку.

Uncaught TypeError: Object(...) is not a function
  at Redirect.componentDidUpdate (Redirect.js:42)

Метод рендеринга моего компонента routeInterceptor:

render() {
  const { forbidden, notFound } = this.state;
  const { authed } = this.props;
  // const { location } = this.props;
  // const { pathname } = location;

  console.log('this.props', this.props);
  console.log('authed', authed);

  // If authentication is complete, but the user is not logged in,
  // redirect to the login view.

  /*
    Problem starts here, if I move the forbidden logic above this
    Everything works, however the user is then always redirected to the
    forbidden page instead of login
  */
  if (authed === false) return <Redirect to="/login" />;

  // If user is logged in and accesses an unathenticated view,
  // redirect to the Products view.
  if (authed === true) return <Products />;

  // if (forbidden && pathname !== '/login') return <Forbidden />;
  if (forbidden) return <Forbidden />;
  if (notFound) return <NotFound />;

  return <Loading />;
}

Там, где код разрывается внутри компонента Redirect:

Redirect.prototype.componentDidUpdate = function componentDidUpdate(prevProps) {
  var prevTo = createLocation(prevProps.to); // <- This line.
  var nextTo = createLocation(this.props.to);

  if (locationsAreEqual(prevTo, nextTo)) {
    warning(false, 'You tried to redirect to the same route you\'re currently on: ' + ('"' + nextTo.pathname + nextTo.search + '"'));
    return;
  }

  this.perform();
};

Вот реализация createLocation, являющаяся частью пакета history:

https://github.com/ReactTraining/history/blob/master/modules/LocationUtils.js

Это лог prevProps:

enter image description here

Есть идеи, что здесь может быть не так?

Вот весь код для LocationUtils.js, который является частью истории и содержит функцию createLocation.

import resolvePathname from "resolve-pathname";
import valueEqual from "value-equal";
import { parsePath } from "./PathUtils";

export const createLocation = (path, state, key, currentLocation) => {
  let location;
  if (typeof path === "string") {
    // Two-arg form: push(path, state)
    location = parsePath(path);
    location.state = state;
  } else {
    // One-arg form: push(location)
    location = { ...path };

    if (location.pathname === undefined) location.pathname = "";

    if (location.search) {
      if (location.search.charAt(0) !== "?")
        location.search = "?" + location.search;
    } else {
      location.search = "";
    }

    if (location.hash) {
      if (location.hash.charAt(0) !== "#") location.hash = "#" + location.hash;
    } else {
      location.hash = "";
    }

    if (state !== undefined && location.state === undefined)
      location.state = state;
  }

  try {
    location.pathname = decodeURI(location.pathname);
  } catch (e) {
    if (e instanceof URIError) {
      throw new URIError(
        'Pathname "' +
          location.pathname +
          '" could not be decoded. ' +
          "This is likely caused by an invalid percent-encoding."
      );
    } else {
      throw e;
    }
  }

  if (key) location.key = key;

  if (currentLocation) {
    // Resolve incomplete/relative pathname relative to current location.
    if (!location.pathname) {
      location.pathname = currentLocation.pathname;
    } else if (location.pathname.charAt(0) !== "/") {
      location.pathname = resolvePathname(
        location.pathname,
        currentLocation.pathname
      );
    }
  } else {
    // When there is no prior location and pathname is empty, set it to /
    if (!location.pathname) {
      location.pathname = "/";
    }
  }

  return location;
};

export const locationsAreEqual = (a, b) =>
  a.pathname === b.pathname &&
  a.search === b.search &&
  a.hash === b.hash &&
  a.key === b.key &&
  valueEqual(a.state, b.state);

Ответы [ 2 ]

0 голосов
/ 01 марта 2019

В моем случае это был Box компонент, который вызывал проблему.

import {unstable_Box as Box} from "@material-ui/core/Box";
...
function MyRoute(props){
...
<Box textAlign="center">Cat Falling</Box> {/*Had to remove this Box.*/}
...
}
0 голосов
/ 29 июня 2018

Разобрался, но это была БОЛЬ!

Это было условие гонки, потому что новые маршруты не были сгенерированы достаточно быстро, когда routeInterceptor изменил маршруты.

А в другом модуле routeManager нам нужно было добавить location в соединение только к watch it.

routeManager

import React from 'react';
import { connect } from 'react-redux';

// React router
import { Route, Switch } from 'react-router-dom';
import { ConnectedRouter } from 'react-router-redux';

// Actions
import { onAuthStateChange } from 'actions/Auth';

// MUI Components
import CssBaseline from '@material-ui/core/CssBaseline';

// Components
import Main from 'components/Common/main';
import Loading from 'components/Common/loading';
import * as components from 'components';

// Utils
import history from 'clientHistory';
import { cleanMapStateToProps } from 'utils/redux';
import { getRoutesArray } from 'utils/routes';

// Copy
import { PAGE_AUTH_REQS } from 'copy/Global/routes';

const {
  AUTHORIZED,
  UNAUTHORIZED
} = PAGE_AUTH_REQS;

const getComponentForRoute = compName => (
  components[compName] || (() => null)
);

const getRouteComponents = FILTER => getRoutesArray(FILTER)
  .map(route => ({
    ...route,
    component: getComponentForRoute(route.component)
  }))
  .map(route => (<Route {...route} key={route.label} />));

class RouteManager extends React.Component {
  componentDidMount() {
    this.props.onAuthStateChange();
  }

  renderAuthorizedRoutes() {
    const routes = getRouteComponents(AUTHORIZED);

    return (
      <ConnectedRouter history={history}>
        <Main
          signOut={this.signOut}
          notifications={this.props.notifications}
          currentNotification={this.props.currentNotification}
        >
          <CssBaseline />
          <Switch>
            {routes}
          </Switch>
        </Main>
      </ConnectedRouter>
    );
  }

  renderUnauthorizedRoutes() {
    const routes = getRouteComponents(UNAUTHORIZED);

    return (
      <ConnectedRouter history={history}>
        <Main
          signOut={this.signOut}
          notifications={this.props.notifications}
          currentNotification={this.props.currentNotification}
        >
          <CssBaseline />
          <Switch>
            {routes}
          </Switch>
        </Main>
      </ConnectedRouter>
    );
  }

  render() {
    const { authed } = this.props;

    if (authed === null) {
      return <Loading />;
    }

    return authed
      ? this.renderAuthorizedRoutes()
      : this.renderUnauthorizedRoutes();
  }
}

export const RouteManagerJest = RouteManager;

const mapDispatchToProps = dispatch => ({
  onAuthStateChange: () => { dispatch(onAuthStateChange()); }
});

export default connect(cleanMapStateToProps([
  'authed',
  'location', // <-- Needed to just add location here.
  'currentNotification',
  'notifications'
]), mapDispatchToProps)(RouteManager);

export default connect(cleanMapStateToProps([
  'authed',
  'location', // <-- Needed to just add location here.
  'currentNotification',
  'notifications'
]), mapDispatchToProps)(RouteManager);

ways.js

import { PAGE_AUTH_REQS } from 'copy/Global/routes';

export const PROP_CONTENT_ROUTE = '[[ContentRoute]]';

const exact = true;

const {
  ANY,
  AUTHORIZED,
  UNAUTHORIZED
} = PAGE_AUTH_REQS;

/**
 * Do not export routes - if you need to get a list of routes (as an array or
 * object), use one of the convenience methods down below.
 */
const routesConfig = {
  // Auth Routes => /:context
  authLogin: {
    label: 'Login',
    path: '/login',
    title: 'Login',
    authReq: UNAUTHORIZED,
    component: 'Login',
    exact
  },
  authResetPassword: {
    label: 'Reset',
    path: '/reset-password',
    title: 'Reset Password',
    authReq: UNAUTHORIZED,
    component: 'ResetPassword',
    exact
  },
  authRedirector: {
    label: 'Redirect',
    path: '/redirect',
    title: 'Firebase Redirect',
    authReq: UNAUTHORIZED,
    component: 'FirebaseRedirector',
    exact
  },
  authChangePassword: {
    label: 'Change',
    path: '/change-password/:oobCode/',
    title: 'Change Password',
    authReq: UNAUTHORIZED,
    component: 'ChangePassword',
    exact
  },
  authVerification: {
    label: 'Verify',
    path: '/verification',
    title: 'Verify your email',
    authReq: UNAUTHORIZED,
    component: 'Login',
    exact
  },
  authRestricted: {
    label: 'Restricted Access',
    path: '/restricted-access',
    title: 'Restricted Access',
    authReq: UNAUTHORIZED,
    component: 'RestrictedAccess',
    exact
  },

  products: {
    label: 'Products',
    path: '/(products)?',
    title: 'Products',
    authReq: AUTHORIZED,
    component: 'Products'
  },

  // ********************************************************
  // ********************************************************
  // ********************************************************
  // ANY ROUTES BELOW RouteInterceptor WILL NOT BE CONSIDERED
  // ********************************************************
  // ********************************************************
  // ********************************************************

  routeInterceptor: {
    label: null,
    title: null,
    authReq: ANY,
    component: 'RouteInterceptor'
  }
};

export default routesConfig;

И теперь, наконец, все работает как положено.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...