uni-app跨端自定義指令實現(xiàn)按鈕權(quán)限操作
前言
初看這個標(biāo)題可能很迷,uni-app明明不支持自定義指令,這文章是在搞笑嗎,本文對于uni-app自定義指令實現(xiàn)按鈕權(quán)限的方式也有可能是多余,但為了給業(yè)務(wù)部門更友好的開發(fā)體驗,還是做了一些可能沒意義的操作,讓移動端和PC端對于按鈕權(quán)限是使用方式一致,都是使用自定義指令的書寫方式來實現(xiàn)按鈕權(quán)限,雖然uni-app不支持自定義指令,但是我們有webpack的loader起到代碼轉(zhuǎn)換的作用,將自定義指令v-perms:add在編譯階段轉(zhuǎn)換為v-if="$perms('add')",然后去處理$perms方法對于權(quán)限的判斷,就可以具有自定義指令實現(xiàn)按鈕權(quán)限的開發(fā)體驗了。
每個工具的實現(xiàn)都不算簡單,本文實現(xiàn)的loader,盡管原理挺簡單的,好像可以一蹴而就,但真的去實現(xiàn)的話,發(fā)現(xiàn)還是需要很多的知識儲備,整體流程也沒想象的那么簡單,但有個點子就是好的開始,經(jīng)過些許努力,終將成就一個不錯的工具。
準(zhǔn)備
實現(xiàn)uni-app自定義指令按鈕權(quán)限需要涉及到對于vue.config.js新增loader配置,基礎(chǔ)正則知識,webpack的loader開發(fā)和調(diào)試,以及npm本地調(diào)試和發(fā)布,接下來就從了解這些前置知識開始。
目錄結(jié)構(gòu)

chainWebpack 新增loader
vue的loader配置是通過chainWebpack來實現(xiàn)鏈?zhǔn)脚渲茫?code>vue.config.js新增一個loader配置可以查看vue-cli文檔,具體實現(xiàn):
// vue.config.js
module.exports = {chainWebpack: config => {// GraphQL Loaderconfig.module.rule('graphql').test(/.graphql$/).use('graphql-tag/loader').loader('graphql-tag/loader').end()}
} 正則
實現(xiàn)自定義指令替換,需要查找所有的v-perms:xxx,并且需要拿到xxx對應(yīng)的值,將其轉(zhuǎn)換為v-if="$perms('xxx')",所以就有了如下表達式,其作用就是匹配v-perms:xxx表達式,并提取到xxx內(nèi)容。
/v-perms:([^/<>\s]*)/g
針對正則基礎(chǔ)比較薄弱的同學(xué),我來解釋一下上面這個正則表達式:
- 首先,v-perms:用于匹配以v-perms:開頭的字符串
- 接著是(),()稱為捕獲組,其圈住的內(nèi)容,就是我們要捕獲起來額外存儲的東西。
- []中的匹配符之間是“或”的關(guān)系,也就是說只要能匹配上其中一個就行了。
- ^原本是匹配字符串的開始位置,但是在[]表達式中使用代表著除什么之外
- *跟在其它表達式后面,意味著“前面這個表達式可以出現(xiàn)0次或多次
- 最后這個/g表示該表達式將用來查找所有可能的匹配,返回的結(jié)果可以是多個,如果不加/g最多只會匹配一個
- 總結(jié)起來這個正則表達式的作用就是匹配以v-perms:開頭的所有內(nèi)容,并提取到v-perms:和空格、< 、>、/之間的內(nèi)容,可以適應(yīng)我們常寫的多個場景
// 結(jié)尾是">" <button v-perms:add>新增按鈕</button> // 結(jié)尾是"/" <input v-perms:add/> // 結(jié)尾是空格 <button v-perms:add class="btn">新增按鈕 v-perms:add</button> // 結(jié)尾是"<" <button>新增按鈕 v-perms:add</button>
npm 包知識點
關(guān)于npm包本地調(diào)試以及發(fā)布的知識點還是蠻重要的,寫一些工具庫,組件,腳手架都能用到,也是面試會考察到的點,內(nèi)容比較多,可以看我之前寫的項目腳手架發(fā)布為npm包文章進行學(xué)習(xí),這邊就講解一下npm的本地調(diào)試。
npm link
執(zhí)行 npm link,可以將本地npm包鏈接到全局的 node_modules
一些包沒辦法使用全局,那么我們可以執(zhí)行npm link packageName將對應(yīng)包連接到當(dāng)前項目的node_modules
npm link packageName
執(zhí)行 npm unlink packageName 刪除本地連接
npm unlink packageName
loader 開發(fā)和調(diào)試
定義
Loader只是一個導(dǎo)出為函數(shù)的JavaScript模塊。
最簡單的Loader
// source入?yún)⒕褪俏覀兣渲玫膶?yīng)規(guī)則匹配到文件類型的文件流或者上一個loader處理后的結(jié)果。
module.exports = function (source) {return source;
}; Loader基礎(chǔ)操作
例如我們的.vue文件中包含如下代碼:
<u-button type="success" v-perms:edit>編輯</u-button>
我們需要將模板的v-perms:edit替換成v-if="$perms('edit')",只需要在source返回之前對它進行正則匹配和替換即可。
module.exports = function (source) {// TODO// 這里實現(xiàn)對于source的操作return source;
}; module.exports = function (source) {// TODO// 這里實現(xiàn)對于source的操作return source;
}; 本地調(diào)試 ResolveLoader
webpack默認情況下只會去 node_modules 目錄下尋找loader,但是我們可以通過ResolveLoader.modules配置loader的加載目錄,通過上述的目錄結(jié)構(gòu)可以看出我將loader寫在根目錄的loaders下,所以可以有如下配置。
// vue.config.js
module.exports = {configureWebpack: {resolveLoader: {// 配置loader可以從node_modules和loaders目錄下查找modules: ["node_modules", "./loaders/"],},},// 為.vue文件新增一個uni-perms-loader的loaderchainWebpack: (config) => {config.module.rule("vue").test(/\.vue$/).use("uni-perms-loader").loader("uni-perms-loader").end();},
}; npm link 在 /loaders/uni-perms-loader目錄下執(zhí)行npm link,將uni-perms-loader連接到全局,執(zhí)行時需要配置好package.json,具體配置可以查看文章。
{"name": "uni-perms-loader", // 包名"version": "1.0.0", // 版本號"description": "uni-app 自定義權(quán)限指令替換", // 介紹"main": "index.js", // 入口"scripts": {"test": "echo \"Error: no test specified\" && exit 1"},"keywords": [ // npm包搜索關(guān)鍵詞"uni-app","自定義指令","directive"],"author": "LBINGXIN","license": "MIT" // 開源協(xié)議
} 在當(dāng)前項目根目錄下執(zhí)行npm link uni-perms-loader,將uni-perms-loader連接到本地node_modules,uni-perms-loader 即使npm包名,我們在package.json配置的名稱name??梢栽诋?dāng)前項目的node_modules中查到uni-perms-loader。

vue.config.js 配置,這邊就不需要ResolveLoader了
// vue.config.js
module.exports = {// 為.vue文件新增一個uni-perms-loader的loaderchainWebpack: (config) => {config.module.rule("vue").test(/\.vue$/).use("uni-perms-loader").loader("uni-perms-loader").end();},
}; 本地調(diào)試是用于方便開發(fā)時對于loader進行測試,在實際的應(yīng)用中還是發(fā)布為npm包,方便多個項目使用
完整代碼
這個loader的代碼還是很簡單,就是匹配到自定義指令,然后獲取對應(yīng)參數(shù),然后替換成v-if
loaders/uni-perms-loader/index.js
uni-perms-loader具體實現(xiàn)
function replaceDirective(source) {// 正則表達式const reg = /v-perms:([^/<>\s]*)/g;// 使用replace實現(xiàn)字符串替換// 這邊的$n就是指捕獲組()匹配的內(nèi)容,索引是從1開始,我們這邊只有一個,所以是$1,如果有// 多個,那就是$2,$3source = source.replace(reg, `v-if="$perms('$1')"`);return source;
}
module.exports = function (source) {// 處理替換source = replaceDirective(source);return source;
}; loaders/uni-perms-loader/package.json
package.json 配置
// 這邊是假數(shù)據(jù),不同團隊對于權(quán)限的數(shù)據(jù)定義都不大一樣,我列舉的算是最簡單的一種,不大需要
// 去考慮不同頁面按鈕名稱重復(fù),用當(dāng)前頁面路由進行區(qū)分
const data = {"/pages/index/index": {add: "新增",edit: "編輯",},"/pages/me/index": {add: "新增",auth: "授權(quán)",},
};
const checkPermission = (key) => {// 獲取所有路由棧const pages = getCurrentPages();const len = pages.length;// 獲取當(dāng)前頁面路由 route的值例如 pages/index/indexconst { route } = pages[len - 1];// 判斷當(dāng)前頁面是否包含對應(yīng)按鈕的權(quán)限r(nóng)eturn !!data[`/${route}`][key];
};
export default checkPermission; src/utils/index.js
按鈕權(quán)限判斷方式實現(xiàn)
import checkPermission from "./utils/index"; Vue.prototype.$perms = checkPermission;
main.js
將實現(xiàn)按鈕權(quán)限判斷的函數(shù)掛載到全局的$perms上
import checkPermission from "./utils/index"; Vue.prototype.$perms = checkPermission;
使用
<template><view class="content"><image class="logo" src="/static/logo.png"></image><view class="buttons"><u-button type="primary" v-perms:add>新增 </u-button><u-button type="success" v-perms:edit>編輯</u-button><u-button type="error" v-perms:delete>刪除</u-button><u-button type="warning" v-perms:auth>授權(quán)</u-button></view></view> </template>
小結(jié)
本文對應(yīng)的uni-app使用uni-perms-loader的demo已經(jīng)提交到Github了, 對應(yīng)的uni-perms-loader也發(fā)布到npm,這個loader算是比較簡單,大家可以自己實現(xiàn),也可以直接使用我發(fā)布的npm包。
這個loader可能不是必須的,但是有了它,可以有更好的開發(fā)體驗,也能和vue項目保持一致性,不算一無是處,具體是否使用,大家根據(jù)自己情況,有問題也歡迎探討。
到此這篇關(guān)于uni-app跨端自定義指令實現(xiàn)按鈕權(quán)限的文章就介紹到這了,更多相關(guān)uni-app按鈕權(quán)限內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
CocosCreator怎樣使用cc.follow進行鏡頭跟隨
這篇文章主要介紹了CocosCreator怎樣使用cc.follow進行鏡頭跟隨,想要學(xué)習(xí)follow的同學(xué),一定要看一下2021-04-04
Echarts柱狀圖實現(xiàn)同時顯示百分比+原始值+匯總值效果實例
echarts是一款功能強大、靈活易用的數(shù)據(jù)可視化庫,它提供了豐富的圖表類型和樣式,包括柱狀圖,這篇文章主要給大家介紹了關(guān)于Echarts柱狀圖實現(xiàn)同時顯示百分比+原始值+匯總值效果的相關(guān)資料,需要的朋友可以參考下2024-08-08
JavaScript 驗證瀏覽器是否支持javascript的方法小結(jié)
一些網(wǎng)站只有客戶端js的驗證,這樣就給一些hacher一些可趁之機了,不信大家測試一些網(wǎng)站的注冊功能看看,有部分網(wǎng)站只有客戶端驗證,我為了達到客戶端驗證,先是判斷瀏覽器是否支持js,如果不支持,提示信息,然后屏蔽掉發(fā)送按鈕2009-05-05
JS字符串分割方法整理匯總示例講解(3種截取方法和6個輔助方法)
JavaScript在開發(fā)中常常會需要截取字符串,而JS提供了slice()?、substring()、substr()?3種方法實現(xiàn)截取操作。另外還有字符串相關(guān)的6種輔助方法:indexOf()、lastIndexOf()、split()、join()、concat()、charAt()?。2023-02-02
three.js中文文檔學(xué)習(xí)之通過模塊導(dǎo)入
這篇文章主要給大家介紹了關(guān)于three.js中文文檔學(xué)習(xí)之通過模塊導(dǎo)入的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或使用three.js具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11
詳解JS截取字符串的三個方法substring,substr,slice
js中有三個截取字符的方法,分別是substring()、substr()、slice(),平時我們可能都用到過,但總是會對這些方法有點混淆。本文將詳細介紹一下這三者的區(qū)別,需要的可以參考一下2022-03-03
JavaScript實現(xiàn)將阿拉伯?dāng)?shù)字轉(zhuǎn)換成中文大寫
現(xiàn)在有需求將億元之內(nèi)的阿拉伯?dāng)?shù)字轉(zhuǎn)換成中文,例如:1234轉(zhuǎn)換后變?yōu)橐磺Ф偃脑俎D(zhuǎn)換成壹仟貳佰叁拾肆,所以本文給大家介紹了用JavaScript實現(xiàn)將阿拉伯?dāng)?shù)字轉(zhuǎn)換成中文大寫,感興趣的小伙伴跟著小編一起來看看吧2024-05-05

