小程序webview的現(xiàn)狀
10年積累的成都做網(wǎng)站、網(wǎng)站建設(shè)經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問(wèn)題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站設(shè)計(jì)制作后付款的網(wǎng)站建設(shè)流程,更有吉利免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
h6頁(yè)面在小程序中的交互(跳轉(zhuǎn))場(chǎng)景
主要痛點(diǎn)
在完成相關(guān)操作后, 頁(yè)面狀態(tài)需要更新,目前常見(jiàn)的更新方式有如下兩種:
第一種方案,功能上沒(méi)有問(wèn)題,但會(huì)導(dǎo)致頁(yè)面刷新,如果頁(yè)面操作復(fù)雜,需要多次刷新
第二種方案,正向操作時(shí)體驗(yàn)比方案一好,但導(dǎo)致了另外一個(gè)問(wèn)題:操作 跳轉(zhuǎn)層級(jí)過(guò)深,尤其返回的時(shí)候簡(jiǎn)直讓人崩潰。
小程序中,h6頁(yè)面打開(kāi)新頁(yè)面方式
我們先來(lái)看下小程序中常見(jiàn)的h6跳h6的方式:
我們采用的是方式3,理由如下:
由于這種方案可能會(huì)達(dá)到小程序的10層限制。所以在一些重要頁(yè)面建議加入“ 回到首頁(yè)”的操作,通過(guò)這個(gè)操作來(lái)縮短小程序歷史棧
回到首頁(yè)方案簡(jiǎn)述
(如果不感興趣這部分可以直接略過(guò))
wx.miniProgram.reLaunch({ url: '/pages/webview/bridge?url=項(xiàng)目首頁(yè)地址' })
先聲明,我們webview的路徑是/pages/webview/webview
/pages/webview/bridge是個(gè)中轉(zhuǎn)頁(yè),有如下特點(diǎn): 該頁(yè)面并 不是最終打開(kāi)h6頁(yè)面的webview頁(yè),而是一個(gè) 中轉(zhuǎn)頁(yè)。
主要用作返回處理
這個(gè)中轉(zhuǎn)頁(yè):主要保證reLaunch到某h6頁(yè)面后,用戶仍然可以點(diǎn)擊返回到小程序首頁(yè)。
該方案通常用于:小程序中內(nèi)嵌了多個(gè)業(yè)務(wù)線的h6頁(yè)面這種場(chǎng)景。
一個(gè)內(nèi)容發(fā)布場(chǎng)景
我們從首頁(yè)進(jìn)入發(fā)布頁(yè),完成發(fā)布后,跳轉(zhuǎn)至商品詳情頁(yè)
那么對(duì)于一個(gè)新用戶來(lái)講,整個(gè)操作過(guò)程是這樣的:
這個(gè)場(chǎng)景就是同一個(gè)頁(yè)面,里面不同的內(nèi)容項(xiàng)需要跳轉(zhuǎn)不同的頁(yè)面去操作,然后再回到原來(lái)頁(yè)面更新?tīng)顟B(tài)的問(wèn)題。
假如商品詳情頁(yè)沒(méi)有“回到首頁(yè)”的入口,那么這個(gè)用戶要想回到首頁(yè)。。。需要按8次“返回” = =!
經(jīng)過(guò)這個(gè)體驗(yàn)后,我想一般的用戶是沒(méi)有勇氣再發(fā)布內(nèi)容的。
當(dāng)然也有另一種這種折中方案
就是商品提到的,在連接中加入某個(gè)標(biāo)志位,比如在url中加入__isonshowrefresh=1,webview在打開(kāi)連接時(shí)候,會(huì)去讀取這個(gè)參數(shù),如果有,則每次在onShow時(shí)候,重新加載url,通過(guò)刷新頁(yè)面進(jìn)行頁(yè)面狀態(tài)更新。
這個(gè)體驗(yàn)也不爽,就是在復(fù)雜的頁(yè)面會(huì)多次刷新。
聲明
我下面要講的這個(gè)方案并不是停留在設(shè)想階段,它已經(jīng)在線上跑了
想看效果的朋友,可以在微信小程序中搜:
“轉(zhuǎn)轉(zhuǎn)二手交易網(wǎng)”-“0元免費(fèi)領(lǐng)”-(底部)“送閑置賺星星”-進(jìn)入到發(fā)布頁(yè)后
分類(lèi)(跳轉(zhuǎn)h6,選中內(nèi)容后返回,將參數(shù)傳給之前的h6)
取件地址(跳轉(zhuǎn)native原生地址選擇,選中后返回,將參數(shù)傳給之前的h6)
OK,我們進(jìn)入今天的主題
小程序中h6頁(yè)面onShow和跨頁(yè)面通信的實(shí)現(xiàn)
首先想到的就是onShow方法的實(shí)現(xiàn),之前有人提議用visibilitychange來(lái)實(shí)現(xiàn)onShow方法。
但調(diào)研過(guò)后,這種方式在ios中表現(xiàn)符合預(yù)期,但是在安卓手機(jī)里,是不能按預(yù)期觸發(fā)的。所以該方案被我否了。
于是就有了下面的方案
原理介紹
這個(gè)方案需要h6和小程序的webview都做處理。
核心思想: 利用webview的hash特性
為什么要執(zhí)行window.history.go(-1)
這一步是整個(gè)方案的精髓:
方案延伸(跨頁(yè)面數(shù)據(jù)傳遞)
小程序里另個(gè)一常見(jiàn)的場(chǎng)景就是調(diào)用第三業(yè)務(wù)(或者己方業(yè)務(wù)),在做完某些操作后需要把選中的數(shù)據(jù)帶回之前的頁(yè)面。
如前面提到的例子:發(fā)布頁(yè),需要選擇發(fā)布類(lèi)型,然后返回,發(fā)布頁(yè)發(fā)布類(lèi)型局部更新
當(dāng)然有些同學(xué)會(huì)說(shuō):我可以用setInterval,監(jiān)控localStorage。在新頁(yè)面選中內(nèi)容后,設(shè)置localStorage,然后在返回不就可以了。
我這里說(shuō)的是 通用方案。如果頁(yè)面都是由己方業(yè)務(wù)線維護(hù)的當(dāng)然可以隨便折騰。
但是一旦涉及到第三方業(yè)務(wù)線,尤其不同域名頁(yè)面的業(yè)務(wù)調(diào)用,這種通信方式就尷尬了。
那我的方案怎么處理呢,我總結(jié)了一張圖
我們來(lái)解讀一下這張圖:
整個(gè)過(guò)程就是這樣
代碼示意:
小程序
小程序webview要先做幾方面考慮:
小程序端webview.wpy
<web-view wx:if="{{url}}" src="{{url}}" binderror="onError" bindload="onLoaded" bindmessage="onPostMessage"></web-view> // 鏈接處理工具方法 import util from '@/lib/util'; // 全局?jǐn)?shù)據(jù)存儲(chǔ)操作類(lèi) import routeParams from '@/lib/routeParams'; const urlReg = /^(https?\:\/\/[^?#]+)(\?[^#]*)?(#[^\?&]+)?(.+)?$/; let messageData = {}; export default class extends wepy.page { data = { // 頁(yè)面展示次數(shù) pageShowCount: 0, // 頁(yè)面url中query部分的參數(shù)對(duì)象 mQuery: {}, ... } onShow(){ ++this.pageShowCount; // 獲取其他頁(yè)面經(jīng)過(guò)操作后,需要傳遞給h6的參數(shù) let data = routeParams.getBackFromData() || {}; // webview頁(yè)面狀態(tài)更新 if(this.pageShowCount > 1 && this.mQuery.__isonshowpro && this.mQuery.__isonshowpro === '1' || data.refresh){ // 獲取需要傳遞給h6頁(yè)面的參數(shù) let refreshParam = data.refreshParam; ... // 如果連接中帶有需要處理onShow邏輯的參數(shù)(通過(guò)url的hash和h6交互,而不是刷頁(yè)面) if (this.pageShowCount > 1 && this.mQuery.__isonshowpro === '1') { let [whole, mainUrl, queryStr, hashStr, hashQueryStr] = urlReg.exec(this.url); // 在url的hash中加入新的參數(shù) hashStr = (hashStr || '#').substring(1); if (refreshParam) { delete refreshParam.refresh; } const messageData = this.getNavigateMessageData(); // 將需要更新的參數(shù)傳給頁(yè)面hash hashStr = util.addQuery(hashStr, Object.assign({ // onshow標(biāo)志位 __isonshow: 1, // wa主動(dòng)觸發(fā)hashchange標(biāo)志位 // 其實(shí)目前通過(guò)__isonshow就可以判斷是wa主動(dòng)觸發(fā)hashchange // 設(shè)置該字段是為了明確功能,且以后擴(kuò)展用 __wachangehash: 1, // 時(shí)間戳刷新 __hashtimestamp: Date.now() }, messageData, refreshParam)); this.url = mainUrl + queryStr + '#' + hashStr; console.log('【webview-hashchange-url】', this.url); // 這里要加個(gè)延遲,否則在webview返回到webview時(shí),無(wú)法觸發(fā)hashchange,應(yīng)該是小程序bug setTimeout(()=> { this.$apply(); }, 50); // 通過(guò)修改query參數(shù),刷新webview } else { ... } ... } } /** * 獲取需要發(fā)送的消息數(shù)據(jù) */ getNavigateMessageData(){ let rst = {}; for(let i in messageData){ /* message結(jié)構(gòu): message: { key: 'xx', // 消息名稱 content: 'xx', // 消息內(nèi)容 trigger: { // 觸發(fā)條件 type: '', // 觸發(fā)類(lèi)型 - immediately 在下一次onshow或者打開(kāi)頁(yè)面中立刻觸發(fā), - url 在找到指定h6鏈接時(shí)觸發(fā) content: '' // 條件內(nèi)容 - type=immediately 時(shí)為空 - type=url 時(shí)候?yàn)閔6鏈接地址 } } */ const message = messageData[i]; const trigger = message.trigger || {}; // 立刻發(fā)送、路徑觸發(fā) if(trigger.type === 'immediately' || trigger.type === 'url' && this.url.indexOf(trigger.content) > -1){ // 將key和content集合到一個(gè)對(duì)象中,便于hash直接設(shè)置 rst[message.key] = message.content; // 消息通知后,從緩存中刪除 delete messageData[message.key]; } } console.log('【webview-get-message】', rst); console.log('【webview-message-cache】', messageData); return rst; } /** * 存儲(chǔ)消息數(shù)據(jù) */ storeNavigateMessageData(message){ if(message && message.key){ console.log('【webview-store-message】', message) // 通過(guò)key設(shè)置每一條消息名稱 messageData[message.key] = message; console.log('【webview-message-cache】', messageData); } } methods = { // 接收發(fā)送過(guò)來(lái)的消息 onPostMessage(e){ if(!e.detail.data)return; const detailData = e.detail.data; // 獲取消息數(shù)據(jù) let messageData = getValueFromMixedArray(detailData, 'messageData', true); if (messageData) { // 存儲(chǔ) this.storeNavigateMessageData(messageData); } ... } } ... }
上面東西看著挺多,總結(jié)下來(lái)就是幾點(diǎn):
h6端
h6端在做修改時(shí)也要考慮幾點(diǎn):
最好能把這些交互邏輯封裝起來(lái)
讓業(yè)務(wù)方比較簡(jiǎn)單方便的調(diào)用
這里我新定義了2個(gè)方法
onShow(callback)
例子:發(fā)布頁(yè)面,需要選擇分類(lèi),返回時(shí)需要更新分類(lèi)信息
import { isZZWA, onShow } from '@/lib/sdk' import URL from '@/lib/url' ... created () { if (isZZWA()) { onShow(() => { // 地址信息 const addressInfo = URL.getHashParam('zzwaAddress') console.log('addressInfo:', decodeURIComponent(addressInfo)) ... // 分類(lèi)信息 const selecteCateInfo = URL.getHashParam('selecteCateInfo') console.log('selecteCateInfo:', selecteCateInfo) ... } else { ... } } ...
serviceDone(data, condition)
描述:業(yè)務(wù)結(jié)束,需要將數(shù)據(jù)傳遞給指定頁(yè)面
參數(shù):
data Object 需要傳遞的數(shù)據(jù) {key: 'xx', content: 'xx'}
condition String|Number 觸發(fā)條件
例子:類(lèi)型選擇頁(yè)
import { isZZWA, serviceDone } from '@/lib/sdk' // 類(lèi)型選擇點(diǎn)擊 typeChooseClick (param, type) { ... if (isZZWA()) { // 需要返回的數(shù)據(jù) const data = { key: 'selecteCateInfo', content: JSON.stringify({...}) } // 通過(guò)postMessage發(fā)送給小程序,-1表示返回上一頁(yè)面 serviceDone(data, -1) } else { ... } }
ok,我們來(lái)看看h6端的sdk是怎么實(shí)現(xiàn)的
import util from './util'; class WASDK { /** * Create a instance. * @ignore */ constructor(){ // hashchang事件處理 if('onhashchange' in window && window.addEventListener && !WASDK.hashInfo.isInit){ // 更新標(biāo)志位 WASDK.hashInfo.isInit = true; // 綁定hashchange window.addEventListener('hashchange', ()=>{ // 如果小程序webview修改的hash,才進(jìn)行處理 if (util.getHash(window.location.href, '__wachangehash') === '1') { // 這塊有個(gè)坑: // ios小程序webview在修改完url的hash之后,頁(yè)面hashchange和更新都可以正常觸發(fā) // 但是:h6調(diào)用部分小程序能力會(huì)失敗(如:ios在設(shè)置完hash后,調(diào)用wx.uploadImg會(huì)失敗,需要重新設(shè)置wx.config) // 因?yàn)閕os小程序的邏輯是,url只要發(fā)生變化,wx.config中的appId就找不到了 // 所以需要重新進(jìn)行wx.config配置 // 這一步是獲取之前設(shè)置wx.config的參數(shù)(需要從服務(wù)端拿,因?yàn)橹耙呀?jīng)獲取過(guò)了,這里從緩存直接?。? const jsticket = window.native && window.native.adapter && window.native.adapter.jsticket || null; const ua = navigator.userAgent; // 非安卓系統(tǒng)要重新設(shè)置wx.config if (jsticket && !(ua.indexOf('Android') > -1 || ua.indexOf('Adr') > -1)) { window.wx.config({ debug: false, appId: jsticket.appId, timestamp: jsticket.timestamp, nonceStr: jsticket.noncestr, signature: jsticket.signature, jsApiList: ['onMenuShareTimeline', 'onMenuShareAppMessage', 'onMenuShareQQ', 'onMenuShareQZone', 'onMenuShareWeibo', 'scanQRCode', 'chooseImage', 'uploadImage', 'previewImage', 'getLocation', 'openLocation'] }) } // 觸發(fā)緩存數(shù)組的回調(diào) WASDK.hashInfo.callbackArr.forEach(callback=>{ callback(); }) // 執(zhí)行返回操作(這一步是重點(diǎn)?。。? // 因?yàn)閣ebview設(shè)置完hash參數(shù)后,會(huì)使webview歷史棧+1 // 而實(shí)際并不需要這次多余的歷史記錄,所以需要執(zhí)行返回操作把它去掉 // 即便是返回操作,也僅僅是hash層面的變更,所以不會(huì)觸發(fā)頁(yè)面刷新 // 用setTimeout表示在下一次事件循環(huán)進(jìn)行返回操作。如果后面有對(duì)dom操作可以在當(dāng)前次事件循環(huán)完成 setTimeout(()=>{ window.history.go(-1); }, 0); } }, false) } } /** * hash相關(guān)信息 */ static hashInfo = { // 是否已經(jīng)初始化 isInit: false, // hash回調(diào)香瓜數(shù)組 callbackArr: [] } /** * 頁(yè)面再次展示時(shí)鉤子方法 * @param {Function} callback - 必填, callback回調(diào)方法, 回傳參數(shù)為hash部分問(wèn)號(hào)后面的參數(shù)解析對(duì)象 */ @execLog onShow(callback){ if (typeof callback === 'function') { // 對(duì)回調(diào)方法進(jìn)行onshow邏輯包裝,并推入緩存數(shù)組 WASDK.hashInfo.callbackArr.push(function(){ // 檢查是否是指定參數(shù)發(fā)生變化 if(util.getHash(window.location.href, '__isonshow') === '1'){ // 觸發(fā)onShow回調(diào) callback(); } }) } else { util.console.error(`參數(shù)錯(cuò)誤,調(diào)用onShow請(qǐng)傳入正確callback回調(diào)`); } } /** * 業(yè)務(wù)處理完成并發(fā)送消息 * @param {Object} obj - 必填項(xiàng),消息對(duì)象 * @param {String} obj.key - 必填項(xiàng),消息名稱 * @param {String} obj.content - 可選項(xiàng),消息內(nèi)容,默認(rèn)空串,如果是內(nèi)容對(duì)象,請(qǐng)轉(zhuǎn)換成字符串 * @param {String|Number} condition - 可選項(xiàng),默認(rèn)僅進(jìn)行postMessage * String - 可以傳指定url的路徑,當(dāng)小程序webview打開(kāi)指定的url或者onshow時(shí),會(huì)觸發(fā)該消息 * 也可傳小程序path,這個(gè)為以后預(yù)留 * Number - 返回到指定的測(cè)試,類(lèi)似history.go(-1),如: -1,-2 */ @execLog serviceDone(obj, condition){ if(obj && obj.key){ // 消息體 const message = { // 消息名稱 key: obj.key, // 消息體 content: obj.content || '', // 觸發(fā)條件 trigger: { // 類(lèi)型 'immediately'在下一次onshow中立刻觸發(fā), 'url',在找到指定h6鏈接時(shí)觸發(fā),'path'在打開(kāi)指定小程序路徑時(shí)觸發(fā) type: 'immediately', // 條件內(nèi)容,immediately是為空,url是為h6鏈接地址,path是為小程序路徑 content: '' } }; // 解析觸發(fā)條件 condition = condition || 0; // 如果是路徑 if(typeof condition === 'string' && (condition.indexOf('http') > -1 || condition.indexOf('pages/') > -1)){ // 設(shè)置消息觸發(fā)條件 message.trigger = { type: condition.indexOf('http') > -1 ? 'url' : 'path', content: condition } } // 發(fā)送消息 wx.miniProgram.postMessage({ data: { messageData: message } }); // 如果不是url或者path觸發(fā),則對(duì)conditon是否需要返回進(jìn)行判斷 if(message.trigger.type === 'immediately'){ // 查看是否需要返回指定的層級(jí),兼容傳入'-1'字符串這種類(lèi)型的場(chǎng)景 try{ condition = parseInt(condition, 10); }catch(e){} // 保證返回級(jí)數(shù)的正確性 if(condition && typeof condition === 'number' && !isNaN(condition)){ this.handler.navigateBack({delta: Math.abs(condition)}); } } }else{ util.console.error(`參數(shù)錯(cuò)誤,調(diào)用serviceDone方法,傳入的對(duì)象中不包含key值`); } } ... } window.native = new Native(); export default native;
這個(gè)看著也挺多,總結(jié)下來(lái)是兩點(diǎn):
onShow方法的實(shí)現(xiàn)
綁定一個(gè)hashchange事件(這里做了防止重復(fù)綁定事件的處理)
將傳入的onShow自定義事件緩存在一個(gè)數(shù)組中,hashchange觸發(fā)時(shí),根據(jù)特有的標(biāo)志位__isonshow和__wachangehash確定是否觸發(fā)
serviceDone方法的實(shí)現(xiàn)
ok,整個(gè)方案就介紹完了
結(jié)語(yǔ)
最早的方案并不完全是這樣的,但原理是一樣的。在我實(shí)現(xiàn)的過(guò)程中發(fā)現(xiàn)原始方案有很多問(wèn)題
于是我又做了大量的改造和細(xì)節(jié)優(yōu)化,于是形成了上面的最終方案。
這個(gè)方案屬于侵入式改造方案,需要各業(yè)務(wù)方改造自己的代碼。雖然有一定改造成本,但用戶體驗(yàn)的收益非常明顯。
ps:我們的QA在測(cè)試時(shí)都說(shuō)“這用起來(lái)就爽多了”
注意:
采用這個(gè)方案需要注意幾點(diǎn):
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。
當(dāng)前文章:詳解小程序中h5頁(yè)面onShow實(shí)現(xiàn)及跨頁(yè)面通信方案
URL標(biāo)題:http://jinyejixie.com/article34/iicgse.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供響應(yīng)式網(wǎng)站、、網(wǎng)站制作、虛擬主機(jī)、搜索引擎優(yōu)化、App開(kāi)發(fā)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)