欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue中的路由導航守衛(wèi)導航解析流程

 更新時間:2023年04月20日 08:30:18   作者:itpeilibo  
這篇文章主要介紹了Vue中的路由導航守衛(wèi)導航解析流程,正如其名,vue-router 提供的導航守衛(wèi)主要用來通過跳轉或取消的方式守衛(wèi)導航。這里有很多方式植入路由導航中:全局的,單個路由獨享的,或者組件級的

正如其名,vue-router 提供的導航守衛(wèi)主要用來通過跳轉或取消的方式守衛(wèi)導航。這里有很多方式植入路由導航中:全局的,單個路由獨享的,或者組件級的。

全局前置守衛(wèi)

你可以使用 router.beforeEach 注冊一個全局前置守衛(wèi):

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

當一個導航觸發(fā)時,全局前置守衛(wèi)按照創(chuàng)建順序調用。守衛(wèi)是異步解析執(zhí)行,此時導航在所有守衛(wèi) resolve 完之前一直處于等待中。

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

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

可以返回的值如下:

  • false: 取消當前的導航。如果瀏覽器的 URL 改變了(可能是用戶手動或者瀏覽器后退按鈕),那么 URL 地址會重置到 from 路由對應的地址。
  • 一個路由地址: 通過一個路由地址跳轉到一個不同的地址,就像你調用 router.push() 一樣,你可以設置諸如 replace: truename: 'home' 之類的配置。當前的導航被中斷,然后進行一個新的導航,就和 from 一樣。
 router.beforeEach(async (to, from) => {
   if (
     // 檢查用戶是否已登錄
     !isAuthenticated &&
     // ?? 避免無限重定向
     to.name !== 'Login'
   ) {
     // 將用戶重定向到登錄頁面
     return { name: 'Login' }
   }
 })

如果遇到了意料之外的情況,可能會拋出一個 Error。這會取消導航并且調用 router.onError() 注冊過的回調。

如果什么都沒有,undefined 或返回 true,則導航是有效的,并調用下一個導航守衛(wèi)

以上所有都同 async 函數(shù) 和 Promise 工作方式一樣:

router.beforeEach(async (to, from) => {
  // canUserAccess() 返回 `true` 或 `false`
  const canAccess = await canUserAccess(to)
  if (!canAccess) return '/login'
})

可選的第三個參數(shù) next

在之前的 Vue Router 版本中,也是可以使用 第三個參數(shù) next 的。這是一個常見的錯誤來源,可以通過 RFC 來消除錯誤。然而,它仍然是被支持的,這意味著你可以向任何導航守衛(wèi)傳遞第三個參數(shù)。在這種情況下,確保 next 在任何給定的導航守衛(wèi)中都被嚴格調用一次。它可以出現(xiàn)多于一次,但是只能在所有的邏輯路徑都不重疊的情況下,否則鉤子永遠都不會被解析或報錯。這里有一個在用戶未能驗證身份時重定向到/login的錯誤用例:

// BAD
router.beforeEach((to, from, next) => {
  if (to.name !== 'Login' && !isAuthenticated) next({ name: 'Login' })
  // 如果用戶未能驗證身份,則 `next` 會被調用兩次
  next()
})

下面是正確的版本:

// GOOD
router.beforeEach((to, from, next) => {
  if (to.name !== 'Login' && !isAuthenticated) next({ name: 'Login' })
  else next()
})

全局解析守衛(wèi)

你可以用 router.beforeResolve 注冊一個全局守衛(wèi)。這和 router.beforeEach 類似,因為它在 每次導航時都會觸發(fā),但是確保在導航被確認之前,同時在所有組件內守衛(wèi)和異步路由組件被解析之后,解析守衛(wèi)就被正確調用。這里有一個例子,確保用戶可以訪問自定義 meta屬性 requiresCamera 的路由:

路由元信息

有時,你可能希望將任意信息附加到路由上,如過渡名稱、誰可以訪問路由等。這些事情可以通過接收屬性對象的meta屬性來實現(xiàn),并且它可以在路由地址和導航守衛(wèi)上都被訪問到。定義路由的時候你可以這樣配置 meta 字段:

const routes = [
  {
    path: '/posts',
    component: PostsLayout,
    children: [
      {
        path: 'new',
        component: PostsNew,
        // 只有經過身份驗證的用戶才能創(chuàng)建帖子
        meta: { requiresAuth: true }
      },
      {
        path: ':id',
        component: PostsDetail
        // 任何人都可以閱讀文章
        meta: { requiresAuth: false }
      }
    ]
  }
]

那么如何訪問這個 meta 字段呢?

首先,我們稱呼 routes 配置中的每個路由對象為 路由記錄。路由記錄可以是嵌套的,因此,當一個路由匹配成功后,它可能匹配多個路由記錄。

例如,根據(jù)上面的路由配置,/posts/new 這個 URL 將會匹配父路由記錄 (path: '/posts') 以及子路由記錄 (path: 'new')。

一個路由匹配到的所有路由記錄會暴露為 $route 對象(還有在導航守衛(wèi)中的路由對象)的$route.matched 數(shù)組。我們需要遍歷這個數(shù)組來檢查路由記錄中的 meta 字段,但是 Vue Router 還為你提供了一個 $route.meta 方法,它是一個非遞歸合并所有 meta 字段的(從父字段到子字段)的方法。這意味著你可以簡單地寫

router.beforeEach((to, from) => {
  // 而不是去檢查每條路由記錄
  // to.matched.some(record => record.meta.requiresAuth)
  if (to.meta.requiresAuth && !auth.isLoggedIn()) {
    // 此路由需要授權,請檢查是否已登錄
    // 如果沒有,則重定向到登錄頁面
    return {
      path: '/login',
      // 保存我們所在的位置,以便以后再來
      query: { redirect: to.fullPath },
    }
  }
})

TypeScript

可以通過擴展 RouteMeta 接口來輸入 meta 字段:

// typings.d.ts or router.ts
import 'vue-router'
declare module 'vue-router' {
  interface RouteMeta {
    // 是可選的
    isAdmin?: boolean
    // 每個路由都必須聲明
    requiresAuth: boolean
  }
}

router.beforeResolve

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ù)也不會改變導航本身:

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

它們對于分析、更改頁面標題、聲明頁面等輔助功能以及許多其他事情都很有用。

它們也反映了 navigation failures 作為第三個參數(shù):

js

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

了解更多關于 navigation failures 的信息在它的指南中。

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

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

const routes = [
  {
    path: '/users/:id',
    component: UserDetails,
    beforeEnter: (to, from) => {
      // reject the navigation
      return false
    },
  },
]

beforeEnter 守衛(wèi) 只在進入路由時觸發(fā),不會在 params、queryhash 改變時觸發(fā)。例如,從 /users/2 進入到 /users/3 或者從 /users/2#info 進入到 /users/2#projects。它們只有在 從一個不同的 路由導航時,才會被觸發(fā)。

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

function removeQueryParams(to) {
  if (Object.keys(to.query).length)
    return { path: to.path, query: {}, hash: to.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)類似的行為。

組件內的守衛(wèi)

最后,你可以在路由組件內直接定義路由導航守衛(wèi)(傳遞給路由配置的)

可用的配置 API

你可以為路由組件添加以下配置:

  • beforeRouteEnter
  • beforeRouteUpdate
  • beforeRouteLeave
const UserDetails = {
  template: `...`,
  beforeRouteEnter(to, from) {
    // 在渲染該組件的對應路由被驗證前調用
    // 不能獲取組件實例 `this` !
    // 因為當守衛(wèi)執(zhí)行時,組件實例還沒被創(chuàng)建!
  },
  beforeRouteUpdate(to, from) {
    // 在當前路由改變,但是該組件被復用時調用
    // 舉例來說,對于一個帶有動態(tài)參數(shù)的路徑 `/users/:id`,在 `/users/1` 和 `/users/2` 之間跳轉的時候,
    // 由于會渲染同樣的 `UserDetails` 組件,因此組件實例會被復用。而這個鉤子就會在這個情況下被調用。
    // 因為在這種情況發(fā)生的時候,組件已經掛載好了,導航守衛(wèi)可以訪問組件實例 `this`
  },
  beforeRouteLeave(to, from) {
    // 在導航離開渲染該組件的對應路由時調用
    // 與 `beforeRouteUpdate` 一樣,它可以訪問組件實例 `this`
  },
}

beforeRouteEnter 守衛(wèi) 不能 訪問 this,因為守衛(wèi)在導航確認前被調用,因此即將登場的新組件還沒被創(chuàng)建。

不過,你可以通過傳一個回調給 next 來訪問組件實例。在導航被確認的時候執(zhí)行回調,并且把組件實例作為回調方法的參數(shù):

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

注意 beforeRouteEnter 是支持給 next 傳遞回調的唯一守衛(wèi)。對于 beforeRouteUpdatebeforeRouteLeave 來說,this 已經可用了,所以不支持 傳遞回調,因為沒有必要了:

beforeRouteUpdate (to, from) {
  // just use `this`
  this.name = to.params.name
}

這個 離開守衛(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

如果你正在使用組合 API 和 setup 函數(shù)來編寫組件,你可以通過 onBeforeRouteUpdateonBeforeRouteLeave 分別添加 update 和 leave 守衛(wèi)。 請參考組合 API 部分以獲得更多細節(jié)。

完整的導航解析流程

  • 導航被觸發(fā)。
  • 在失活的組件里調用 beforeRouteLeave 守衛(wèi)。
  • 調用全局的 beforeEach 守衛(wèi)。
  • 在重用的組件里調用 beforeRouteUpdate 守衛(wèi)(2.2+)。
  • 在路由配置里調用 beforeEnter。
  • 解析異步路由組件。
  • 在被激活的組件里調用 beforeRouteEnter。
  • 調用全局的 beforeResolve 守衛(wèi)(2.5+)。
  • 導航被確認。
  • 調用全局的 afterEach 鉤子。
  • 觸發(fā) DOM 更新。
  • 調用 beforeRouteEnter 守衛(wèi)中傳給 next 的回調函數(shù),創(chuàng)建好的組件實例會作為回調函數(shù)的參數(shù)傳入。

到此這篇關于Vue中的路由導航守衛(wèi)導航解析流程的文章就介紹到這了,更多相關Vue路由導航守衛(wèi)內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 手寫實現(xiàn)vue2下拉菜單dropdown組件實例

    手寫實現(xiàn)vue2下拉菜單dropdown組件實例

    這篇文章主要為大家介紹了手寫vue2下拉菜單dropdown組件實例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • vue3?錨點定位的多種實現(xiàn)方式

    vue3?錨點定位的多種實現(xiàn)方式

    這篇文章主要介紹了vue3?多種方法的錨點定位,使用?Vue?Router?導航守衛(wèi)可以簡化導航邏輯、統(tǒng)一管理導航邏輯和進行權限控制,但需要學習和理解相關概念,并且需要手動編寫和管理導航守衛(wèi)的邏輯,本文給大家介紹的非常詳細,需要的朋友可以參考下
    2023-07-07
  • Vue3新屬性之css中使用v-bind的方法(v-bind?in?css)

    Vue3新屬性之css中使用v-bind的方法(v-bind?in?css)

    這篇文章主要介紹了Vue3新屬性css中使用v-bind(v-bind?in?css)的方法,本文結合實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-01-01
  • ElementUI之Message功能拓展詳解

    ElementUI之Message功能拓展詳解

    這篇文章主要介紹了ElementUI之Message功能拓展詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-10-10
  • Vue3 構建 Web Components使用詳解

    Vue3 構建 Web Components使用詳解

    這篇文章主要為大家介紹了Vue3 構建 Web Components使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-09-09
  • Vue框架中正確引入JS庫的方法介紹

    Vue框架中正確引入JS庫的方法介紹

    最近在學習使用vue框架,在使用中遇到了一個問題,查找相關資料終于找了正確的姿勢,所以這篇文章主要給大家介紹了關于在Vue框架中正確引入JS庫的方法,需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-07-07
  • Vue如何從1.0遷移到2.0

    Vue如何從1.0遷移到2.0

    這篇文章主要介紹了Vue如何從1.0遷移到2.0的相關資料,需要的朋友可以參考下
    2017-10-10
  • vue實現(xiàn)導航標題欄隨頁面滾動漸隱漸顯效果

    vue實現(xiàn)導航標題欄隨頁面滾動漸隱漸顯效果

    這篇文章主要介紹了vue實現(xiàn)導航標題欄隨頁面滾動漸隱漸顯效果,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-03-03
  • vue 數(shù)組和對象不能直接賦值情況和解決方法(推薦)

    vue 數(shù)組和對象不能直接賦值情況和解決方法(推薦)

    這篇文章主要介紹了vue 數(shù)組和對象不能直接賦值情況和解決方法,需要的朋友可以參考下
    2017-10-10
  • vue 基于abstract 路由模式 實現(xiàn)頁面內嵌的示例代碼

    vue 基于abstract 路由模式 實現(xiàn)頁面內嵌的示例代碼

    這篇文章主要介紹了vue 基于abstract 路由模式 實現(xiàn)頁面內嵌的示例代碼,幫助大家更好的理解和使用vue框架,感興趣的朋友可以了解下
    2020-12-12

最新評論