Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Cannot read property 'history' of undefined (useHistory hook of React Router 5)

I am using the new useHistory hook of React Router, which came out a few weeks ago. My React-router version is 5.1.2. My React is at version 16.10.1. You can find my code at the bottom.

Yet when I import the new useHistory from react-router, I get this error:

Uncaught TypeError: Cannot read property 'history' of undefined

which is caused by this line in React-router

function useHistory() {   if (process.env.NODE_ENV !== "production") {     !(typeof useContext === "function") ? process.env.NODE_ENV !== "production" ? invariant(false, "You must use React >= 16.8 in order to use useHistory()") : invariant(false) : void 0;   }    return useContext(context).history; <---------------- ERROR IS ON THIS LINE !!!!!!!!!!!!!!!!! } 

Since it is related to useContext and perhaps a conflict with context is at fault, I tried completely removing all calls to useContext, creating the provider, etc. However, that did nothing. Tried with React v16.8; same thing. I have no idea what could be causing this, as every other feature of React router works fine.

***Note that the same thing happens when calling the other React router hooks, such as useLocation or useParams.

Has anyone else encountered this? Any ideas to what may cause this? Any help would be greatly appreciated, as I found nothing on the web related to this issue.

import React, {useEffect, useContext} from 'react'; import { BrowserRouter as Router, Route, Link } from "react-router-dom"; import { Switch, useHistory } from 'react-router' import { useTranslation } from 'react-i18next';  import lazyLoader from 'CommonApp/components/misc/lazyLoader'; import {AppContext} from 'CommonApp/context/context';  export default function App(props) {     const { i18n } = useTranslation();     const { language } = useContext(AppContext);     let history = useHistory();      useEffect(() => {         i18n.changeLanguage(language);     }, []);      return(         <Router>             <Route path="/">                 <div className={testClass}>HEADER</div>             </Route>         </Router>     ) }  
like image 911
Radu Sturzu Avatar asked Oct 03 '19 14:10

Radu Sturzu


People also ask

How do you use withRouter in react?

React Router has an higher-order component called withRouter with which we can pass in the React Router's history, location, and match objects to our React components as props. To use withRouter , you should wrap your App component inside withRouter() as a parameter.


1 Answers

It's because the react-router context isn't set in that component. Since its the <Router> component that sets the context you could use useHistory in a sub-component, but not in that one.

Here is a very basic strategy for solving this issue:

const AppWrapper = () => {   return (     <Router> // Set context       <App /> // Now App has access to context     </Router>   ) }  const App = () => {   let history = useHistory(); // Works! ... // Render routes in this component 

Then just be sure to use the "wrapper" component instead of App directly.

like image 103
Brian Thompson Avatar answered Sep 21 '22 03:09

Brian Thompson