自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

從 Vue2 到 Vue3 ,你必須要掌握的路由差異和使用場景!

開發(fā) 前端
今天把 Vue-Router4 的主要功能跟大家過了一遍,大部分來自官網(wǎng),也有一些來自自己的實踐心得。希望對你的開發(fā)工作有所幫助。

很多兄弟在使用 Vue3 了,但對 Vue3 的路由卻了解的非常少。甚至只知道基本的跳轉(zhuǎn)和參數(shù)獲取,這樣做一些稍微復雜的功能肯定不夠用的。最近就把 Vue3 的路由(Vue-Router4)的版本差異和使用場景整理了一下分享給大家。會的兄弟可以復習一下,不會的兄弟抓緊學起來哦!

路由模式

Vue3 中不再使用 new Router() 創(chuàng)建 router ,而是調(diào)用 createRouter 方法:

import { createRouter } from 'vue-router'

const router = createRouter({
// ...
})

路由模式 mode 配置改為 history ,屬性值調(diào)整為:

  • "history" => createWebHistory()
  • "hash" => createWebHashHistory()
  • "abstract" => createMemoryHistory()
import { createRouter, createWebHistory } from 'vue-router'
// createWebHashHistory 和 createMemoryHistory (SSR相關(guān)) 同理

createRouter({
history: createWebHistory(),
routes: []
})

基礎(chǔ)路徑 base 被作為 createWebHistory 的第一個參數(shù)進行傳遞(其他路由模式也是一樣):

import { createRouter, createWebHistory } from 'vue-router'
createRouter({
history: createWebHistory('/base-url/'),
routes: []
})

路由跳轉(zhuǎn)

使用組件跳轉(zhuǎn),方式還是和 Vue2 一樣:

<RouterLink to="/user">User</RouterLink>
<RouterLink :to="{ path: '/user', query: { username: 'Jack' } }">User</RouterLink>
<RouterLink :to="{ name: 'user', params: { username: 'Tom' } }">User</RouterLink>

當然,最常見的還是編程式導航,這時候需要引入 useRouter 方法:

import { useRouter } from 'vue-router'

const router = useRouter()

// 字符串路徑
router.push('/user')

// 帶有路徑的對象
router.push({ path: '/user', query: { username: 'Jack' } })
router.push({ path: '/user', hash: '#team' })

// 帶有命名的對象
router.push({ name: 'user', query: { username: 'Jack' } })
router.push({ name: 'user', params: { username: 'Tom' } })
router.push({ name: 'user', hash: '#team' })

注意:參數(shù) params 不能和 path 一起使用。RouterLink 組件 to 屬性與 router.push() 接受的參數(shù)相同,兩者的規(guī)則也完全相同。

導航守衛(wèi)

全局前置守衛(wèi)

全局前置守衛(wèi)通常用來做權(quán)限控制,使用 router.beforeEach 即可添加:

const router = createRouter({ ... })

router.beforeEach((to, from) => {
// ...
// 返回 false 以取消導航
return false
})

每個守衛(wèi)方法接收兩個參數(shù):

  • to:即將進入的目標路由
  • from:當前正要離開的路由

可以返回的值如下:

  • false:取消當前的導航。
  • true 或 undefined,調(diào)用下一個守衛(wèi)。
  • 一個路由地址:字符串或?qū)ο蟆1硎局袛喈斍皩Ш?,進行一個新的導航。
router.beforeEach(async (to, from) => {
// 檢查用戶是否已登錄,并且避免無限重定向
if (!isAuthenticated && to.name !== 'Login') {
return { name: 'Login' } // 將用戶重定向到登錄頁面
}
})

在之前的 Vue Router 版本中,也是可以使用第三個參數(shù) next 的。目前,它仍然是被支持的,這意味著你可以向任何導航守衛(wèi)傳遞第三個參數(shù)。在這種情況下,要確保 next 在導航守衛(wèi)中只被調(diào)用一次。

全局解析守衛(wèi)

router.beforeResolve 用法和 router.beforeEach 類似。它是在導航被確認之前,所有組件內(nèi)守衛(wèi)和異步路由組件被解析之后被調(diào)用。下面這個例子,確保用戶可以訪問自定義 meta 屬性:

router.beforeResolve(async to => {
if (to.meta.requiresCamera) {
try {
await askForCameraPermission()
} catch (error) {
if (error instanceof NotAllowedError) {
// ... 處理錯誤,然后取消導航
return false
} else {
// 意料之外的錯誤,取消導航并把錯誤傳給全局處理器
throw error
}
}
}
})

router.beforeResolve 是獲取數(shù)據(jù)或執(zhí)行任何其他操作(進入所有頁面后都執(zhí)行的操作)的理想位置。

全局后置鉤子

和守衛(wèi)不同的是,全局后置鉤子不接受 next 函數(shù),也不能跳轉(zhuǎn)到其他的路由地址:

router.afterEach((to, from) => {
sendToAnalytics(to.fullPath)
})

但它可以接收 failure 作為第三個參數(shù):

router.afterEach((to, from, failure) => {
if (!failure) sendToAnalytics(to.fullPath)
})

router.afterEach 對于訪問分析、更改頁面標題、聲明頁面等輔助功能都很有幫助。

路由獨享的守衛(wèi)

我們可以直接在路由配置上定義 beforeEnter 守衛(wèi):

const routes = [
{
path: '/users/:id',
component: UserDetails,
beforeEnter: (to, from) => {
// 取消導航
return false
},
},
]

beforeEnter 守衛(wèi)只在進入路由時觸發(fā),不會在 params、query 或 hash 改變時觸發(fā)。例如,從 /users/2 進入到 /users/3 或者從 /users/2#info 進入到 /users/2#projects 不會觸發(fā)。

我們也可以將一個函數(shù)數(shù)組傳遞給 beforeEnter,這在為不同的路由重用守衛(wèi)時很有用:

// 清除 query 參數(shù)
function removeQueryParams(to) {
if (Object.keys(to.query).length)
return { path: to.path, query: {}, hash: to.hash }
}
// 清除 hash 值
function removeHash(to) {
if (to.hash) return { path: to.path, query: to.query, hash: '' }
}

const routes = [
{
path: '/users/:id',
component: UserDetails,
beforeEnter: [removeQueryParams, removeHash]
},
{
path: '/about',
component: UserDetails,
beforeEnter: [removeQueryParams]
}
]

當然,你也可以通過使用路由的 meta 屬性和 全局導航守衛(wèi) 來實現(xiàn)以上功能。

組件內(nèi)的守衛(wèi)

使用聲明式 API ,你可以為組件添加以下守衛(wèi):

  • beforeRouteEnter
  • beforeRouteUpdate
  • beforeRouteLeave

beforeRouteEnter 守衛(wèi)不能訪問 this,因為此時組件還沒有被創(chuàng)建。你可以通過傳一個回調(diào)給 next 來訪問組件實例。在導航被確認的時候執(zhí)行回調(diào),并且把組件實例作為回調(diào)方法的參數(shù):

beforeRouteEnter (to, from, next) {
next(vm => {
// 通過 `vm` 訪問組件實例
})
}

注意:beforeRouteEnter 是支持 next 傳遞回調(diào)函數(shù)的唯一守衛(wèi)。

beforeRouteUpdate 在當前路由改變,但是該組件被復用時調(diào)用。比如,對于一個帶有動態(tài)參數(shù)的路徑 /users/:id,在 /users/1 和 /users/2 之間跳轉(zhuǎn)的時候被調(diào)用。因為這種情況發(fā)生的時候,組件已經(jīng)掛載好了,導航守衛(wèi)可以訪問組件實例 this。

beforeRouteUpdate (to, from) {
// 可以使用 this
this.name = to.params.name
}

beforeRouteLeave 通常用來預防用戶在還未保存修改前突然離開。該守衛(wèi)可以通過返回 false 來取消導航。

beforeRouteLeave (to, from) {
const answer = window.confirm('Do you really want to leave? you have unsaved changes!')
// 取消導航并停留在當前頁面
if (!answer) return false
}

使用組合式 API,你可以為組件添加 onBeforeRouteUpdate 、onBeforeRouteLeave 導航守衛(wèi):

<script setup lang="ts">
import { ref } from 'vue'
import { onBeforeRouteLeave, onBeforeRouteUpdate } from 'vue-router'

const userData = ref()

onBeforeRouteUpdate(async (to, from) => {
//僅當 id 更改時才獲取用戶信息
if (to.params.id !== from.params.id) {
userData.value = await fetchUser(to.params.id)
}
})

onBeforeRouteLeave((to, from) => {
const answer = window.confirm('Do you really want to leave? you have unsaved changes!')
// 取消導航并停留在當前頁面
if (!answer) return false
})
</script>

注意:由于 setup 函數(shù)調(diào)用時機的問題,使用組合式 API 不存在 onBeforeRouteEnter。

路由組件傳參

當我們獲取路由參數(shù)時,通常在模板中使用 $route ,在邏輯中調(diào)用 useRoute() 方法,如:

<template>
<div>User {{ $route.params.id }}</div>
</template>

<script setup lang="ts">
import { useRoute } from 'vue-router'
const route = useRoute()
console.log(route.params.id)
</script>

以上方法比較麻煩,而且與路由緊密耦合,不利于組件封裝。我們可以在創(chuàng)建路由時通過 props 配置來解除這種行為:

const routes = [
{
path: '/user/:id',
name: 'user',
component: User,
props: true
}
]

此時 route.params 將直接被設置為組件的 props,這樣組件就和路由參數(shù)解耦了:

<template>
<div>User {{ id }}</div>
</template>

<script setup lang="ts">
const props = defineProps<{
id: string
}>()
console.log(props.id)
</script>

布爾模式

當 props 設置為 true 時,route.params 將被設置為組件的 props。

命名視圖

對于有命名視圖的路由,你必須為每個命名視圖定義 props 配置:

const routes = [
{
path: '/user/:id',
components: { default: User, sidebar: Sidebar },
props: { default: true, sidebar: false }
}
]

對象模式

當 props 是一個對象時,它會將此對象設置為組件 props 。當 props 是靜態(tài)的時候很有用。

const routes = [
{
path: '/user',
component: User,
props: { newsletterPopup: false }
}
]

函數(shù)模式

我們也可以創(chuàng)建一個返回 props 的函數(shù)。這允許你將參數(shù)轉(zhuǎn)換為其他類型:

const routes = [
{
path: '/user',
component: User,
props: route => ({ id: route.query.userId })
}
]

如 /user?userId=123 參數(shù)會被轉(zhuǎn)為 { id: '123' } 作為 props 傳給 User 組件。

滾動行為

我們可以通過 vue-router 自定義路由切換時頁面如何滾動。比如,當跳轉(zhuǎn)到新路由時,頁面滾動到某個位置;切換路由時頁面回到之前的滾動位置。

當創(chuàng)建路由實例時,我們只需要提供一個 scrollBehavior 方法:

const router = createRouter({
history: createWebHashHistory(),
routes: [...],
scrollBehavior (to, from, savedPosition) {
// return 期望滾動到哪個的位置
}
})

scrollBehavior 函數(shù)接收 to from 路由對象。第三個參數(shù) savedPosition,只有當這是一個 popstate 導航時才可用(點擊瀏覽器的后退/前進按鈕,或者調(diào)用 router.go() 方法)。

滾動到固定距離

該函數(shù)可以返回一個 ScrollToOptions 位置對象:

const router = createRouter({
scrollBehavior(to, from, savedPosition) {
// 始終滾動到頂部
return { top: 0 }
},
})

滾動到元素位置

也可以通過 el 傳遞一個 CSS 選擇器或一個 DOM 元素。在這種情況下,top 和 left 將被視為該元素的相對偏移量。

const router = createRouter({
scrollBehavior(to, from, savedPosition) {
// 始終在元素 #main 上方滾動 10px
return {
// 也可以這么寫
// el: document.getElementById('main'),
el: '#main',
top: -10,
}
},
})

滾動到錨點位置

還可以模擬 “滾動到錨點” :

const router = createRouter({
scrollBehavior(to, from, savedPosition) {
if (to.hash) {
return {
el: to.hash,
}
}
},
})

滾動到之前的位置

返回 savedPosition,在按下瀏覽器 后退/前進 按鈕,或者調(diào)用 router.go() 方法時,頁面會回到之前的滾動位置:

const router = createRouter({
scrollBehavior(to, from, savedPosition) {
if (savedPosition) {
return savedPosition
} else {
return { top: 0 }
}
},
})

提示:如果返回一個 falsy 的值,或者是一個空對象,則不會發(fā)生滾動。我們還可以在返回的對象中添加 behavior: 'smooth' ,讓滾動更加絲滑。

延遲滾動

有時候,我們不希望立即執(zhí)行滾動行為。例如,當頁面做了過渡動效,我們希望過渡結(jié)束后再執(zhí)行滾動。要做到這一點,我們可以返回一個 Promise :

const router = createRouter({
scrollBehavior(to, from, savedPosition) {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve({ left: 0, top: 0 })
}, 500)
})
}
})

過渡動效

基本用法

如果想要在路由組件上使用轉(zhuǎn)場,對導航進行動畫處理,我可以使用 v-slot 結(jié)合 Animete.css 來實現(xiàn):

<RouterView v-slot="{ Component }">
<transition enter-active-class="animate__animated animate__fadeIn">
<component :is="Component" />
</transition>
</RouterView>

單個路由的過渡

上面的用法會對所有的路由使用相同的過渡。如果你想讓每個路由的組件有不同的過渡,可以將 元信息 和動態(tài)的 enter-active-class 結(jié)合在一起,放在<transition> 上:

const routes = [
{
path: '/home',
component: Home,
meta: { transition: 'animate__fadeIn' },
},
{
path: '/user',
component: User,
meta: { transition: 'animate__bounceIn' },
},
]
<RouterView v-slot="{ Component }">
<transition :enter-active-class="`animate__animated ${$route.meta.transition}`">
<component :is="Component" />
</transition>
</RouterView>

復用的組件之前進行過渡

const routes = [
{
path: '/user/:id',
component: User,
meta: { transition: 'animate__bounceIn' },
},
]

定義以上路由,當從 /user/123 切換到 /user/456 時是沒有任何過渡效果的。這時候我們可以添加一個 key 屬性來強制進行過渡,key 值只要不同就行了。說白了就是讓 Dom 不要被復用,和 v-for 的 key 屬性原理剛好相反。

<router-view v-slot="{ Component, route }">
<transition :enter-active-class="`animate__animated ${$route.meta.transition}`">
<component :is="Component" :key="route.path" />
</transition>
</router-view>

動態(tài)路由

添加路由

當我們做用戶權(quán)限的時候,添加路由非常有用??梢允褂?nbsp;router.addRoute() 來添加一個路由:

router.addRoute({ path: '/about', name: 'about', component: About })

注意:跟之前版本不同的是,路由只能一個一個添加,不能批量添加。

刪除路由

以下幾個方法都可以刪除路由:

1、通過使用 router.removeRoute() 按名稱刪除路由:

router.addRoute({ path: '/about', name: 'about', component: About })
// 刪除路由
router.removeRoute('about')

2、通過添加一個名稱相同的路由,替換掉之前的路由:

router.addRoute({ path: '/about', name: 'about', component: About })
// 這將會刪除之前已經(jīng)添加的路由,因為他們具有相同的名字且名字必須是唯一的
router.addRoute({ path: '/other', name: 'about', component: Other })

3、通過調(diào)用 router.addRoute() 返回的回調(diào)函數(shù):

const removeRoute = router.addRoute(routeRecord)
removeRoute() // 刪除路由如果存在的話

當路由沒有名稱時,這種方法非常有用。

添加嵌套路由

要將嵌套路由添加到現(xiàn)有的路由中,可以將路由的 name 作為第一個參數(shù)傳遞給 router.addRoute() ,這和通過 children 添加的效果一樣:

router.addRoute({ name: 'admin', path: '/admin', component: Admin })
// 添加嵌套路由
router.addRoute('admin', { path: 'settings', component: AdminSettings })

這相當于:

router.addRoute({
name: 'admin',
path: '/admin',
component: Admin,
children: [{ path: 'settings', component: AdminSettings }]
})

小結(jié)

今天把 Vue-Router4 的主要功能跟大家過了一遍,大部分來自官網(wǎng),也有一些來自自己的實踐心得。希望對你的開發(fā)工作有所幫助。當然這并不是 Vue-Router4 的所有內(nèi)容,比如還有路由匹配、重定向和別名等等,大家可以自行在官網(wǎng)查看。后面我會分享更多 Vue3 相關(guān)的干貨,歡迎大家關(guān)注我,關(guān)注我的專欄。

責任編輯:姜華 來源: 今日頭條
相關(guān)推薦

2022-07-18 10:43:12

項目TienChinJava

2022-12-09 09:39:20

Vue3Vue2

2020-03-25 18:23:07

Vue2Vue3組件

2023-04-28 08:35:22

Vue 3Vue 2

2022-06-21 12:09:18

Vue差異

2021-03-22 10:05:25

開源技術(shù) 項目

2022-06-29 16:59:21

Vue3Vue2面試

2021-06-26 06:29:14

Vue 2Vue 3開發(fā)

2021-05-06 07:40:56

Vue3 Vue2 組件

2024-10-14 09:34:39

vue3通信emit

2021-12-08 09:09:33

Vue 3 Computed Vue2

2021-12-01 08:11:44

Vue3 插件Vue應用

2023-11-28 09:03:59

Vue.jsJavaScript

2019-06-20 17:39:12

Android啟動優(yōu)化

2021-11-26 05:59:31

Vue3 插件Vue應用

2024-11-06 10:16:22

2021-09-26 00:24:58

開發(fā)項目TypeScript

2022-07-26 01:06:18

Vue3自定義指令

2021-11-30 08:19:43

Vue3 插件Vue應用

2024-03-01 08:38:34

WebpackVue2sass
點贊
收藏

51CTO技術(shù)棧公眾號