Modern.js 的路由基于 React Router 6,提供了基于文件约定的路由能力,并支持了业界流行的约定式路由模式:嵌套路由。当入口被识别为 约定式路由 时,Modern.js 会自动基于文件系统,生成对应的路由结构。
本小节提到的路由,都是约定式路由。
嵌套路由是一种将 URL 分段与组件层次结构和数据耦合起来的路由模式。通常,URL 段会决定:
因此,使用嵌套路由时,页面的路由与 UI 结构是相呼应的,我们将会详细介绍这种路由模式。
在 routes/
目录下,子目录名会作为路由 URL 的映射,Modern.js 有两个文件约定 layout.tsx
和 page.tsx
。这两个文件决定了应用的布局层次,其中:
page.tsx
为内容组件,所在目录下存在该文件时,对应的路由 URL 可访问。layout.tsx
为布局组件,控制所在目录下所有子路由的布局,使用 <Outlet>
表示子组件。.ts
、.js
、.jsx
或 .tsx
文件扩展名可用于上述约定文件。
<Page>
组件是指 routes/
目录下所有 page.tsx
文件,也是所有路由的叶子组件。除了通配路由外,任何路由都应该由 <Page>
组件结束。
当应用中存在以下目录结构时:
会产出下面两条路由:
/
/user
<Layout>
组件是指 routes/
目录下所有 layout.tsx
文件,它们表示对应路由片段的布局,使用 <Outlet>
表示子组件。
<Outlet>
是 React Router 6 中提供的 API,详情可以查看 Outlet。
不同目录结构下,<Outlet>
所代表的组件也不同。为了方便介绍 <Layout>
与 <Outlet>
的关系,以下面的文件目录举例:
/
时,routes/layout.tsx
中的 <Outlet>
代表的是 routes/page.tsx
中导出的组件,路由的 UI 结构为:/blog
时,routes/layout.tsx
中的 <Outlet>
代表的是 routes/blog/page.tsx
中导出的组件,路由的 UI 结构为:/user
时,routes/layout.tsx
中的 <Outlet>
代表的是 routes/user/layout.tsx
中导出的组件。routes/user/layout.tsx
中的 <Outlet>
代表的是 routes/user/page.tsx
中导出的组件,路由的 UI 结构为:总结而言,如果子路由的文件目录下存在 layout.tsx
,上一级 layout.tsx
中的 <Outlet>
即为子路由文件目录下的 layout.tsx
,否则为子路由文件目录下的 page.tsx
。
通过 []
命名的文件目录,生成的路由会作为动态路由。例如以下文件目录:
routes/[id]/page.tsx
文件会转为 /:id
路由。除了可以确切匹配的 /blog
路由,其他所有 /xxx
都会匹配到该路由。
在组件中,可以通过 useParams 获取对应命名的参数。
通过 [$]
命名的文件目录,生成的路由会作为动态可选路由。例如以下文件目录:
routes/user/[id$]/page.tsx
文件会转为 /user/:id?
路由。/user
下的所有路由都会匹配到该路由,并且 id
参数可选存在。通常在区分创建与编辑时,可以使用该路由。
如果在某个子目录下存在 $.tsx
文件,该文件会作为通配路由组件,当没有匹配的路由时,会渲染该路由组件。
$.tsx
可以认为是一种特殊的 <Page>
组件,如果路由无法匹配,则 $.tsx
会作为 <Layout>
的子组件渲染。
如果当前目录下不存在 <Layout>
组件时,则 $.tsx
不会生效。
例如以下目录结构:
当你访问 /blog/a
时,无法匹配到任意路由,则页面会渲染 routes/blog/$.tsx
组件,路由的 UI 结构为:
如果希望访问 /blog
时,也匹配到 blog/$.tsx
文件,需要删除同目录下的 blog/layout.tsx
文件,同时保证 blog
下面没有其他子路由。
同样,$.tsx
中可以使用 useParams 捕获 url 的剩余部分。
通配路由可以被添加到 routes/
目录下的任意子目录中,一种常见的使用场景是通过 $.tsx
文件去定制任意层级的 404 内容。
例如你需要对所有未匹配到的路由,都展示一个 404 页面,可以添加 routes/$.tsx
文件:
此时,当访问除了 /
或 /blog/*
以外的路由时,都会匹配到 routes/$.tsx
组件,展示 404 页面。
某些场景下,每个路由会拥有属于自己的数据,应用需要在其他组件中获取匹配到的路由的这些数据。一个常见的例子是在布局中获取到匹配路由的面包屑信息。
Modern.js 提供了独立的约定,每个 Layout
, $
或 Page
文件都可以定义一个自己的 config
文件,如 page.config.ts
,该文件中我们约定了一个具名导出 handle
,这个字段中你可以定义任意属性:
定义的这些属性可以通过 useMatches
hook 获取。
当目录名以 __
开头时,对应的目录名不会转换为实际的路由路径,例如以下文件目录:
Modern.js 会生成 /login
和 /sign
两条路由,__auth/layout.tsx
组件会作为 login/page.tsx
和 sign/page.tsx
的布局组件,但__auth
不会作为路由路径片段出现在用户访问的 URL 中。
当需要为某些类型的路由,做独立的布局,或是想要将路由做归类时,这一功能非常有用。
有些情况下,项目需要较为复杂的路由,但这些路由又不存在独立的 UI 布局,如果像普通文件目录那样创建路由会导致目录层级较深。
因此 Modern.js 支持了通过 .
来分割路由片段,代替文件目录。例如,当需要 /user/profile/2022/edit
时,可以直接创建如下文件:
访问路由时,将得到如下 UI 布局:
某些应用中,可能需要根据用户的身份信息,或是其他数据条件,选择重定向到其他路由。在 Modern.js 中,你可以使用 Data Loader
文件来获取数据,或是和传统 React 组件那样,在 useEffect
中请求数据。
在任意的 page.tsx
同级目录中创建,page.data.ts
文件,这个文件就是该路由的 Data Loader。在 Data Loader 中,你可以通过调用 redirect
API 来完成路由的重定向。
在组件内做重定向,则可以通过 useNavigate
hook,示例如下:
routes/
下每一层目录中,开发者同样可以定义一个 error.tsx
文件,默认导出一个 <ErrorBoundary>
组件。当路由目录下存在该组件时,组件渲染出错会被 ErrorBoundary
组件捕获。
<ErrorBoundary>
可以返回出错时的 UI 视图,当前层级未声明 <ErrorBoundary>
组件时,错误会向上冒泡到更上层的组件,直到被捕获或抛出错误。同时,当组件出错时,只会影响捕获到该错误的路由组件及子组件,其他组件的状态和视图不受影响,可以继续交互。
在 <ErrorBoundary>
组件内,可以使用 useRouteError 获取的错误的具体信息:
此功能当前是实验性功能,后续 API 可能有调整。
在约定式路由下, Modern.js 会根据路由,自动地对路由进行分片(每一个路由都作为单独的 JS Chunk 进行加载),当用户访问具体的路由时,再自动加载对应的分片。这样可以有效地减少首屏加载的时间。但这也带来了一个问题,当用户访问一个路由时,如果该路由对应的分片还未加载完成,就会出现白屏的情况。
Modern.js 支持通过 loading.tsx
文件来解决这个问题,routes/
下每一层目录中,都可以创建 loading.tsx
文件,默认导出一个 <Loading>
组件。
如果当前目录下不存在 <Layout>
组件时,则 loading.tsx
不会生效。为了保证用户体验,Modern.js 推荐每个应用添加根 Loading 组件。
当路由目录下同时存在该组件和 layout
组件时,这一级路由下所有的子路由切换时,会先展示 loading.tsx
中导出的组件 UI,直到对应的 JS Chunk 加载完成。例如以下文件目录:
当定义 loading.tsx
时,当路由从 /
跳转到 /blog
,或从 /blog
跳转到 /blog/123
时,如果路由对应的 JS Chunk 还未加载,都会先展示 loading.tsx
中导出的组件 UI。相当于最终的 UI 结构如下:
大多数切换路由时白屏的情况,都可以通过定义 <Loading>
组件来优化体验。Modern.js 也支持在 <Link>
组件上定义 prefetch
属性,提前对静态资源和数据进行加载。
对于有更高性能要求的应用,预加载可以进一步提升用户体验,减少展示 <Loading>
组件的时间:
对数据的预加载目前只会预加载 SSR 项目中 Data Loader 中返回的数据。
prefetch
属性有三个可选值:
none
, 默认值,不会做 prefetch,没有任何额外的行为。intent
,这是我们推荐大多数场景下使用的值,当你把鼠标放在 Link 上时,会自动开始加载对应的分片和 Data Loader 中定义的数据,当鼠标移开时,会自动取消加载。在我们的测试中,即使是快速点击,也能减少大约 200ms 的加载时间。render
,当 <Link>
组件渲染时,就会加载对应的分片和 Data Loader 中定义的数据。render
加载路由分片的时机是可控的,只有在 <Link>
组件进入视窗时才触发,可以通过控制 <Link>
组件的渲染位置来控制分片加载时机。render
仅在空闲时对静态资源进行加载,不占用重要模块的加载时间。render
在 SSR 项目中还会发起数据预取。在根 <Layout>
组件中(routes/layout.ts
),可以动态地定义运行时配置:
在有些场景下,需要在应用渲染前做一些操作,可以在 routes/layout.tsx
中定义 init
钩子,init
在客户端和服务端均会执行,基本使用示例如下:
通过 init
钩子可以挂载一些全局的数据,在应用的其他地方可以访问 runtimeContext
变量:
该功能在应用需要页面前置的数据、自定义数据注入或是框架迁移(如 Next.js)时会非常有用。
配合 SSR 功能时,浏览器端可以获取到 SSR 时 init
返回的数据,开发者可以自行判断是否要在浏览器端重新获取数据来覆盖 SSR 数据,例如:
@modern-js/runtime/router
来导出 React Router API ?可以发现,在文档中所有的代码用例都是使用 @modern-js/runtime/router
包导出的 API,而不是直接使用 React Router 包导出的 API。那两者有什么区别呢?
首先,在 @modern-js/runtime/router
中导出的 API 和 React Router 包的 API 是完全一致的,如果某个 API 使用出现问题,请先检查 React Router 的文档和 Issues。
在使用约定式路由的情况下,务必使用 @modern-js/runtime/router
中的 API,不直接使用 React Router 的 API。因为 Modern.js 内部会安装 React Router,如果应用中使用了 React Router 的 API,可能会导致两个版本的 React Router 同时存在,出现不符合预期的行为。
如果应用中必须直接使用 React Router 包的 API,例如部分路由行为被封装在统一的 npm 包中,那应用可以通过设置 source.alias
,将 react-router
和 react-router-dom
统一指向项目的依赖,避免两个版本的 React Router 同时存在的问题。