uni-app三部曲之三: 路由攔截

1.引言

路由攔截,個人理解就是在頁面跳轉的時候,增加一級攔截器,實現一些自定義的功能,其中最重要的就是判斷跳轉的頁面是否需要登錄后查看,如果需要登錄后查看且此時系統并未登錄,就需要跳轉到登錄頁,登錄后跳轉到原來想要訪問的頁面。

2.實現

uni-app的路由跳轉分別是uni.navigateTo:保留當前頁面,跳轉到應用內的某個頁面,使用uni.navigateBack可以返回到原頁面;uni.redirectTo:關閉當前頁面,跳轉到應用內的某個頁面;uni.reLaunch:關閉所有頁面,打開到應用內的某個頁面;uni.switchTab:跳轉到 tabBar 頁面,并關閉其他所有非 tabBar 頁面;uni.navigateBack:關閉當前頁面,返回上一頁面或多級頁面,可通過?getCurrentPages()?獲取當前的頁面棧,決定需要返回幾層。

在進行路由跳轉時,通過uni.addInterceptor,添加攔截器,實現路由攔截。

攔截是否需要登錄的基本思路是通過pages.json文件對應的配置needLogin,進行頁面的配置,在攔截時,找到有此配置的所有頁面,得到所有頁面的路徑,和本次訪問的路徑進行匹配,如果匹配成功,則判斷當前是否是登錄狀態,若沒有登錄,跳轉到登錄界面,進行登錄。

3.代碼

代碼主要分為攔截器代碼,登錄頁pages配置,登錄頁登錄按鈕功能。

1.攔截器代碼

/*** by 菲鴿 on 2024-03-06* 路由攔截,通常也是登錄攔截* 可以設置路由白名單,或者黑名單,看業務需要選哪一個* 我這里應為大部分都可以隨便進入,所以使用黑名單*/
import { useUserStore } from '@/store'
import { getNeedLoginPages, needLoginPages as _needLoginPages } from '@/utils'
import { getAccessToken } from '@/utils/auth'// 登錄頁面路徑
const loginRoute = '/pages/login/index'const isLogined = () => {const userStore = useUserStore()return userStore.userInfo.isLogin && getAccessToken()
}const isDev = import.meta.env.DEV// 黑名單登錄攔截器 - (適用于大部分頁面不需要登錄,少部分頁面需要登錄)
const navigateToInterceptor = {// 注意,這里的url是 '/' 開頭的,如 '/pages/index/index',跟 'pages.json' 里面的 path 不同invoke({ url }: { url: string }) {console.log(url) // /pages/route-interceptor/index?name=feige&age=30const path = url.split('?')[0]let needLoginPages: string[] = []// 為了防止開發時出現BUG,這里每次都獲取一下。生產環境可以移到函數外,性能更好if (isDev) {needLoginPages = getNeedLoginPages()} else {needLoginPages = _needLoginPages}const isNeedLogin = needLoginPages.includes(path)if (!isNeedLogin || isLogined()) {return true}const redirectRoute = `${loginRoute}?redirect=${encodeURIComponent(url)}`console.log(redirectRoute)uni.navigateTo({ url: redirectRoute })return false},
}export const routeInterceptor = {install() {uni.addInterceptor('navigateTo', navigateToInterceptor)uni.addInterceptor('reLaunch', navigateToInterceptor)uni.addInterceptor('redirectTo', navigateToInterceptor)},
}

2.輔助函數代碼

import { pages, subPackages, tabBar } from '@/pages.json'
export const getLastPage = () => {// getCurrentPages() 至少有1個元素,所以不再額外判斷// const lastPage = getCurrentPages().at(-1)// 上面那個在低版本安卓中打包回報錯,所以改用下面這個【雖然我加了src/interceptions/prototype.ts,但依然報錯】const pages = getCurrentPages()return pages[pages.length - 1]
}/** 判斷當前頁面是否是tabbar頁  */
export const getIsTabbar = () => {if (!tabBar) {return false}if (!tabBar.list.length) {// 通常有tabBar的話,list不能有空,且至少有2個元素,這里其實不用處理return false}const lastPage = getLastPage()const currPath = lastPage.routereturn !!tabBar.list.find((e) => e.pagePath === currPath)
}/*** 獲取當前頁面路由的 path 路徑 和 redirectPath 路徑* path 如 ‘/pages/login/index’* redirectPath 如 ‘/pages/demo/base/route-interceptor’*/
export const currRoute = () => {const lastPage = getLastPage()const currRoute = (lastPage as any).$page// console.log('lastPage.$page:', currRoute)// console.log('lastPage.$page.fullpath:', currRoute.fullPath)// console.log('lastPage.$page.options:', currRoute.options)// console.log('lastPage.options:', (lastPage as any).options)// 經過多端測試,只有 fullPath 靠譜,其他都不靠譜const { fullPath } = currRoute as { fullPath: string }// console.log(fullPath)// eg: /pages/login/index?redirect=%2Fpages%2Fdemo%2Fbase%2Froute-interceptor (小程序)// eg: /pages/login/index?redirect=%2Fpages%2Froute-interceptor%2Findex%3Fname%3Dfeige%26age%3D30(h5)return getUrlObj(fullPath)
}const ensureDecodeURIComponent = (url: string) => {if (url.startsWith('%')) {return ensureDecodeURIComponent(decodeURIComponent(url))}return url
}
/*** 解析 url 得到 path 和 query* 比如輸入url: /pages/login/index?redirect=%2Fpages%2Fdemo%2Fbase%2Froute-interceptor* 輸出: {path: /pages/login/index, query: {redirect: /pages/demo/base/route-interceptor}}*/
export const getUrlObj = (url: string) => {const [path, queryStr] = url.split('?')// console.log(path, queryStr)if (!queryStr) {return {path,query: {},}}const query: Record<string, string> = {}queryStr.split('&').forEach((item) => {const [key, value] = item.split('=')// console.log(key, value)query[key] = ensureDecodeURIComponent(value) // 這里需要統一 decodeURIComponent 一下,可以兼容h5和微信y})return { path, query }
}
/*** 得到所有的需要登錄的pages,包括主包和分包的* 這里設計得通用一點,可以傳遞key作為判斷依據,默認是 needLogin, 與 route-block 配對使用* 如果沒有傳 key,則表示所有的pages,如果傳遞了 key, 則表示通過 key 過濾*/
export const getAllPages = (key = 'needLogin') => {// 這里處理主包const mainPages = [...pages.filter((page) => !key || page[key]).map((page) => ({...page,path: `/${page.path}`,})),]// 這里處理分包const subPages: any[] = []subPackages.forEach((subPageObj) => {// console.log(subPageObj)const { root } = subPageObjsubPageObj.pages.filter((page) => !key || page[key]).forEach((page: { path: string } & Record<string, any>) => {subPages.push({...page,path: `/${root}/${page.path}`,})})})const result = [...mainPages, ...subPages]// console.log(`getAllPages by ${key} result: `, result)return result
}/*** 得到所有的需要登錄的pages,包括主包和分包的* 只得到 path 數組*/
export const getNeedLoginPages = (): string[] => getAllPages('needLogin').map((page) => page.path)/*** 得到所有的需要登錄的pages,包括主包和分包的* 只得到 path 數組*/
export const needLoginPages: string[] = getAllPages('needLogin').map((page) => page.path)

3.需要登錄頁面配置

<route lang="json5">
{style: {navigationBarTitleText: '辦公',},needLogin: true,
}
</route>
<template><view class="bg-white overflow-hidden pt-2 px-4"><view>123</view></view>
</template><script setup lang="ts"></script><style lang="scss"></style>

能在組件中配置頁面的信息,主要得益于@uni-helper/vite-plugin-uni-pages?插件的功勞,該插件由?uni-helper?官方團隊開發,可參考uni 插件 | unibest。

4.登錄按鈕功能

// 登錄系統 一進系統就需要登錄
const handleLogin = async () => {const loginRes = await loginApi.login(loginForm)console.log(loginRes)setAccessToken(loginRes.data.accessToken)setRefreshToken(loginRes.data.refreshToken)// 獲取路由路徑 進行跳轉const fullPath = currRoute()console.log(fullPath)uni.redirectTo({ url: fullPath.query.redirect })
}

登錄按鈕就是獲取登錄數據,存儲token,重定向至原來訪問的界面。

4.功能展示

uni-app登錄驗證

5.寫在最后

本文首先感謝unibestuniapp?開發框架,在unibest項目基礎上,添加了一些小的功能,基本能滿足路由跳轉時的攔截,為后續業務的權限管理打下堅實基礎。

本文如有疏漏之處,歡迎批評指正。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/43547.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/43547.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/43547.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Python地震波逆問題解構算法復雜信號分析

&#x1f3af;要點 &#x1f3af;時域、時頻域以及時間和頻率相關聯偏振特性分析三種算法 | &#x1f3af;時域波參數估計算法 | &#x1f3af;機器學習模型波形指紋分析算法 | &#x1f3af;色散曲線和頻率相關波分析算法 | &#x1f3af;動態傾斜校正算法 | &#x1f3af;聲…

【JS|第21期】JavaScript模塊化:深入解析三種文件暴露方式

日期:2024年7月6日 作者:Commas 簽名:(? ?_?)? 積跬步以致千里,積小流以成江海…… 注釋:如果您覺得有所幫助,幫忙點個贊,也可以關注我,我們一起成長;如果有不對的地方,還望各位大佬不吝賜教,謝謝^ - ^ 1.01365 = 37.7834;0.99365 = 0.0255 1.02365 = 1377.4083…

前后端項目部署方案匯總

前端項目 1、本地打包部署 # 本地打包部署到線上服務器 npm run build && \ rsync -r ./dist/* root127.0.0.1:/www/www.demo.com/www2、服務器端打包部署 步驟 拉取代碼 -> 安裝依賴 -> 打包編譯 -> 拷貝到運行目錄 -> 發送成功消息shell命令 git pu…

新手小白報考學習PMP會遇到哪些“坑”?

PMP考試的陷阱實際上與其他大型證書考試差不多&#xff0c;主要是在選擇培訓機構和各種收費方面會遇到一些坑。 首先&#xff0c;并不是每個人都能經歷這些坑&#xff0c;因為PMP考試有一定的門檻。 作為引進國外的考試&#xff0c;報名都有中英文之分&#xff0c;所以先來看…

STM32的 DMA(直接存儲器訪問) 詳解

STM32的DMA&#xff08;Direct Memory Access&#xff0c;直接存儲器存取&#xff09;是一種在單片機中用于高效實現數據傳輸的技術。它允許外設設備直接訪問RAM&#xff0c;不需要CPU的干預&#xff0c;從而釋放CPU資源&#xff0c;提高CPU工作效率&#xff0c;本文基于STM32F…

[極客大挑戰 2019]RCE ME

[極客大挑戰 2019]RCE ME <?php error_reporting(0); if(isset($_GET[code])){$code$_GET[code];if(strlen($code)>40){die("This is too Long.");}if(preg_match("/[A-Za-z0-9]/",$code)){die("NO.");}eval($code); } else{highlight_f…

(附源碼)c#+winform實現遠程開機(廣域網可用)

實現邏輯 利用UDP協議發送特定格式的魔術包&#xff0c;以遠程喚醒具有特定MAC地址的目標計算機。目標計算機的BIOS和網絡配置需要支持Wake-on-LAN&#xff08;WOL&#xff09;功能&#xff0c;并且需要在目標計算機上配置正確的網絡喚醒設置。 源碼在最后 準備工作 進入Bio…

力學有限元的基石:虛功原理的推導

推導虛功方程的過程 彈性力學的平衡方程 在張量形式中&#xff0c;平衡方程為&#xff1a; ? ? σ b 0 \nabla \cdot \sigma b 0 ??σb0 用下標表示為&#xff1a; ? σ i j ? x j b i 0 \frac{\partial \sigma_{ij}}{\partial x_j} b_i 0 ?xj??σij??b…

知識圖譜入門筆記

自學參考&#xff1a; 視頻&#xff1a;斯坦福CS520 | 知識圖譜 最全知識圖譜綜述 詳解知識圖譜的構建全流程 知識圖譜構建&#xff08;概念&#xff0c;工具&#xff0c;實例調研&#xff09; 一、基本概念 知識圖譜&#xff08;Knowledge graph&#xff09;&#xff1a;由結…

Redis管理禁用命令

在redis數據量比較大時&#xff0c;執行 keys * &#xff0c;fluashdb 這些命令&#xff0c;會導致redis長時間阻塞&#xff0c;大量請求被阻塞&#xff0c;cpu飆升&#xff0c;嚴重可能導致redis宕機&#xff0c;數據庫雪崩。所以一些命令在生產環境禁止使用。 Redis 禁用命令…

【C語言】指針(3):探索-不同類型指針變量

目錄 一、字符指針變量 二、數組指針變量 三、二維數組傳參的本質 四、函數指針變量 4.1 函數指針變量 4.2 函數指針變量的使用 4.3 函數指針變量的拓展 五、函數指針數組 六、轉移表的應用 通過深入理解指針&#xff08;1&#xff09;和深入理解指針&#xff08;2&am…

67.SAP FICO-憑證類型學習

目錄 SAP憑證類型 憑證類型的作用 - OBA7 SAP默認的憑證類型更改 FI相應事務代碼默認憑證類型 - OBU1 對FB50、60、70默認憑證類型的更改 - OBZO 后勤貨物移動默認憑證類型 - OMBA 發貨憑證類型 收貨憑證類型 自動移動憑證類型 存貨盤點憑證類型 發票默認的憑證類…

深度學習Day-24:ResNeXt-50算法思考

&#x1f368; 本文為&#xff1a;[&#x1f517;365天深度學習訓練營] 中的學習記錄博客 &#x1f356; 原作者&#xff1a;[K同學啊 | 接輔導、項目定制] 要求&#xff1a; 閱讀給出代碼&#xff0c;判斷是否存在錯誤&#xff0c;正確與否都請給出你的思考&#xff1b;查找…

如何減少開發過程中的bug-數據庫篇

1.1慢查詢 1.1.1 是否命中索引 提起慢查詢&#xff0c;我們馬上就會想到加索引。如果一條SQL沒加索引&#xff0c;或者沒有命中索引的話&#xff0c;就會產生慢查詢。 索引哪些情況會失效&#xff1f; 查詢條件包含or&#xff0c;可能導致索引失效 如果字段類型是字符串&am…

LeetCode 0724.尋找數組的中心下標:前綴和(時空復雜度O(n)+O(1))

title: 724.尋找數組的中心下標 date: 2024-07-08 13:22:58 tags: [題解, LeetCode, 簡單, 數組, 前綴和] 【LetMeFly】724.尋找數組的中心下標&#xff1a;前綴和&#xff08;時空復雜度O(n)O(1)&#xff09; 力扣題目鏈接&#xff1a;https://leetcode.cn/problems/find-pi…

數據結構--二叉樹相關習題5(判斷二叉樹是否是完全二叉樹 )

1.判斷二叉樹是否是完全二叉樹 辨別&#xff1a; 不能使用遞歸或者算節點個數和高度來判斷。 滿二叉樹可以用高度和節點來判斷&#xff0c;因為是完整的。 但是完全二叉樹前面是滿的&#xff0c;但是最后一層是從左到右連續這種 如果仍然用這種方法的話&#xff0c;如下圖…

暑期備考2024小學生古詩文大會:吃透真題和知識點(持續)

2024年上海市小學生古詩文大會的自由報名初賽將于10月19日&#xff08;星期六&#xff09;正式開始&#xff0c;還有3個多月的時間。 為幫助孩子們備考&#xff0c;我持續分享往年上海小學生古詩文大會真題&#xff0c;這些題目來自我去重、合并后的1700在線題庫&#xff0c;每…

加密與安全_密鑰體系的三個核心目標之完整性解決方案

文章目錄 Pre機密性完整性1. 哈希函數&#xff08;Hash Function&#xff09;定義特征常見算法應用散列函數常用場景散列函數無法解決的問題 2. 消息認證碼&#xff08;MAC&#xff09;概述定義常見算法工作原理如何使用 MACMAC 的問題 不可否認性數字簽名&#xff08;Digital …

SketchUp Pro 2024:現代科技之詩意體驗

在那遙遠的唐朝&#xff0c;李白曾以詩酒為伴&#xff0c;游歷山川&#xff0c;揮灑才情。而今&#xff0c;若李白穿越時空&#xff0c;手握現代科技之利器——SketchUp Pro 2024&#xff0c;定會以詩意之筆&#xff0c;描繪這款軟件的神奇與魅力。 初識SketchUp Pro 2024 初…

Vue Router:History 模式 vs. Hash 模式

在開發 SPA&#xff08;單頁應用程序&#xff09;時&#xff0c;路由管理是不可或缺的一部分。Vue.js 框架中的 Vue Router 提供了兩種主要的路由模式&#xff1a;History 模式和 Hash 模式。理解這兩種模式的區別及其實現方式&#xff0c;對于開發和部署 Vue 應用至關重要。 …