詳解IOS微信上Vue單頁面應(yīng)用JSSDK簽名失敗解決方案
背景
手機(jī)型號:
型號:iphone 7 / iphone xs max
版本:ios 10.3.1 / ios 12.1
微信版本:WeChat 6.7.3
問題還原:
Vue應(yīng)用(vue-router)上使用history模式,在某個頁面內(nèi)調(diào)用微信JSSDK相關(guān)API,如掃碼、分享等,使用當(dāng)前頁面URL總會出現(xiàn)簽名錯誤(invalid signature),導(dǎo)致API調(diào)用失敗。
問題根源
Vue-Router進(jìn)行路由切換的時候,總是會操作瀏覽器的歷史記錄,從而響應(yīng)頁面URL變化。
在JSSDK文檔頁面有這么一句話:
同一個url僅需調(diào)用一次,對于變化url的SPA的web app可在每次url變化時進(jìn)行調(diào)用,目前Android微信客戶端不支持pushState的H5新特性,所以使用pushState來實現(xiàn)web app的頁面會導(dǎo)致簽名失敗,此問題會在Android6.2中修復(fù)
但根據(jù)多次測試情況來看,情況恰好相反,在Android下直接使用 window.location.href 得出的URL進(jìn)行簽名是完全沒問題(可能已升級至Android6.2以上版本),在IOS上就不行了。
這是因為在IOS上,無論路由切換到哪個頁面,實際真正有效的的簽名URL是【第一次進(jìn)入應(yīng)用時的URL】。
比如進(jìn)入應(yīng)用首頁是: https://m.app.com,需要使用JSSDK的頁面A是:https://m.app.com/product1/123,無論從首頁進(jìn)入到A頁面之前,中間跳轉(zhuǎn)過多少次路由,最終簽名有效的URL還是首頁URL。
解決方案
方案一
直接粗暴處理方式:
在進(jìn)入需要使用JSSDK頁面(B)的前一個頁面(A),即 A > B,直接使用 window.location.href 或 window.open 打開B頁面,此時因為B頁面是直接刷新方式進(jìn)入,所以當(dāng)前B頁面URL無論IOS或Android都是可以直接拿來簽名的。
這種方式處理缺點也很明顯,如頁面刷新抖動太厲害不夠平滑過渡,再比如B頁面需要從vuex中取出緩存信息,如果這些緩存信息不是通過vuex保存在localstorage的話,取出來的肯定都是空的。
方案二
思路:既然IOS僅可使用第一次進(jìn)入應(yīng)用的URL來簽名,那么在vuex上緩存一個微信簽名URL,IOS保存第一次進(jìn)入應(yīng)用的URL,Android則緩存為每個頁面的URL。簽名時,直接從緩存拿出簽名URL來處理。
// 全局判斷是否IOS方法 function isIos(){ const u = navigator.userAgent; return u.indexOf("iPhone") > -1 || u.indexOf("Mac OS") > -1; }
1. 定義vuex緩存
... { state: { wechatSignUrl: "" }, mutations: { setWechatSignUrl(state, wxSignUrl) { // 關(guān)鍵點 // IOS僅記錄第一次進(jìn)入頁面時的URL // IOS微信切換路由實際URL不變,只能使用第一進(jìn)入頁面的URL進(jìn)行簽名 if (isIos() && state.wxSignUrl !== '') { return; } state.wxSignUrl = wxSignUrl; } }, getters: { getWechatSignUrl: (state) => state.wxSignUrl } } ...
關(guān)鍵點在于設(shè)置更新微信簽名URL判斷的地方:首次進(jìn)入應(yīng)用頁面的時候肯定會觸發(fā)更新,若是IOS且簽名URL已經(jīng)設(shè)置過了,那么就不需要更新設(shè)置了,只要不退出或刷新應(yīng)用,緩存永遠(yuǎn)都會是首次進(jìn)入頁面URL。
2. 路由守衛(wèi)內(nèi)觸發(fā)更新簽名URL
import store form "@/stores" // 獲取真實有效微信簽名URL function getWechatSignUrl(to){ if(isIos()) { return window.location.href; } else { // 此處$appHost需要自行處理 return $appHost + to.fullPath } } ... $router.beforeEach((to, from, next) => { store.commit("setWechatSignUrl", getWechatSignUrl(to)); }) ...
在路由守衛(wèi)內(nèi)更新簽名URL,保證IOS是使用當(dāng)前頁面URL,Android是使用目標(biāo)路由完整地址再加上域名
3. 使用簽名URL調(diào)用JSSDK API
在使用JSSDK API的頁面通過vuex取出緩存的微信簽名URL,然后進(jìn)行簽名。
比如:
import store form "@/stores" ... { methods: { initWechatShareConfig() { const that = this; const wxSignUrl = store.getters['getWechatSignUrl']; const wxShareConfigs = { // 微信分享配置 } // 初始化微信分享 $wechat.share(wxSignUrl, wxShareConfigs); } } } ...
$wechat.share 是根據(jù)JSSDK文檔二次封裝的分享方法,內(nèi)部是根據(jù)wxSignUrl先進(jìn)行簽名,然后再調(diào)用分享API
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Vue中的ref作用詳解(實現(xiàn)DOM的聯(lián)動操作)
這篇文章主要介紹了Vue中的ref作用詳解(實現(xiàn)DOM的聯(lián)動操作),需要的朋友可以參考下2017-08-08在vue中v-bind使用三目運(yùn)算符綁定class的實例
今天小編就為大家分享一篇在vue中v-bind使用三目運(yùn)算符綁定class的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09