路由

Nuxt 的文件系统路由会为 pages/ 目录中的每个文件创建一条路由。

Nuxt 的核心功能之一是文件系统路由。pages/ 目录中的每个 Vue 文件都会生成一个对应的 URL(或路由),显示该文件的内容。通过为每个页面使用动态导入,Nuxt 利用代码分割来确保请求的路由发送最少量的 JavaScript。

页面

Nuxt 的路由基于 vue-router,根据 pages/ 目录 中创建的每个组件的文件名生成路由。

这种文件系统路由使用命名约定来创建动态和嵌套路由:

-| pages/
---| about.vue
---| index.vue
---| posts/
-----| [id].vue
阅读更多 Docs > Guide > Directory Structure > Pages.

导航

<NuxtLink> 组件用于在页面之间建立链接。它会渲染一个带有 href 属性设置为页面路由的 <a> 标签。一旦应用完成水合,页面过渡将通过 JavaScript 更新浏览器 URL 来执行。这可以避免整页刷新,并允许实现动画过渡。

当客户端的 <NuxtLink> 进入视口时,Nuxt 会自动提前预取链接页面的组件和数据(生成页面),从而加快导航速度。

pages/app.vue
<template>
  <header>
    <nav>
      <ul>
        <li><NuxtLink to="/about">关于</NuxtLink></li>
        <li><NuxtLink to="/posts/1">文章 1</NuxtLink></li>
        <li><NuxtLink to="/posts/2">文章 2</NuxtLink></li>
      </ul>
    </nav>
  </header>
</template>
阅读更多 Docs > API > Components > Nuxt Link.

路由参数

useRoute() 组合式函数可以在 Vue 组件的 <script setup> 块或 setup() 方法中使用,以访问当前路由的详细信息。

pages/posts/[id].vue
<script setup lang="ts">
const route = useRoute()

// 当访问 /posts/1 时,route.params.id 将为 1
console.log(route.params.id)
</script>
阅读更多 Docs > API > Composables > Use Route.

路由中间件

Nuxt 提供了一个可定制的路由中间件框架,你可以在整个应用中使用,非常适合提取在导航到特定路由之前需要运行的代码。

路由中间件在 Nuxt 应用的 Vue 部分运行。尽管名称相似,但它们与在 Nitro 服务器部分运行的服务器中间件完全不同。

路由中间件有三种类型:

  1. 匿名(或内联)路由中间件,直接在使用的页面中定义。
  2. 命名路由中间件,放置在 middleware/ 目录中,并在页面使用时通过异步导入自动加载。(注意:路由中间件名称会标准化为 kebab-case,例如 someMiddleware 会变为 some-middleware。)
  3. 全局路由中间件,放置在 middleware/ 目录 中(带有 .global 后缀),会在每次路由更改时自动运行。

以下是一个保护 /dashboard 页面的 auth 中间件示例:

export default defineNuxtRouteMiddleware((to, from) => {
  // isAuthenticated() 是一个示例方法,用于验证用户是否已认证
  if (isAuthenticated() === false) {
    return navigateTo('/login')
  }
})
阅读更多 Docs > Guide > Directory Structure > Middleware.

路由验证

Nuxt 通过在每个需要验证的页面中使用 definePageMeta()validate 属性提供路由验证。

validate 属性接受 route 作为参数。你可以返回一个布尔值来确定该路由是否是使用此页面渲染的有效路由。如果返回 false,并且找不到其他匹配项,将触发 404 错误。你也可以直接返回一个包含 statusCode/statusMessage 的对象,以立即响应错误(不会检查其他匹配项)。

如果你的用例更复杂,可以使用匿名路由中间件。

pages/posts/[id].vue
<script setup lang="ts">
definePageMeta({
  validate: async (route) => {
    // 检查 id 是否由数字组成
    return typeof route.params.id === 'string' && /^\d+$/.test(route.params.id)
  }
})
</script>
阅读更多 Docs > API > Utils > Define Page Meta.