Remix 扁平路由

All Contributors

此包使您可以使用 flat-routes 约定来定义路由。它基于 Ryan Florence 的 gist

💡 React Router v7 支持

React Router v7 使用新的路由配置。为了简化从 Remix 的迁移,该团队发布了一个适配器包,该包将把现有的基于 Remix 文件的路由转换为新的配置格式。

要使用现有的基于文件的路由,请安装适配器并更新 routes.ts 来包装您的适配器。

npm install -D @react-router/remix-config-routes-adapter
npm install -D remix-flat-routes
// app/routes.ts
import { remixConfigRoutes } from "@react-router/remix-config-routes-adapter";
import { flatRoutes } from "remix-flat-routes";

export const routes = remixConfigRoutes((defineRoutes) => {
  return flatRoutes("routes", defineRoutes, {/* options */});
});

✨🎉 v0.5.0 中的新功能

Remix v2 扁平路由约定

remix-flat-routes 是扁平路由规范的最初实现。我根据用户反馈添加了一些增强功能。当 Remix v2 将扁平路由约定添加为默认值时,他们使用了原始规范。

如果您想要混合路由、扩展的路由文件名、自定义参数前缀等增强功能,您将需要继续使用此包。

remix-flat-routes 将始终保持与默认 Remix 约定的兼容性。此包只是核心约定的超集/扩展。

注意:像 Kent C. Dodds 的 Epic Stack 这样的热门项目使用了 remix-flat-routes

混合路由

您现在可以使用嵌套文件夹作为路由名称,同时仍然保留扁平路由的同位置功能。

如果您有一个大型应用程序,路由嵌套多层并不罕见。使用默认的扁平路由,文件夹名称就是整个路由路径:some.really.long.route.edit/index.tsx

通常,您可能有多个父布局,如 _publicadmin。您可以创建顶层文件夹,然后在它们下面嵌套您的路由,而不是必须在每个路由中重复名称。这样,您仍然可以利用带有同位置的扁平文件夹。

之前

❯ tree app/routes-folders
app/routes-folders
├── _index
│   └── page.tsx
├── _public
│   └── _layout.tsx
├── _public.about
│   └── index.tsx
├── _public.contact[.jpg]
│   └── index.tsx
├── test.$
│   ├── _route.server.tsx
│   └── _route.tsx
├── users
│   ├── _layout.tsx
│   └── users.css
├── users.$userId
│   ├── _route.tsx
│   └── avatar.png
├── users.$userId_.edit
│   └── _route.tsx
└── users._index
    └── index.tsx

之后

❯ tree app/routes-hybrid
app/routes-hybrid
├── _index
│   └── index.tsx
├── _public
│   ├── _layout.tsx
│   ├── about
│   │   └── _route.tsx
│   └── contact[.jpg]
│       └── _route.tsx
├── test.$
│   └── _route.tsx
└── users
    ├── $userId
    │   ├── _route.tsx
    │   └── avatar.png
    ├── $userId_.edit
    │   └── _route.tsx
    ├── _index
    │   └── index.tsx
    ├── _layout.tsx
    └── users.css

带有 flat-files 约定的嵌套文件夹(✨ v0.5.1 中的新功能)

要创建一个文件夹但将其视为扁平文件,只需在文件夹名称后附加 +

_auth+/forgot-password.tsx => _auth.forgot-password.tsx

注意:您可以在文件夹中包含 _layout.tsx 文件。您不需要有 _public.tsx 或 users.tsx 文件。

您仍然可以使用扁平文件夹进行同位置。因此,这是两种格式的最佳结合。

❯ tree app/routes-hybrid-files/
app/routes-hybrid-files/
├── _auth+
│   ├── forgot-password.tsx
│   └── login.tsx
├── _public+
│   ├── _layout.tsx
│   ├── about.tsx
│   ├── contact[.jpg].tsx
│   └── index.tsx
├── project+
│   ├── _layout.tsx
│   ├── parent.child
│   │   └── index.tsx
│   └── parent.child.grandchild
│       ├── index.tsx
│       └── styles.css
└── users+
    ├── $userId.tsx
    ├── $userId_.edit.tsx
    ├── _layout.tsx
    └── index.tsx
<Routes>
  <Route file="root.tsx">
    <Route
      path="forgot-password"
      file="routes-hybrid-files/_auth+/forgot-password.tsx"
    />
    <Route path="login" file="routes-hybrid-files/_auth+/login.tsx" />
    <Route file="routes-hybrid-files/_public+/_layout.tsx">
      <Route path="about" file="routes-hybrid-files/_public+/about.tsx" />
      <Route
        path="contact.jpg"
        file="routes-hybrid-files/_public+/contact[.jpg].tsx"
      />
      <Route index file="routes-hybrid-files/_public+/index.tsx" />
    </Route>
    <Route path="project" file="routes-hybrid-files/project+/_layout.tsx">
      <Route
        path="parent/child"
        file="routes-hybrid-files/project+/parent.child/index.tsx"
      >
        <Route
          path="grandchild"
          file="routes-hybrid-files/project+/parent.child.grandchild/index.tsx"
        />
      </Route>
    </Route>
    <Route path="users" file="routes-hybrid-files/users+/_layout.tsx">
      <Route path=":userId" file="routes-hybrid-files/users+/$userId.tsx" />
      <Route
        path=":userId/edit"
        file="routes-hybrid-files/users+/$userId_.edit.tsx"
      />
      <Route index file="routes-hybrid-files/users+/index.tsx" />
    </Route>
  </Route>
</Routes>

扩展的路由文件名

除了标准的 index | route | page | layout 名称之外,任何具有 _ 前缀的文件都将被视为路由文件。这将更容易找到特定的路由,而不是浏览一堆 route.tsx 文件。这受到了 SolidStart "重命名索引" 功能的启发。

所以,而不是

_public.about/route.tsx
_public.contact/route.tsx
_public.privacy/route.tsx

您可以将其命名为

_public.about/_about.tsx
_public.contact/_contact.tsx
_public.privacy/_privacy.tsx

多个路由文件夹

您现在可以传递除了默认 routes 文件夹之外的其他路由文件夹。这些路由将合并到一个命名空间中,因此您可以在一个文件夹中拥有路由,这些路由将使用另一个文件夹中的共享路由。

自定义参数前缀

您可以覆盖默认的参数前缀 $。一些 shell 使用 $ 前缀作为变量,这可能会由于 shell 扩展而出现问题。使用任何有效的字符作为文件名,例如:^

users.^userId.tsx  => users/:userId
test.^.tsx         => test/*

自定义基本路径

您可以覆盖默认的基本路径 /。这会将您的基本路径添加到根路径。

可选路由段

React Router 将为可选路由段引入一项新功能。要在扁平路由中使用可选段,只需将您的路由名称包装在 () 中。

parent.(optional).tsx   => parent/optional?

自定义应用程序目录

您可以覆盖默认的应用程序目录 app

🛠 安装

npm install -D remix-flat-routes

⚙️ 配置

更新您的 *remix.config.js* 文件并使用自定义路由配置选项。

const { flatRoutes } = require('remix-flat-routes')

/**
 * @type {import("@remix-run/dev").AppConfig}
 */
module.exports = {
  // ignore all files in routes folder to prevent
  // default remix convention from picking up routes
  ignoredRouteFiles: ['**/*'],
  routes: async defineRoutes => {
    return flatRoutes('routes', defineRoutes)
  },
}

API

function flatRoutes(
  routeDir: string | string[],
  defineRoutes: DefineRoutesFunction,
  options: FlatRoutesOptions,
)

type FlatRoutesOptions = {
  appDir?: string // optional app directory (defaults to app)
  basePath?: string // optional base path (default is '/')
  paramPrefixChar?: string // optional param prefix (default is '$')
  ignoredRouteFiles?: string[] // optional files to ingore as routes (same as Remix config option)
  visitFiles?: VisitFilesFunction // optional visitor (useful for tests to provide files without file system)
}

注意:routeDir 应该相对于 app 文件夹。如果要使用 routes 文件夹,您将需要更新 ignoredRouteFiles 属性以忽略所有文件:**/*

🔨 扁平路由约定

示例(扁平文件)

routes/
  _auth.forgot-password.tsx
  _auth.login.tsx
  _auth.reset-password.tsx
  _auth.signup.tsx
  _auth.tsx
  _landing.about.tsx
  _landing.index.tsx
  _landing.tsx
  app.calendar.$day.tsx
  app.calendar.index.tsx
  app.calendar.tsx
  app.projects.$id.tsx
  app.projects.tsx
  app.tsx
  app_.projects.$id.roadmap.tsx
  app_.projects.$id.roadmap[.pdf].tsx

作为 React Router 路由

<Routes>
  <Route element={<Auth />}>
    <Route path="forgot-password" element={<Forgot />} />
    <Route path="login" element={<Login />} />
    <Route path="reset-password" element={<Reset />} />
    <Route path="signup" element={<Signup />} />
  </Route>
  <Route element={<Landing />}>
    <Route path="about" element={<About />} />
    <Route index element={<Index />} />
  </Route>
  <Route path="app" element={<App />}>
    <Route path="calendar" element={<Calendar />}>
      <Route path=":day" element={<Day />} />
      <Route index element={<CalendarIndex />} />
    </Route>
    <Route path="projects" element={<Projects />}>
      <Route path=":id" element={<Project />} />
    </Route>
  </Route>
  <Route path="app/projects/:id/roadmap" element={<Roadmap />} />
  <Route path="app/projects/:id/roadmap.pdf" />
</Routes>

单独解释

文件名 URL 嵌套在...
_auth.forgot-password.tsx /forgot-password _auth.tsx
_auth.login.tsx /login _auth.tsx
_auth.reset-password.tsx /reset-password _auth.tsx
_auth.signup.tsx /signup _auth.tsx
_auth.tsx root.tsx
_landing.about.tsx /about _landing.tsx
_landing.index.tsx / _landing.tsx
_landing.tsx root.tsx
app.calendar.$day.tsx /app/calendar/:day app.calendar.tsx
app.calendar.index.tsx /app/calendar app.calendar.tsx
app.projects.$id.tsx /app/projects/:id app.projects.tsx
app.projects.tsx /app/projects app.tsx
app.tsx /app root.tsx
app_.projects.$id.roadmap.tsx /app/projects/:id/roadmap root.tsx
app_.projects.$id.roadmap[.pdf].tsx /app/projects/:id/roadmap.pdf 无 (资源路由)

嵌套布局

默认匹配

默认情况下,flat-routes 会将当前路由嵌套到具有最长匹配前缀的父布局中。

给定布局路由 app.calendar.tsx,以下路由将嵌套在 app.calendar.tsx 下,因为 app.calendar 是最长的匹配前缀。

  • app.calendar.index.tsx
  • app.calendar.$day.tsx

覆盖匹配

有时您想使用路由层次结构中较高的父布局。使用默认的 Remix 约定,您将使用点(.)表示法而不是嵌套文件夹。使用 flat-routes,由于路由文件始终使用点,因此有一个不同的约定来指定要嵌套在哪个布局下。

假设您有一个 app.tsx 布局,并且您有一个不想与该布局共享的路由,而是想与 root.tsx 匹配。要覆盖默认的父匹配,请在要嵌套在其下的路由的直接子段后附加一个尾随下划线(_)。

app_.projects.$id.roadmap.tsx 将嵌套在 root 下,因为没有匹配的路由

  • app_.projects.$id.tsx
  • app_.projects.tsx
  • app_.tsx
  • root.tsx

约定

文件名 约定 行为
privacy.jsx 文件名 普通路由
pages.tos.jsx 带点的无布局 普通路由,. -> /
about.jsx 带子级的文件名 父布局路由
about.contact.jsx 布局的子路由
about.index.jsx 索引文件名 布局的索引路由
about._index.jsx index.tsx 的别名 布局的索引路由*
about_.company.jsx 尾随下划线 URL 段,无布局
app_.projects.$id.roadmap.tsx 尾随下划线 更改默认父布局
_auth.jsx 前导下划线 布局嵌套,无 URL 段
_auth.login.jsx 前导下划线 无路径布局路由的子级
users.$userId.jsx 前导 $ URL 参数
docs.$.jsx 裸 $ 通配符路由
dashboard.route.jsx 路由后缀 可选,完全忽略
investors/[index].jsx 方括号 转义传统字符

注意:索引路由的下划线前缀是可选的,但有助于将文件排序到目录列表的顶部。

理由

  • 更容易查看应用程序定义的路由 - 只需打开 "routes/",它们都在那里。由于文件系统通常先对文件夹进行排序,当您有数十个路由时,很难看到哪些文件夹具有布局,哪些没有。现在,所有相关的路由都排序在一起。

  • 减少重构/重新设计的摩擦 - 虽然代码编辑器在您移动文件时非常擅长修复导入,并且 Remix 具有 "~" 导入别名,但重构没有一堆嵌套文件夹的代码库通常更容易。Remix 不再强制执行此操作。

    此外,在重新设计用户界面时,调整文件名称比创建/删除文件夹并移动路由来更改它们的嵌套方式更简单。

  • 帮助应用迁移到 Remix - 现有的应用程序通常没有像现在约定俗成的嵌套路由文件夹结构。迁移到 Remix 很困难,因为你必须处理所有的导入。

并置

虽然示例中只使用了文件,但它们实际上只是“导入路径”。因此,你可以为路由创建一个文件夹,并且会导入 `index` 文件,从而使路由的所有模块彼此并存。这与上面详细介绍的 *扁平文件* 约定相反,是一种 *扁平文件夹* 约定。

示例 (扁平文件夹)

routes/
  _auth.forgot-password.tsx
  _auth.login.tsx
  _auth.tsx
  _landing.about.tsx
  _landing.index.tsx
  _landing.tsx
  app.projects.tsx
  app.projects.$id.tsx
  app.tsx
  app_.projects.$id.roadmap.tsx

每个路由都变成一个文件夹,文件夹名称为路由名称去掉文件扩展名。路由文件然后被命名为 *index.tsx*。

所以 *app.projects.tsx* 变为 *app.projects/index.tsx*

routes/
  _auth/
    index.tsx x <- route file (same as _auth.tsx)
  _auth.forgot-password/
    index.tsx  <- route file (same as _auth.forgot-password.tsx)
  _auth.login/
    index.tsx   <- route files (same as _auth.login.tsx)
  _landing.about/
    index.tsx   <- route file (same as _landing.about.tsx)
    employee-profile-card.tsx
    get-employee-data.server.tsx
    team-photo.jpg
  _landing.index/
    index.tsx   <- route file (same as _landing.index.tsx)
    scroll-experience.tsx
  _landing/
    index.tsx   <- route file (same as _landing.tsx)
    header.tsx
    footer.tsx
  app/
    index.tsx   <- route file (same as app.tsx)
    primary-nav.tsx
    footer.tsx
  app_.projects.$id.roadmap/
    index.tsx   <- route file (same as app_.projects.$id.roadmap.tsx)
    chart.tsx
    update-timeline.server.tsx
  app.projects/
    index.tsx <- layout file (sames as app.projects.tsx)
    project-card.tsx
    get-projects.server.tsx
    project-buttons.tsx
  app.projects.$id/
    index.tsx  <- route file (sames as app.projects.$id.tsx)

别名

由于路由文件现在命名为 *index.tsx*,并且你可以在同一个路由文件夹中并置其他文件,因此 *index.tsx* 文件可能会在文件列表中迷失。你还可以为 *index.tsx* 使用以下别名。下划线前缀会将文件排序到目录列表的顶部。

  • _index.tsx
  • _layout.tsx
  • _route.tsx

注意:*_layout.tsx* 和 *_route.tsx* 文件只是对其作用更加明确。它们与 *index.tsx* 的工作方式相同。

与扁平文件一样,索引路由(不要与索引路由*文件*混淆)也可以使用下划线前缀。路由 `_landing.index` 可以保存为 `_landing.index/index.tsx` 或 `_landing._index/_index.tsx`。

这有点主观,但我认为这最终是大多数开发人员会喜欢的。每个路由都变成它自己的“迷你应用”,所有依赖项都在一起。使用 `ignoredRouteFiles` 选项,完全不清楚哪些文件是路由,哪些不是。

🚚 迁移现有路由

你现在可以将现有的路由迁移到新的 `flat-routes` 约定。只需运行

npx migrate-flat-routes <sourceDir> <targetDir> [options]

Example:
  npx migrate-flat-routes ./app/routes ./app/flatroutes --convention=flat-folders

NOTE:
  sourceDir and targetDir are relative to project root

Options:
  --convention=<convention>
    The convention to use when migrating.
      flat-files - Migrates to flat files
      flat-folders - Migrates to flat directories with route.tsx files
      hybrid - Keep folder structure with '+' suffix and _layout files
  --force
    Overwrite target directory if it exists

😍 贡献者

感谢这些优秀的人们(表情符号键


Kiliman

💻 📖

Ryan Florence

📖

Brandon Pittman

📖 💻

Mehdi Achour

📖

Fidel González

📖

Andrew Haines

💻

Wonu Lee

💻

Markus Wolf

💻

Sarat Chandra Balla

💻

本项目遵循 all-contributors 规范。欢迎任何形式的贡献!