Nested Routes in React Router v4

React RouterReact Router-V4

React Router Problem Overview


I'm trying to set up some nested routes to add a common layout. Check the code out:

  <Router>
    <Route component={Layout}>
      <div>
        <Route path='/abc' component={ABC} />
        <Route path='/xyz' component={XYZ} />
      </div>
    </Route>
  </Router>

While this works perfectly fine, I still get the warning:

> Warning: You should not use <Route component> and <Route children> in the same > route; will be ignored

React Router Solutions


Solution 1 - React Router

CESCO's answer renders first the component AppShell then one of the components inside Switch. But these components are NOT going to render inside AppShell, they will NOT be children of AppShell.

In v4 to wrap components you don't put anymore your Routes inside another Route, you put your Routes directly inside a component.
I.E : for the wrapper instead of <Route component={Layout}> you directly use <Layout>.

Full code :

  <Router>
    <Layout>
      <Route path='/abc' component={ABC} />
      <Route path='/xyz' component={XYZ} />
    </Layout>
  </Router>

The change is probably explained by the idea to make React Router v4 to be pure React so you only use React elements like with any other React element.

EDIT : I removed the Switch component as it's not useful here. See when it's useful here.

Solution 2 - React Router

You need to use the switch component to nesting to work nice. Also, see this question

// main app
<div>
    // not setting a path prop, makes this always render
    <Route component={AppShell}/>
    <Switch>
        <Route exact path="/" component={Login}/>
        <Route path="/dashboard" component={AsyncDashboard(userAgent)}/>
        <Route component={NoMatch}/>
    </Switch>
</div>

And version-4 components do not take children, instead, you should use the render prop.

<Router>
    <Route render={(props)=>{
      return <div>Whatever</div>}>
    </Route>
  </Router>

Solution 3 - React Router

Try:

<Router>
    <Layout>
        <Route path='/abc' component={ABC} />
        <Route path='/xyz' component={XYZ} />
    </Layout>
</Router>

Solution 4 - React Router

If you do not want Layout to run at loaded. Use this method:

<div className="container">
	<Route path="/main" component={ChatList}/>
	<Switch>
		<Route exact path="/" component={Start} />
		<Route path="/main/single" component={SingleChat} />
		<Route path="/main/group" component={GroupChat} />
		<Route path="/login" component={Login} />
	</Switch>
</div>

Whenever history changes, componentWillReceiveProps in the ChatList will run.

Solution 5 - React Router

You can also try this :

<Route exact path="/Home"
                 render={props=>(
                                 <div>
                                      <Layout/>
                                      <Archive/>
                                </div>
                       )} 
    />
  

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionSean GillespieView Question on Stackoverflow
Solution 1 - React RouterIlan SchemoulView Answer on Stackoverflow
Solution 2 - React RouterCESCOView Answer on Stackoverflow
Solution 3 - React RoutermarrekkView Answer on Stackoverflow
Solution 4 - React RouterAvare KodcuView Answer on Stackoverflow
Solution 5 - React RouterMohamed YahyaView Answer on Stackoverflow