任何框架的学习,要想学习好,首先必须全部了解最基本的 api,然后才是框架的封装,框架的优化,框架的底层原理,框架的源码。
工作项目的开发没有问题,不代表你就已经完全了解了vue-Router,认真的研究文档的每一个细节,会保证你的开发思路更加清晰,开发效率更高。
基础
入门
router-link
使用 router-link 组件进行导航
通过传递 to
来指定链接<router-link>
将呈现一个带有正确 href
属性的 <a>
标签
请注意,我们没有使用常规的 a 标签,而是使用一个自定义组件 router-link 来创建链接。这使得 Vue Router 可以在不重新加载页面的情况下更改 URL,处理 URL 的生成以及编码。
router-view
路由出口
路由匹配到的组件将渲染在这里
router-view 将显示与 url 对应的组件。你可以把它放在任何地方,以适应你的布局。
1 | <script src="https://unpkg.com/vue@3"></script> |
1 | // 1. 定义路由组件. |
通过调用 app.use(router),我们会触发第一次导航且可以在任意组件中以 this.$router 的形式访问它,并且以 this.$route 的形式访问当前路由:
1 | // Home.vue |
要在 setup 函数中访问路由,请调用 useRouter 或 useRoute 函数。我们将在 Composition API 中了解更多信息。
在整个文档中,我们会经常使用 router 实例,请记住,this.$router 与直接使用通过 createRouter 创建的 router 实例完全相同。我们使用 this.$router 的原因是,我们不想在每个需要操作路由的组件中都导入路由。
动态路由匹配
带参数的动态路由匹配
很多时候,我们需要将给定匹配模式的路由映射到同一个组件。例如,我们可能有一个 User 组件,它应该对所有用户进行渲染,但用户 ID 不同。在 Vue Router 中,我们可以在路径中使用一个动态字段来实现,我们称之为 路径参数 :
1 | const User = { |
现在像 /users/johnny 和 /users/jolyne 这样的 URL 都会映射到同一个路由。
路径参数 用冒号 : 表示。当一个路由被匹配时,它的 params 的值将在每个组件中以 this.$route.params 的形式暴露出来。因此,我们可以通过更新 User 的模板来呈现当前的用户 ID:
1 | const User = { |
你可以在同一个路由中设置有多个 路径参数,它们会映射到 $route.params 上的相应字段。例如:
模式 | 匹配路径 | $route.params |
---|---|---|
/user/:username | /user/evan | username: ‘evan’ |
/user/:username/post/:post_id | /user/evan/post/123 | username: ‘evan’, post_id: 123 |
除了 $route.params 之外,$route 对象还公开了其他有用的信息,如 $route.query(如果 URL 中存在参数)、$route.hash 等。你可以在 API参考 中查看完整的细节。
响应路由参数的变化
使用带有参数的路由时需要注意的是,当用户从 /users/johnny 导航到 /users/jolyne 时,相同的组件实例将被重复使用。因为两个路由都渲染同个组件,比起销毁再创建,复用则显得更加高效。不过,这也意味着组件的生命周期钩子不会被调用。
要对同一个组件中参数的变化做出响应的话,你可以简单地 watch $route 对象上的任意属性,在这个场景中,就是 $route.params :
1 | const User = { |
或者,使用 beforeRouteUpdate 导航守卫,它也可以取消导航:
1 | const User = { |
捕获所有路由或 404 Not found 路由
常规参数只匹配 url 片段之间的字符,用 / 分隔。如果我们想匹配任意路径,我们可以使用自定义的 路径参数 正则表达式,在 路径参数 后面的括号中加入 正则表达式 :
1 | const routes = [ |
在这个特定的场景中,我们在括号之间使用了自定义正则表达式,并将pathMatch 参数标记为可选可重复。这样做是为了让我们在需要的时候,可以通过将 path 拆分成一个数组,直接导航到路由:
1 | this.$router.push({ |
更多内容请参见重复参数部分。
如果你正在使用历史模式,请务必按照说明正确配置你的服务器。
路由的匹配语法
大多数应用都会使用 /about 这样的静态路由和 /users/:userId 这样的动态路由,就像我们刚才在动态路由匹配中看到的那样,但是 Vue Router 可以提供更多的方式!
在参数中自定义正则
当定义像 :userId 这样的参数时,我们内部使用以下的正则 ([^/]+) (至少有一个字符不是斜杠 / )来从 URL 中提取参数。这很好用,除非你需要根据参数的内容来区分两个路由。想象一下,两个路由 /:orderId 和 /:productName,两者会匹配完全相同的 URL,所以我们需要一种方法来区分它们。最简单的方法就是在路径中添加一个静态部分来区分它们:
1 | const routes = [ |
但在某些情况下,我们并不想添加静态的 /o /p 部分。由于,orderId 总是一个数字,而 productName 可以是任何东西,所以我们可以在括号中为参数指定一个自定义的正则:
1 | const routes = [ |
现在,转到 /25 将匹配 /:orderId,其他情况将会匹配 /:productName。routes 数组的顺序并不重要!
可重复的参数
如果你需要匹配具有多个部分的路由,如 /first/second/third,你应该用 *(0 个或多个)和 +(1 个或多个)将参数标记为可重复:
1 | const routes = [ |
这将为你提供一个参数数组,而不是一个字符串,并且在使用命名路由时也需要你传递一个数组:1
2
3
4
5
6
7
8
9// 给定 { path: '/:chapters*', name: 'chapters' },
router.resolve({ name: 'chapters', params: { chapters: [] } }).href
// 产生 /
router.resolve({ name: 'chapters', params: { chapters: ['a', 'b'] } }).href
// 产生 /a/b
// 给定 { path: '/:chapters+', name: 'chapters' },
router.resolve({ name: 'chapters', params: { chapters: [] } }).href
// 抛出错误,因为 `chapters` 为空
这些也可以通过在右括号后添加它们与自定义正则结合使用:
1 | const routes = [ |
Sensitive 与 strict 路由配置
默认情况下,所有路由是不区分大小写的,并且能匹配带有或不带有尾部斜线的路由。例如,路由 /users 将匹配 /users、/users/、甚至 /Users/。这种行为可以通过 strict 和 sensitive 选项来修改,它们可以既可以应用在整个全局路由上,又可以应用于当前路由上:
1 | const router = createRouter({ |
可选参数
你也可以通过使用 ? 修饰符(0 个或 1 个)将一个参数标记为可选:
1 | const routes = [ |
请注意,* 在技术上也标志着一个参数是可选的,但 ? 参数不能重复。
嵌套路由
一些应用程序的 UI 由多层嵌套的组件组成。在这种情况下,URL 的片段通常对应于特定的嵌套组件结构,例如:
1 | /user/johnny/profile /user/johnny/posts |
通过 Vue Router,你可以使用嵌套路由配置来表达这种关系。
接着上节创建的 app :
1 | <div id="app"> |
1 | const User = { |
这里的 <router-view>
是一个顶层的 router-view。它渲染顶层路由匹配的组件。同样地,一个被渲染的组件也可以包含自己嵌套的 <router-view>
。例如,如果我们在 User 组件的模板内添加一个 <router-view>
:
1 | const User = { |
要将组件渲染到这个嵌套的 router-view
中,我们需要在路由中配置 children
:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20const routes = [
{
path: '/user/:id',
component: User,
children: [
{
// 当 /user/:id/profile 匹配成功
// UserProfile 将被渲染到 User 的 <router-view> 内部
path: 'profile',
component: UserProfile,
},
{
// 当 /user/:id/posts 匹配成功
// UserPosts 将被渲染到 User 的 <router-view> 内部
path: 'posts',
component: UserPosts,
},
],
},
]
注意,以 / 开头的嵌套路径将被视为根路径。这允许你利用组件嵌套,而不必使用嵌套的 URL。
如你所见,children
配置只是另一个路由数组,就像 routes 本身一样。因此,你可以根据自己的需要,不断地嵌套视图。
此时,按照上面的配置,当你访问 /user/eduardo
时,在 User 的 router-view
里面什么都不会呈现,因为没有匹配到嵌套路由。也许你确实想在那里渲染一些东西。在这种情况下,你可以提供一个空的嵌套路径:
1 | const routes = [ |
嵌套的命名路由
在处理命名路由时,你通常会给子路由命名:
1 | const routes = [ |
这将确保导航到 /user/:id
时始终显示嵌套路由。
在一些场景中,你可能希望导航到命名路由而不导航到嵌套路由。例如,你想导航 /user/:id
而不显示嵌套路由。那样的话,你还可以命名父路由,但请注意重新加载页面将始终显示嵌套的子路由,因为它被视为指向路径/users/:id
的导航,而不是命名路由:
1 | const routes = [ |
编程导航
除了使用 <router-link>
创建 a 标签来定义导航链接,我们还可以借助 router
的实例方法,通过编写代码来实现。
导航到不同的位置
router.push()
注意:在 Vue 实例中,你可以通过 $router 访问路由实例。因此你可以调用
this.$router.push
。
想要导航到不同的 URL,可以使用 router.push
方法。这个方法会向 history 栈添加一个新的记录,所以,当用户点击浏览器后退按钮时,会回到之前的 URL。
当你点击 <router-link>
时,内部会调用这个方法,所以点击 <router-link :to="...">
相当于调用 router.push(...) :
声明式 | 编程式 |
---|---|
<router-link :to="..."> |
router.push(...) |
该方法的参数可以是一个字符串路径,或者一个描述地址的对象。例如:
1 | // 字符串路径 |
注意:如果提供了 path
,params
会被忽略,上述例子中的 query
并不属于这种情况。取而代之的是下面例子的做法,你需要提供路由的 name
或手写完整的带有参数的 path
:
1 | const username = 'eduardo' |
当指定 params
时,可提供 string
或 number
参数(或者对于可重复的参数可提供一个数组)。任何其他类型(如 undefined、false 等)都将被自动字符串化。对于可选参数,你可以提供一个空字符串(""
)来跳过它。
由于属性 to
与 router.push
接受的对象种类相同,所以两者的规则完全相同。
router.push
和所有其他导航方法都会返回一个 Promise
,让我们可以等到导航完成后才知道是成功还是失败。
替换当前位置
router.replace()
它的作用类似于 router.push
,唯一不同的是,它在导航时不会向 history
添加新记录,正如它的名字所暗示的那样——它取代了当前的条目。
声明式 | 编程式 |
---|---|
<router-link :to="..." replace> |
router.replace(...) |
也可以直接在传递给 router.push
的 routeLocation 中增加一个属性 replace: true
:
1 | router.push({ path: '/home', replace: true }) |
横跨历史
router.go()
该方法采用一个整数作为参数,表示在历史堆栈中前进或后退多少步,类似于 window.history.go(n)
。
1 | // 向前移动一条记录,与 router.forward() 相同 |
篡改历史
你可能已经注意到,router.push
、router.replace
和 router.go
是 window.history.pushState
、window.history.replaceState
和 window.history.go
的翻版,它们确实模仿了 window.history
的 API。
因此,如果你已经熟悉 Browser History APIs,在使用 Vue Router
时,操作历史记录就会觉得很熟悉。
值得一提的是,无论在创建路由器实例时传递什么样的 history 配置,Vue Router 的导航方法( push
、replace
、go
)都能始终正常工作。
命名路由
除了 path 之外,你还可以为任何路由提供 name。这有以下优点:
- 没有硬编码的 URL
- params 的自动编码/解码。
- 防止你在 url 中出现打字错误。
- 绕过路径排序(如显示一个)
1 | const routes = [ |
要链接到一个命名的路由,可以向 router-link
组件的 to
属性传递一个对象:
1 | <router-link :to="{ name: 'user', params: { username: 'erina' }}"> |
这跟代码调用 router.push()
是一回事:
1 | router.push({ name: 'user', params: { username: 'erina' } }) |
在这两种情况下,路由将导航到路径 /user/erina
。
命名视图
有时候想同时 (同级) 展示多个视图,而不是嵌套展示,例如创建一个布局,有 sidebar
(侧导航) 和 main
(主内容) 两个视图,这个时候命名视图就派上用场了。你可以在界面中拥有多个单独命名的视图,而不是只有一个单独的出口。如果 router-view
没有设置名字,那么默认为 default
。
1 | <router-view class="view left-sidebar" name="LeftSidebar"></router-view> |
一个视图使用一个组件渲染,因此对于同个路由,多个视图就需要多个组件。确保正确使用 components
配置 (带上 s):
1 | const router = createRouter({ |
嵌套命名视图
我们也有可能使用命名视图创建嵌套视图的复杂布局。这时你也需要命名用到的嵌套 router-view
组件。我们以一个设置面板为例:
1 | /settings/emails /settings/profile |
Nav
只是一个常规组件。UserSettings
是一个视图组件。UserEmailsSubscriptions
、UserProfile
、UserProfilePreview
是嵌套的视图组件。
注意:我们先忘记 HTML/CSS 具体的布局的样子,只专注在用到的组件上。UserSettings
组件的 <template>
部分应该是类似下面的这段代码:
1 | <!-- UserSettings.vue --> |
那么你就可以通过这个路由配置来实现上面的布局:
1 | { |
1 | // router.js |
重定向和别名
重定向
重定向也是通过 routes
配置来完成,下面例子是从 /home
重定向到 /
:
1 | const routes = [{ path: '/home', redirect: '/' }] |
重定向的目标也可以是一个命名的路由:
1 | const routes = [{ path: '/home', redirect: { name: 'homepage' } }] |
甚至是一个方法,动态返回重定向目标:
1 | const routes = [ |
请注意,导航守卫并没有应用在跳转路由上,而仅仅应用在其目标上。在上面的例子中,在 /home
路由中添加 beforeEnter
守卫不会有任何效果。
在写 redirect 的时候,可以省略 component
配置,因为它从来没有被直接访问过,所以没有组件要渲染。唯一的例外是嵌套路由:如果一个路由记录有 children
和 redirect
属性,它也应该有 component
属性。
相对重定向
也可以重定向到相对位置:
1 | const routes = [ |
别名
重定向是指当用户访问 /home
时,URL 会被 /
替换,然后匹配成 /
。那么什么是别名呢?
将 /
别名为 /home
,意味着当用户访问 /home
时,URL 仍然是 /home
,但会被匹配为用户正在访问 /
。
上面对应的路由配置为:
1 | const routes = [{ path: '/', component: Homepage, alias: '/home' }] |
通过别名,你可以自由地将 UI 结构映射到一个任意的 URL,而不受配置的嵌套结构的限制。使别名以 /
开头,以使嵌套路径中的路径成为绝对路径。你甚至可以将两者结合起来,用一个数组提供多个别名:
1 | const routes = [ |
如果你的路由有参数,请确保在任何绝对别名中包含它们:
1 | const routes = [ |
路由组件传参
将 props
传递给路由组件
在你的组件中使用 $route
会与路由紧密耦合,这限制了组件的灵活性,因为它只能用于特定的 URL。虽然这不一定是件坏事,但我们可以通过 props
配置来解除这种行为:
我们可以将下面的代码
1 | const User = { |
替换成
1 | const User = { |
这允许你在任何地方使用该组件,使得该组件更容易重用和测试。
布尔模式
当 props
设置为 true
时,route.params
将被设置为组件的 props
。
具体看上面代码路由设置。
命名视图
对于有命名视图的路由,你必须为每个命名视图定义 props
配置:
1 | const routes = [ |
对象模式
当 props
是一个对象时,它将原样设置为组件 props
。当 props
是静态的时候很有用。
1 | const routes = [ |
函数模式
你可以创建一个返回 props 的函数。这允许你将参数转换为其他类型,将静态值与基于路由的值相结合等等。
1 | const routes = [ |
URL /search?q=vue
将传递 {query: 'vue'}
作为 props
传给 SearchUser
组件。
请尽可能保持 props
函数为无状态的,因为它只会在路由发生变化时起作用。如果你需要状态来定义 props
,请使用包装组件,这样 vue
才可以对状态变化做出反应。
不同的历史记录模式
在创建路由器实例时,history
配置允许我们在不同的历史模式中进行选择。
Hash 模式
hash 模式是用 createWebHashHistory()
创建的:
1 | import { createRouter, createWebHashHistory } from 'vue-router' |
它在内部传递的实际 URL 之前使用了一个哈希字符(#
)。由于这部分 URL 从未被发送到服务器,所以它不需要在服务器层面上进行任何特殊处理。不过,它在 SEO 中确实有不好的影响。如果你担心这个问题,可以使用 HTML5
模式。
HTML5 模式
用 createWebHistory()
创建 HTML5 模式,推荐使用这个模式:
1 | import { createRouter, createWebHistory } from 'vue-router' |
当使用这种历史模式时,URL 会看起来很 “正常”,例如 https://example.com/user/id
。漂亮!
不过,问题来了。由于我们的应用是一个单页的客户端应用,如果没有适当的服务器配置,用户在浏览器中直接访问 https://example.com/user/id
,就会得到一个 404 错误。这就尴尬了。
不用担心:要解决这个问题,你需要做的就是在你的服务器上添加一个简单的回退路由。如果 URL 不匹配任何静态资源,它应提供与你的应用程序中的 index.html
相同的页面。漂亮依旧!
进阶
导航守卫
正如其名,vue-router 提供的导航守卫主要用来通过跳转或取消的方式守卫导航。这里有很多方式植入路由导航中:全局的,单个路由独享的,或者组件级的。
全局前置守卫
router.beforeEach(to, from, next)
你可以使用 router.beforeEach
注册一个全局前置守卫
1 | const router = createRouter({ ... }) |
当一个导航触发时,全局前置守卫按照创建顺序调用。守卫是异步解析执行,此时导航在所有守卫 resolve
完之前一直处于等待中.
每个守卫方法接收两个参数:
- to: 即将要进入的目标
- from: 当前导航正要离开的路由
可以返回的值如下:
false
: 取消当前的导航。如果浏览器的 URL 改变了(可能是用户手动或者浏览器后退按钮),那么 URL 地址会重置到from
路由对应的地址。- 一个路由地址: 通过一个路由地址跳转到一个不同的地址,就像你调用
router.push()
一样,你可以设置诸如replace: true
或name: 'home'
之类的配置。当前的导航被中断,然后进行一个新的导航,就和from
一样。
1 | router.beforeEach(async (to, from) => { |
如果遇到了意料之外的情况,可能会抛出一个 Error
。这会取消导航并且调用 router.onError()
注册过的回调。
如果什么都没有,undefined
或返回 true
,则导航是有效的,并调用下一个导航守卫
以上所有都同 async
函数 和 Promise
工作方式一样:
1 | router.beforeEach(async (to, from) => { |
next()
在之前的 Vue Router
版本中,也是可以使用 第三个参数 next
的。这是一个常见的错误来源,可以通过 RFC 来消除错误。然而,它仍然是被支持的,这意味着你可以向任何导航守卫传递第三个参数。在这种情况下,确保 next
在任何给定的导航守卫中都被严格调用一次。它可以出现多于一次,但是只能在所有的逻辑路径都不重叠的情况下,否则钩子永远都不会被解析或报错。这里有一个在用户未能验证身份时重定向到/login的错误用例:
1 | // BAD |
下面是正确的版本:
1 | // GOOD |
全局解析守卫
router.beforeResolve
你可以用 router.beforeResolve
注册一个全局守卫。这和 router.beforeEach
类似,因为它在每次导航时都会触发,不同的是,解析守卫刚好会在导航被确认之前、所有组件内守卫和异步路由组件被解析之后调用。这里有一个例子,确保用户可以访问自定义 meta 属性 requiresCamera
的路由:
1 | router.beforeResolve(async to => { |
router.beforeResolve
是获取数据或执行任何其他操作(如果用户无法进入页面时你希望避免执行的操作)的理想位置。
全局后置钩子
router.afterEach
你也可以注册全局后置钩子,然而和守卫不同的是,这些钩子不会接受 next
函数也不会改变导航本身:
1 | router.afterEach((to, from) => { |
路由独享的守卫
beforeEnter
你可以通过 beforeEnter
守卫来为特定的路由添加守卫:
1 | const routes = [ |
beforeEnter
守卫 只在进入路由时触发,不会在 params
、query
或 hash
改变时触发。例如,从 /users/2
进入到 /users/3
或者从 /users/2#info
进入到 /users/2#projects
。它们只有在 从一个不同的 路由导航时,才会被触发。
你也可以将一个函数数组传递给 beforeEnter
,这在为不同的路由重用守卫时很有用:
1 | function removeQueryParams(to) { |
请注意,你也可以通过使用路径 meta 字段和全局导航守卫来实现类似的行为。
组件内的守卫
最后,你可以在路由组件内直接定义路由导航守卫(传递给路由配置的)
beforeRouteEnter
beforeRouteUpdate
beforeRouteLeave
1 | const UserDetails = { |
离开守卫 通常用来预防用户在还未保存修改前突然离开。该导航可以通过返回 false 来取消。
1 | beforeRouteLeave (to, from) { |
使用组合 API
如果你正在使用组合 API 和 setup 函数来编写组件,你可以通过 onBeforeRouteUpdate
和 onBeforeRouteLeave
分别添加 update 和 leave 守卫。 请参考组合 API 部分以获得更多细节。
完整的导航解析流程
- 导航被触发。
- 在失活的组件里调用
beforeRouteLeave
守卫。 - 调用全局的
beforeEach
守卫。 - 在重用的组件里调用
beforeRouteUpdate
守卫(2.2+)。 - 在路由配置里调用
beforeEnter
。 - 解析异步路由组件。
- 在被激活的组件里调用
beforeRouteEnter
。 - 调用全局的
beforeResolve
守卫(2.5+)。 - 导航被确认。
- 调用全局的
afterEach
钩子。 - 触发
DOM
更新。 - 调用
beforeRouteEnter
守卫中传给next
的回调函数,创建好的组件实例会作为回调函数的参数传入。
路由元信息
有时,你可能希望将任意信息附加到路由上,如过渡名称、谁可以访问路由等。这些事情可以通过接收属性对象的 meta
属性来实现,并且它可以在路由地址和导航守卫上都被访问到。定义路由的时候你可以这样配置 meta
字段:
1 | const routes = [ |
那么如何访问这个 meta
字段呢?
首先,我们称呼 routes
配置中的每个路由对象为 路由记录。路由记录可以是嵌套的,因此,当一个路由匹配成功后,它可能匹配多个路由记录。
例如,根据上面的路由配置,/posts/new
这个 URL 将会匹配父路由记录 (path: '/posts'
) 以及子路由记录 (path: 'new'
)。
一个路由匹配到的所有路由记录会暴露为 $route
对象(还有在导航守卫中的路由对象)的 $route.matched
数组。我们需要遍历这个数组来检查路由记录中的 meta
字段,但是 Vue Router
还为你提供了一个 $route.meta
方法,它是一个非递归合并所有 meta
字段的(从父字段到子字段)的方法。这意味着你可以简单地写
1 | router.beforeEach((to, from) => { |
TypeScript
可以通过扩展 RouteMeta
接口来输入 meta
字段:
1 | // typings.d.ts or router.ts |
数据获取
有时候,进入某个路由后,需要从服务器获取数据。例如,在渲染用户信息时,你需要从服务器获取用户的数据。我们可以通过两种方式来实现:
导航完成之后获取:先完成导航,然后在接下来的组件生命周期钩子中获取数据。在数据获取期间显示“加载中”之类的指示。
导航完成之前获取:导航完成前,在路由进入的守卫中获取数据,在数据获取成功后执行导航。
从技术角度讲,两种方式都不错 —— 就看你想要的用户体验是哪种。
导航完成后获取数据
当你使用这种方式时,我们会马上导航和渲染组件,然后在组件的 created
钩子中获取数据。这让我们有机会在数据获取期间展示一个 loading
状态,还可以在不同视图间展示不同的 loading
状态。
假设我们有一个 Post
组件,需要基于 $route.params.id
获取文章数据:
1 | <template> |
1 | export default { |
导航完成之前获取
通过这种方式,我们在导航转入新的路由前获取数据。我们可以在接下来的组件的 beforeRouteEnter
守卫中获取数据,当数据获取成功后只调用 next
方法:
1 | export default { |
在为后面的视图获取数据时,用户会停留在当前的界面,因此建议在数据获取期间,显示一些进度条或者别的指示。如果数据获取失败,同样有必要展示一些全局的错误提醒。
组合式 API
Vue Router 和 组合式 API
引入 setup
和 Vue 的组合式 API
,开辟了新的可能性,但要想充分发挥 Vue Router
的潜力,我们需要使用一些新的函数来代替访问 this
和组件内导航守卫。
在 setup
中访问路由和当前路由
因为我们在 setup
里面没有访问 this
,所以我们不能再直接访问 this.$router
或 this.$route
。作为替代,我们使用 useRouter
和 useRoute
函数:
1 | import { useRouter, useRoute } from 'vue-router' |
route
对象是一个响应式对象,所以它的任何属性都可以被监听,但你应该避免监听整个 route
对象。在大多数情况下,你应该直接监听你期望改变的参数。
1 | import { useRoute } from 'vue-router' |
请注意,在模板中我们仍然可以访问
$router
和$route
,所以不需要在setup
中返回router
或route
。
导航守卫
虽然你仍然可以通过 setup
函数来使用组件内的导航守卫,但 Vue Router
将更新和离开守卫作为 组合式 API 函数公开:
1 | import { onBeforeRouteLeave, onBeforeRouteUpdate } from 'vue-router' |
组合式 API 守卫也可以用在任何由 <router-view>
渲染的组件中,它们不必像组件内守卫那样直接用在路由组件上。
useLink
Vue Router 将 RouterLink
的内部行为作为一个组合式函数 (composable) 公开。它接收一个类似 RouterLink
所有 prop 的响应式对象,并暴露底层属性来构建你自己的 RouterLink
组件或生成自定义链接:
1 | import { RouterLink, useLink } from 'vue-router' |
注意在 RouterLink 的 v-slot 中可以访问与 useLink 组合式函数相同的属性。
过渡动效
想要在你的路径组件上使用转场,并对导航进行动画处理,你需要使用 v-slot API:
Transition 的 API 在这里同样适用。
单个路由的过渡
上面的用法会对所有的路由使用相同的过渡。如果你想让每个路由的组件有不同的过渡,你可以将元信息(meta
)和动态的 name
结合在一起,放在<transition>
上:
1 | const routes = [ |
1 | <router-view v-slot="{ Component, route }"> |
基于路由的动态过渡
也可以根据目标路由和当前路由之间的关系,动态地确定使用的过渡。使用和刚才非常相似的片段:
1 | <!-- 使用动态过渡名称 --> |
我们可以添加一个 after
navigation
hook
,根据路径的深度动态添加信息到 meta
字段。
1 | router.afterEach((to, from) => { |
强制在复用的视图之间进行过渡
Vue 可能会自动复用看起来相似的组件,从而忽略了任何过渡。幸运的是,可以添加一个 key
属性来强制过渡。这也允许你在相同路由上使用不同的参数触发过渡:
1 | <router-view v-slot="{ Component, route }"> |
滚动行为
使用前端路由,当切换到新路由时,想要页面滚到顶部,或者是保持原先的滚动位置,就像重新加载页面那样。 vue-router 能做到,而且更好,它让你可以自定义路由切换时页面如何滚动。
注意: 这个功能只在支持 history.pushState 的浏览器中可用。
当创建一个 Router 实例,你可以提供一个 scrollBehavior
方法:
1 | const router = createRouter({ |
scrollBehavior
函数接收 to
和 from
路由对象。第三个参数 savedPosition
,只有当这是一个 popstate
导航时才可用(由浏览器的后退/前进按钮触发)。
该函数可以返回一个 ScrollToOptions 位置对象:1
2
3
4
5
6const router = createRouter({
scrollBehavior(to, from, savedPosition) {
// 始终滚动到顶部
return { top: 0 }
},
})
你也可以通过 el
传递一个 CSS
选择器或一个 DOM
元素。在这种情况下,top
和 left
将被视为该元素的相对偏移量。
1 | const router = createRouter({ |
如果返回一个 false
的值,或者是一个空对象
,那么不会发生滚动。
返回 savedPosition
,在按下 后退/前进 按钮时,就会像浏览器的原生表现那样:1
2
3
4
5
6
7
8
9onst router = createRouter({
scrollBehavior(to, from, savedPosition) {
if (savedPosition) {
return savedPosition
} else {
return { top: 0 }
}
},
})
如果你要模拟 “滚动到锚点” 的行为:
1 | const router = createRouter({ |
如果你的浏览器支持滚动行为,你可以让它变得更流畅:
1 | const router = createRouter({ |
延迟滚动
有时候,我们需要在页面中滚动之前稍作等待。例如,当处理过渡时,我们希望等待过渡结束后再滚动。要做到这一点,你可以返回一个 Promise,它可以返回所需的位置描述符。下面是一个例子,我们在滚动前等待 500ms:
1 | const router = createRouter({ |
路由懒加载
当打包构建应用时,JavaScript 包会变得非常大,影响页面加载。如果我们能把不同路由对应的组件分割成不同的代码块,然后当路由被访问的时候才加载对应组件,这样就会更加高效。
Vue Router 支持开箱即用的动态导入,这意味着你可以用动态导入代替静态导入:
1 | // 将 |
component
(和 components
) 配置接收一个返回 Promise
组件的函数,Vue Router 只会在第一次进入页面时才会获取这个函数,然后使用缓存数据。这意味着你也可以使用更复杂的函数,只要它们返回一个 Promise
:
1 | const UserDetails = () => |
一般来说,对所有的路由都使用动态导入是个好主意。
注意:
不要在路由中使用异步组件。异步组件仍然可以在路由组件中使用,但路由组件本身就是动态导入的。
如果你使用的是 webpack
之类的打包器,它将自动从代码分割中受益。
如果你使用的是 Babel
,你将需要添加 syntax-dynamic-import
插件,才能使 Babel
正确地解析语法。
把组件按组分块
使用 webpack
有时候我们想把某个路由下的所有组件都打包在同个异步块 (chunk) 中。只需要使用命名 chunk,一个特殊的注释语法来提供 chunk name (需要 Webpack > 2.4):
1 | const UserDetails = () => |
webpack 会将任何一个异步模块与相同的块名称组合到相同的异步块中。
使用 Vite
在Vite中,你可以在 rollupOptions
下定义分块:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17// vite.config.js
export default defineConfig({
build: {
rollupOptions: {
// https://rollupjs.org/guide/en/#outputmanualchunks
output: {
manualChunks: {
'group-user': [
'./src/UserDetails',
'./src/UserDashboard',
'./src/UserProfileEdit',
],
},
},
},
},
})
动态路由
对路由的添加通常是通过 routes
选项来完成的,但是在某些情况下,你可能想在应用程序已经运行的时候添加或删除路由。具有可扩展接口(如Vue CLI UI
)这样的应用程序可以使用它来扩展应用程序。
添加路由
动态路由主要通过两个函数实现。router.addRoute()
和 router.removeRoute()
。它们只注册一个新的路由,也就是说,如果新增加的路由与当前位置相匹配,就需要你用 router.push()
或 router.replace()
来手动导航,才能显示该新路由。我们来看一个例子:
想象一下,只有一个路由的以下路由:
1 | const router = createRouter({ |
进入任何页面,/about
,/store
,或者 /3-tricks-to-improve-your-routing-code
最终都会呈现 Article
组件。如果我们在 /about
上添加一个新的路由:
1 | router.addRoute({ path: '/about', component: About }) |
1 | router.addRoute({ path: '/about', component: About }) |
记住,如果你需要等待新的路由显示,可以使用 await router.replace()
。
在导航守卫中添加路由
如果你决定在导航守卫内部添加或删除路由,你不应该调用 router.replace()
,而是通过返回新的位置来触发重定向:
1 | router.beforeEach(to => { |
上面的例子有两个假设:
第一,新添加的路由记录将与 to
位置相匹配,实际上导致与我们试图访问的位置不同。
第二,hasNecessaryRoute()
在添加新的路由后返回 false
,以避免无限重定向。
因为是在重定向中,所以我们是在替换将要跳转的导航,实际上行为就像之前的例子一样。而在实际场景中,添加路由的行为更有可能发生在导航守卫之外,例如,当一个视图组件挂载时,它会注册新的路由。
删除路由
有几个不同的方法来删除现有的路由:
- 通过添加一个名称冲突的路由。如果添加与现有途径名称相同的途径,会先删除路由,再添加路由:
1 | router.addRoute({ path: '/about', name: 'about', component: About }) |
- 通过调用
router.addRoute()
返回的回调:
1 | const removeRoute = router.addRoute(routeRecord) |
当路由没有名称时,这很有用。
- 通过使用
router.removeRoute()
按名称删除路由:
1 | router.addRoute({ path: '/about', name: 'about', component: About }) |
需要注意的是,如果你想使用这个功能,但又想避免名字的冲突,可以在路由中使用 Symbol
作为名字。
当路由被删除时,所有的别名和子路由也会被同时删除
添加嵌套路由
要将嵌套路由添加到现有的路由中,可以将路由的 name 作为第一个参数传递给 router.addRoute()
,这将有效地添加路由,就像通过 children
添加的一样:
1 | router.addRoute({ name: 'admin', path: '/admin', component: Admin }) |
这等效于:
1 | router.addRoute({ |
查看现有路由
Vue Router 提供了两个功能来查看现有的路由:
router.hasRoute()
:检查路由是否存在。router.getRoutes()
:获取一个包含所有路由记录的数组。