如何在 React Router v4 中嵌套路由?

IT技术 reactjs react-router react-router-v4
2021-04-20 06:33:45

有没有办法在 React Router v4 中嵌套路由?

这有效:

  <Router basename='/app'>
    <main>
      <Route path='/' component={AppBar} />
      <Route path='/customers' component={Customers} />
    </main>
  </Router>

这不会:

  <Router basename='/app'>
    <Route path='/' component={AppBar}>
      <Route path='/customers' component={Customers} />
    </Route>
  </Router>

客户组成:

import React, { Component, PropTypes } from 'react'
import styled from 'styled-components'

export default class Customers extends Component {
  render () {
    return (
      <Container>
        <h1>Customers</h1>
      </Container>
    )
  }
}

const Container = styled.section`
  height: 100%;
  padding: 15px;
  overflow: auto;
`
6个回答

迄今为止我发现的最佳模式。

// 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>

我可以继续将它嵌套在一个组件中,一切都很好,包括 hmr(如果使用 webpack,不要忘记设置output.publicPath"/"

// dashboard component
<div>
    // the same way as before, not setting a path prop
    // makes it render on every /dashboard/** request 
    <Route component={DashboardTAB}/>
    <Switch>
        // longer path (with same root) than others first
        <Route path="/dashboard/graphs/longerpath" component={GraphForm}/>
        <Route path="/dashboard/graphs" component={Graphs}/>
        <Route path="/dashboard/workers" component={List}/>
        <Route path="/dashboard/insert" component={InsertComponent}/>
    </Switch>
</div>
完美,没关系,我刚刚开始了第一条最短路径的路线,因此将最短路径作为唯一可用的路径。
2021-06-06 06:33:45
我确实喜欢这样,但是使用主父布局,它可以正确运行而没有错误,但仅在 <switch> 中显示第一条路由(在组件中),并且当我单击链接时没有页面更改,我也使用 webpack 并放置输出。 publicPath 正如你所说的......对另一个问题有什么想法吗?谢谢
2021-06-12 06:33:45
不设置路径道具解决了我遇到的问题。谢谢!
2021-06-21 06:33:45

我从文档中改编了这个,到目前为止似乎有效。可能遗漏了一些明显的东西,是的,这不是 v4 方式,但我们需要在一个地方定义所有路由。

function RouteNest(props){ return (
  <Route exact={props.exact} path={props.path} render={ p => <props.component {...p} children={props.children}/> } />
)}

export const MainRoutes = props => 

<div className='content layout'>

  <Route exact path="/" component={Landing}/>
  <Route  path={'/contact'} component={Contact}/>

  <RouteNest  path={'/thing'} component={CompoWithSub}>
    <RouteNest  path={'/thing/suba'} component={SubComponentA}/>
    <RouteNest  path={'/thing/subb'} component={SubComponentB}/>
  </RouteNest>


</div>

export const CompoWithSub = props => <div>{props.children)</div>

您是 AppBar 组件,负责呈现客户。对于要调用的客户,您必须呈现 AppBar 的子项。任何直接嵌套在 AppBar 下的东西都是 AppBar 的子项。

import React from 'react';

const AppBar = ({ children }) => (
  <div>
    <header>
       <h1> stuff </h1>
    </header>
    {children}
  </div>
);

export default AppBar

请注意,当您访问“/”时,只有 AppBar 会呈现。AppBar 和客户会在您访问“/customers”时呈现。

文档中对嵌套路由的唯一参考是:reacttraining.com/react-router/#match但我只是不明白如何使用它来构建嵌套路由。
2021-05-26 06:33:45
是的。/app/customers. 未嵌套时,<Customers />组件会在该 URL 处呈现。
2021-05-29 06:33:45
好狗屎。我以为我在 v4 上,但我在 v3 上。对于那个很抱歉。不确定两个版本之间有什么变化。
2021-05-29 06:33:45
即使使用了<AppBar />'s { children }<Customers />当嵌套在 React Router v4 中时组件也不会渲染。
2021-06-03 06:33:45
你在网址“/customers”?
2021-06-14 06:33:45

如果有人想要嵌套路由而不输入包装路由的前缀,我在 TSX 中创建了这样的东西:

进口:

import * as React from 'react';
import { Route, RouteComponentProps, RouteProps, Switch } from 'react-router-dom';
import Index from 'views/index';
import Login from 'views/login';
import NoMatch from 'views/no-match';

接口:

interface INestedRoutes {
    nested?: string;
}

interface INestedRoute extends RouteProps, INestedRoutes {}

NestedRoute 和 NestedRoutes 包装器:

class NestedRoutes extends React.Component<INestedRoutes> {
    public render() {
        const childrenWithProps = React.Children.map(this.props.children, (child) => {
            return React.cloneElement(
                child as React.ReactElement<any>, { nested: this.props.nested },
            );
        })
        return childrenWithProps;
    }
}


const NestedRoute: React.SFC<INestedRoute> = (props: INestedRoute) => {
    return <Route path={`${props.nested}${props.path}`} component={props.component} />;
};

和带有包装器的路由:

const MultiLanguage: React.SFC<RouteComponentProps<any>> = (props: RouteComponentProps<any>) => {
    return (
        <NestedRoutes nested={props.match.path} >
            <NestedRoute path="/test" component={Login} />
            <NestedRoute path="/no-match" component={NoMatch} />
        </NestedRoutes>
    );
};


export default (
    <Switch>
        <Route path="/:language" component={MultiLanguage}/>
        <Route exact={true} path="/" component={Index} />
        <Route path="/login" component={Login} />
        <Route component={NoMatch} />
    </Switch>
);

对于嵌套路由,我使用了一种非常简单的方法。

示例主路由器就是这样

<Router history={history}>
  <Switch >
    <Route path="/" component={Home}></Route>
  </Switch>
</Router>

使用嵌套路由的 Home 组件内部如下:

<div className="App">
  <Navbar title="Home" links = { NavbarLinks }/>
  {this.renderContentPage()}
</div>

renderContentPage 将检查 URL 并呈现嵌套路由。

<Route exact path="/" component={Page1}></Route>
<Route exact path="/page1" component={Page1}></Route>
<Route exact path='/page2' component={Page2} />

因此在 Home 组件内部呈现了 page1 和 page2 组件。