React-router 手动实现静态路由(staticRoute)

简介: 使用vue的同学都知道,vue的路由是静态的。什么叫做静态的呢?那就是可以通过一个配置文件来进行路由的配置,而我们的react的路由是动态的。为啥这么说呢?react路由具有先天的优势——jsx . 每一个页面都是组件,每一个组件可以随意的嵌套。所以react的路由是动态加载的。那么我们是否也可以向vue一样,使用一个静态文件来自动配置呢?答案是:肯定可以的。

使用vue的同学都知道,vue的路由是静态的。什么叫做静态的呢?那就是可以通过一个配置文件来进行路由的配置,而我们的react的路由是动态的。为啥这么说呢?react路由具有先天的优势——jsx . 每一个页面都是组件,每一个组件可以随意的嵌套。所以react的路由是动态加载的。那么我们是否也可以向vue一样,使用一个静态文件来自动配置呢?答案是:肯定可以的。


  • 使用 npm install react-router-config 安装这个包,就可以使用静态路由了。详情配置文件查看官网
  • 我们可以自己手动来实现,因为react-router是非常灵活的。本文就描述大概手动实现的思路


效果


20210421124814764.gif


正文


声明配置文件


通过vue的配置文件,我们知道,配置文件最外层一定是一个数组,并且数组里面的对象有children属性。那么我们就构建一个这样的数组。如下:


// 这个对象数组的目的是,含有 Route 里面所有的属性,然后我们也可以自定义一些 name 啥,只是需要我们去手动处理。
const routeConfig = [
  {
    path: '/news', component: New, exact: false,
    // 这里的children里面的path是需要合并上父级的path。这样path就不会重复
    children: [
      { path: '/content', component: NewsCont, exact: true },
      { path: '/other', component: NewsOther, exact: true },
      { path: '/', component: NewsIndex, exact: true },
    ]
  },
  { path: '/', component: Index, exact: true },
]


展示组件试图(view)


vue 在注册好router 后,有一个组件叫做 <router-view> ,来作为所有组件的出口。因此我们在react 中也可以实现一个组件,来作为跟组件的出口。


/**
 * 核心组件的声明
 */
interface IRootRouteProp {
  routeConfig: routeObj[],
  basePath?: string,
}
/**
 * 核心关键组件
 * @param param0 
 * @returns 
 */
function RootRoute({ routeConfig, basePath = '' }: IRootRouteProp) {
  if (routeConfig.length === 0) {
    return <></>
  }
  // 判断根路径不能为空
  const rt = routeConfig.map((it, index) => {
    const { component: Comp, children: Child, path, ...rest } = it;
    // 获取一个新的路径,并且把 // 替换成 /
    const newPath = (basePath + path).replace(/\/\./g, '/');
    // 是否需要使用exact 进行精确匹配
    const isExact = 'exact' in it ? it.exact : true
    // ... 这里还可以拓展其他的属性,例如 如果需要使用vue的name属性的话
    return (
      <Route
        key={newPath}
        {...rest}
        exact={isExact}
        path={newPath}
        render={(value) => {
          let crt: JSX.Element = <></>;
          if (Array.isArray(Child) && Child.length > 0) {
            crt = RootRoute({ routeConfig: Child, basePath: newPath })
          }
          return (
            <Comp {...value} >
              {/* 把子组件的route 放在children中, */}
              {crt}
            </Comp>
          )
        }}
      />
    )
  })
  return (
    <Switch>
      {rt}
    </Switch>
  )
}


然后我们在跟组件中就可以直接这么使用:


 {/* 类似 vue的route-view */} 这里传递配置文件的参数
  <RootRoute routeConfig={routeConfig} />


在子组件中我们也是需要使用出口的,方式是:


  {/* 重新匹配子路由,这一行很关键,子组件是否能够匹配到路由 */}
  // 因为我们把嵌套的子路由是放在了children属性中,需要在子组件中进行一下渲染
      {prop.children}


最终代码


import React from 'react'
import { BrowserRouter as Router, NavLink, Route, RouteComponentProps, Switch, useRouteMatch } from 'react-router-dom'
/**
 * 测试组件
 * @returns 
 */
export default function StaticRoute() {
  return (
    <div>
      <Router>
        <NavLink exact activeStyle={{ color: '#F40' }} to='/' style={{ marginRight: '20px' }}>首页</NavLink>
        <NavLink activeStyle={{ color: '#F40' }} to='/news'>新闻页</NavLink>
        {/* 类似 vue的route-view */}
        <RootRoute routeConfig={routeConfig} />
      </Router>
    </div>
  )
}
/**
 * 首页
 * @returns 
 */
function Index() {
  return <h1>首页</h1>
}
interface INew {
  children?: JSX.Element
}
/**
 * 新闻组件
 * @param prop 
 * @returns 
 */
function New(prop: INew) {
  const rm = useRouteMatch()
  return (
    <div>
      <div style={{ margin: '30px' }}> 新闻页</div>
      <NavLink exact activeStyle={{ color: '#F40' }} to={`${rm.path}/`} style={{ marginRight: '20px' }}>新闻首页</NavLink>
      <NavLink exact activeStyle={{ color: '#F40' }} to={`${rm.path}/content`} style={{ marginRight: '20px' }}>新闻内容</NavLink>
      <NavLink exact activeStyle={{ color: '#F40' }} to={`${rm.path}/other`} style={{ marginRight: '20px' }}>新闻其他页</NavLink>
      {/* 重新匹配子路由,这一行很关键,子组件是否能够匹配到路由 */}
      {prop.children}
    </div>
  )
}
/**
 * 新闻首页
 * @returns 
 */
function NewsIndex() {
  return <h2>新闻首页的详情士大夫十分</h2>
}
/**
 * 新闻内容
 * @returns 
 */
function NewsCont() {
  return <h2>新闻内容的详情孙菲菲</h2>
}
/**
 * 其他新闻
 * @returns 
 */
function NewsOther() {
  return <h2>新闻其他是的方法可你男看就</h2>
}
/**
 * 核心组件的声明
 */
interface IRootRouteProp {
  routeConfig: routeObj[],
  basePath?: string,
}
/**
 * 核心关键组件
 * @param param0 
 * @returns 
 */
function RootRoute({ routeConfig, basePath = '' }: IRootRouteProp) {
  if (routeConfig.length === 0) {
    return <></>
  }
  // 判断根路径不能为空
  const rt = routeConfig.map((it, index) => {
    const { component: Comp, children: Child, path, ...rest } = it;
    // 获取一个新的路径,并且把 // 替换成 /
    const newPath = (basePath + path).replace(/\/\./g, '/');
    // 是否需要使用exact 进行精确匹配
    const isExact = 'exact' in it ? it.exact : true
    // ... 这里还可以拓展其他的属性
    return (
      <Route
        key={newPath}
        {...rest}
        exact={isExact}
        path={newPath}
        render={(value) => {
          let crt: JSX.Element = <></>;
          if (Array.isArray(Child) && Child.length > 0) {
            crt = RootRoute({ routeConfig: Child, basePath: newPath })
          }
          return (
            <Comp {...value} >
              {/* 把子组件的route 放在children中, */}
              {crt}
            </Comp>
          )
        }}
      />
    )
  })
  return (
    <Switch>
      {rt}
    </Switch>
  )
}
/**
 * 路由对象声明
 */
interface routeObj {
  path: string;
  component: React.ComponentType<RouteComponentProps<any>> | React.ComponentType<any>;
  exact: boolean;
  children?: routeObj[]
}
// 配置文件
const routeConfig: routeObj[] = [
  {
    path: '/news', component: New, exact: false,
    children: [
      { path: '/content', component: NewsCont, exact: true },
      { path: '/other', component: NewsOther, exact: true },
      { path: '/', component: NewsIndex, exact: true },
    ]
  },
  { path: '/', component: Index, exact: true },
]


相关文章
|
2月前
|
前端开发 API UED
React 路由守卫 Guarded Routes
【10月更文挑战第26天】本文介绍了 React 中的路由守卫(Guarded Routes),使用 `react-router-dom` 实现权限验证、登录验证和数据预加载等场景。通过创建 `AuthContext` 管理认证状态,实现 `PrivateRoute` 组件进行路由保护,并在 `App.js` 中使用。文章还讨论了常见问题和易错点,提供了处理异步操作的示例,帮助开发者提升应用的安全性和用户体验。
78 1
|
2月前
|
前端开发 安全 网络安全
React——路由Route
React——路由Route
43 2
React——路由Route
|
3月前
|
资源调度 前端开发 测试技术
React Router 路由管理
【10月更文挑战第10天】本文介绍了 React Router,一个在 React 应用中管理路由的强大工具。内容涵盖基本概念、安装与使用方法、常见问题及解决方案,如路由嵌套、动态路由和路由守卫等,并提供代码示例。通过学习本文,开发者可以更高效地使用 React Router,提升应用的导航体验和安全性。
413 19
|
5月前
|
移动开发 资源调度 前端开发
React Router V6 useRoutes的使用
【8月更文挑战第29天】 React Router V6 useRoutes的使用
237 3
|
5月前
|
前端开发 JavaScript 网络架构
为什么 React 中需要路由器?
【8月更文挑战第31天】
44 0
|
5月前
|
前端开发 JavaScript UED
什么是 React Router?
【8月更文挑战第31天】
34 0
|
8月前
|
资源调度 前端开发 JavaScript
React Router:React应用的路由管理
【4月更文挑战第25天】React Router是React的官方路由库,用于管理SPA的路由。它基于组件,将URL映射到React组件,核心概念包括路由、链接和导航。设置路由时,在根组件中使用BrowserRouter或HashRouter,包裹Routes组件,定义Route规则。Link组件用于创建内部链接,实现导航。高级特性包括嵌套路由、参数化路由和编程式导航,如子路由、动态参数和JavaScript控制的导航。掌握React Router能帮助开发者更高效地构建复杂的React应用。
|
缓存 JavaScript
Vue Router 学习 new Router
Vue Router 学习 new Router
158 0
|
8月前
|
JavaScript 前端开发 网络架构
Vue Router:让你的应用路由起来!
Vue Router:让你的应用路由起来!
|
8月前
|
前端开发 JavaScript API
React Router v6 完全指南(下)
React Router v6 完全指南(下)
278 0