其实这个话题在网上找已经能找到很多解决方案了,之所以写这篇文章主要也是让自己对路由拦截加深点印象吧(毕竟自己没实现过路由拦截。。。)。用vue时已经有导航守卫,在公司里做项目路由拦截的部分也早已经写好,那就只能自己做demo练习一下了。react-router本身没提供导航守卫,按作者的说法是可以自己来实现,那就自己来实现一下吧。
我在上家公司用的是react-router4,现在已经有react-router6可以使用了,我们先使用react-router4写一遍再来使用react-router6,会使用一个再来使用另一个也不难了,现在先定义最简单的需求:
- 定义三个页面,登录页,首页,关于我们页面
- 登录页和关于我们是免登录页面。要进入首页必须状态认证通过,否则会跳转到登录页
react-router4:
xml
<HashRouter>
<Switch>
<Route exact path="/" component={Home} />
<Route exact path="/aboutUs" component={AboutUs} />
<Route exact path="/login" component={Login} />
</Switch>
</HashRouter>
定义好路由后,通过修改地址栏,就能匹配路径渲染相应的组件。所以要做路由拦截,我们可以选择不要直接渲染Route组件,可以自定义AuthRoute组件,组件最终会返回Route组件,在返回Route组件之前我们就可以判断状态不通过时跳转登录页的操作,从而实现路由拦截。
方法一:自定义权限组件
- 自定义AuthRoute组件,AuthRoute组件会返回Route组件
- AuthRoute返回Route之前进行状态认证,不通过会跳转到登录页
javascript
<HashRouter>
<Switch>
<AuthRoute exact path="/" component={Home} />
<Route exact path="/aboutUs" component={AboutUs} />
<Route exact path="/login" component={Login} />
</Switch>
</HashRouter>
const AuthRoute = (props) => {
const auth = null
if (!auth) {
alert('跳转到登录页')
return <Redirect to="/login" />
}
return (
<Route {...props} />
)
}
效果:
方法二:使用Route自带的render属性
xml
<HashRouter>
<Switch>
<Route exact path="/" render={() => auth ? <Home /> : <Redirect to="/login" /> } />
<Route exact path="/aboutUs" component={AboutUs} />
<Route exact path="/login" component={Login} />
</Switch>
</HashRouter>
这种方法也可以通过判断状态来渲染不同的组件,认证不通过则跳转到登录页。但是需要路由拦截的页面可能会有很多个,而且需要认证的状态也可能不止一个,所以最好还是封装一个自定义组件来实现。
react-router6:
我们先试下同样使用自定义组件来进行路由拦截。
xml
<HashRouter>
<Routes>
<AuthRoute exact path="/" element={<Home />} />
<Route exact path="/login" element={<Login />}></Route>
<Route exact path="/aboutUs" element={<AboutUs />}></Route>
</Routes>
</HashRouter>
效果:
结果报了错,在Routes组件里只能是Route组件不能是自定义组件,所以稍微改造一下,Routes里只放Route组件,把自定义组件放到element属性上。
自定义权限组件
javascript
const AuthRoute = (props) => {
const auth = null
if (!auth) {
return <Navigate to="login" />
}
return props.component
}
<Routes>
<Route exact path="/" element={<AuthRoute component={<Home />} />} />
...
</Routes>
效果:
现在路由拦截完成了,不过Route组件可以嵌套,所以可以在优化下Route的写法,外层Route是自定义组件,内层Route是页面组件,这样代码更清晰
javascript
AuthRoute组件:
import { Navigate, Outlet } from 'react-router-dom'
const AuthRoute = (props) => {
const auth = null
if (!auth) {
return <Navigate to="login" />
}
return <Outlet />
}
export default AuthRoute
<HashRouter>
<Routes>
<Route path="/" element={<AuthRoute />}>
<Route path='/' element={<Home/>}/>
</Route>
<Route path="/login" element={<Login />}></Route>
</Routes>
</HashRouter>
其中Outlet组件就是内层的Route组件,这样就可以把所有需要状态认证的路由都放到内层的Route组件。
总结:react-router4和react-router6的路由拦截大同小异,主要核心是在返回页面组件前先进行一些状态判断,根据判断的结果来进行路由跳转,从而实现路由拦截。