偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

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

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

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

路由模式

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 的第一個(gè)參數(shù)進(jìn)行傳遞(其他路由模式也是一樣):

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>

當(dāng)然,最常見(jiàn)的還是編程式導(dǎo)航,這時(shí)候需要引入 useRouter 方法:

import { useRouter } from 'vue-router'

const router = useRouter()

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

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

// 帶有命名的對(duì)象
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ī)則也完全相同。

導(dǎo)航守衛(wèi)

全局前置守衛(wèi)

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

const router = createRouter({ ... })

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

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

  • to:即將進(jìn)入的目標(biāo)路由
  • from:當(dāng)前正要離開(kāi)的路由

可以返回的值如下:

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

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

全局解析守衛(wèi)

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

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

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

全局后置鉤子

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

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

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

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

router.afterEach 對(duì)于訪問(wèn)分析、更改頁(yè)面標(biāo)題、聲明頁(yè)面等輔助功能都很有幫助。

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

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

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

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

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

// 清除 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]
}
]

當(dāng)然,你也可以通過(guò)使用路由的 meta 屬性和 全局導(dǎo)航守衛(wèi) 來(lái)實(shí)現(xiàn)以上功能。

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

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

  • beforeRouteEnter
  • beforeRouteUpdate
  • beforeRouteLeave

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

beforeRouteEnter (to, from, next) {
next(vm => {
// 通過(guò) `vm` 訪問(wèn)組件實(shí)例
})
}

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

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

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

beforeRouteLeave 通常用來(lái)預(yù)防用戶在還未保存修改前突然離開(kāi)。該守衛(wèi)可以通過(guò)返回 false 來(lái)取消導(dǎo)航。

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

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

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

const userData = ref()

onBeforeRouteUpdate(async (to, from) => {
//僅當(dāng) id 更改時(shí)才獲取用戶信息
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!')
// 取消導(dǎo)航并停留在當(dāng)前頁(yè)面
if (!answer) return false
})
</script>

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

路由組件傳參

當(dāng)我們獲取路由參數(shù)時(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)建路由時(shí)通過(guò) props 配置來(lái)解除這種行為:

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

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

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

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

布爾模式

當(dāng) props 設(shè)置為 true 時(shí),route.params 將被設(shè)置為組件的 props。

命名視圖

對(duì)于有命名視圖的路由,你必須為每個(gè)命名視圖定義 props 配置:

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

對(duì)象模式

當(dāng) props 是一個(gè)對(duì)象時(shí),它會(huì)將此對(duì)象設(shè)置為組件 props 。當(dāng) props 是靜態(tài)的時(shí)候很有用。

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

函數(shù)模式

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

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

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

滾動(dòng)行為

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

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

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

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

滾動(dòng)到固定距離

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

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

滾動(dòng)到元素位置

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

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

滾動(dòng)到錨點(diǎn)位置

還可以模擬 “滾動(dòng)到錨點(diǎn)” :

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

滾動(dòng)到之前的位置

返回 savedPosition,在按下瀏覽器 后退/前進(jìn) 按鈕,或者調(diào)用 router.go() 方法時(shí),頁(yè)面會(huì)回到之前的滾動(dòng)位置:

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

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

延遲滾動(dòng)

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

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

過(guò)渡動(dòng)效

基本用法

如果想要在路由組件上使用轉(zhuǎn)場(chǎng),對(duì)導(dǎo)航進(jìn)行動(dòng)畫(huà)處理,我可以使用 v-slot 結(jié)合 Animete.css 來(lái)實(shí)現(xiàn):

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

單個(gè)路由的過(guò)渡

上面的用法會(huì)對(duì)所有的路由使用相同的過(guò)渡。如果你想讓每個(gè)路由的組件有不同的過(guò)渡,可以將 元信息 和動(dòng)態(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>

復(fù)用的組件之前進(jìn)行過(guò)渡

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

定義以上路由,當(dāng)從 /user/123 切換到 /user/456 時(shí)是沒(méi)有任何過(guò)渡效果的。這時(shí)候我們可以添加一個(gè) key 屬性來(lái)強(qiáng)制進(jìn)行過(guò)渡,key 值只要不同就行了。說(shuō)白了就是讓 Dom 不要被復(fù)用,和 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>

動(dòng)態(tài)路由

添加路由

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

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

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

刪除路由

以下幾個(gè)方法都可以刪除路由:

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

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

2、通過(guò)添加一個(gè)名稱相同的路由,替換掉之前的路由:

router.addRoute({ path: '/about', name: 'about', component: About })
// 這將會(huì)刪除之前已經(jīng)添加的路由,因?yàn)樗麄兙哂邢嗤拿智颐直仨毷俏ㄒ坏?/span>
router.addRoute({ path: '/other', name: 'about', component: Other })

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

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

當(dāng)路由沒(méi)有名稱時(shí),這種方法非常有用。

添加嵌套路由

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

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

這相當(dāng)于:

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

小結(jié)

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

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

2022-07-18 10:43:12

項(xiàng)目TienChinJava

2022-12-09 09:39:20

Vue3Vue2

2023-04-28 08:35:22

Vue 3Vue 2

2020-03-25 18:23:07

Vue2Vue3組件

2022-06-21 12:09:18

Vue差異

2021-03-22 10:05:25

開(kāi)源技術(shù) 項(xiàng)目

2022-06-29 16:59:21

Vue3Vue2面試

2021-06-26 06:29:14

Vue 2Vue 3開(kāi)發(fā)

2021-05-06 07:40:56

Vue3 Vue2 組件

2019-06-20 17:39:12

Android啟動(dòng)優(yōu)化

2021-12-08 09:09:33

Vue 3 Computed Vue2

2022-07-26 01:06:18

Vue3自定義指令

2024-10-14 09:34:39

vue3通信emit

2021-09-26 00:24:58

開(kāi)發(fā)項(xiàng)目TypeScript

2021-12-01 08:11:44

Vue3 插件Vue應(yīng)用

2021-11-26 05:59:31

Vue3 插件Vue應(yīng)用

2023-11-28 09:03:59

Vue.jsJavaScript

2021-10-21 08:13:11

Springboot

2024-11-06 10:16:22

2022-03-07 11:15:25

Pinia狀態(tài)庫(kù)vue3
點(diǎn)贊
收藏

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