iOS Native和H5交互

1. WKWebView

1.1 H5調(diào)Native(H5向Native傳遞消息)

1.1.1 H5調(diào)Native_直接攔截URL方式

顧名思義就是在WebView加載URL的時(shí)候父虑,讀出URL字符串存筏,看看里面有沒(méi)有你和H5特別約定的標(biāo)記饶辙,如果有說(shuō)明H5想向你傳遞消息职辅,就干H5讓你干的事考传,忽略這個(gè)URL的定向跳轉(zhuǎn)。
攔截方法在下面回調(diào):

webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void)

H5那邊的代碼
就是重定向當(dāng)前頁(yè)面URL。就是改變了location.href的值棚愤。這里舉例子標(biāo)記是bridge://doingSomething,傳參{parma1=sss,parma2=dy}

 h5AskNativeDoSomething1(){
    location.href = "bridge://doingSomething&parma1=sss&parma2=dy"
 }

Native代碼
發(fā)現(xiàn)URL中包含bridge://doingSomething戚啥,可以判斷H5要讓Native干事情奋单,那么就decisionHandler(.cancel)忽略這個(gè)URL的跳轉(zhuǎn),取出對(duì)應(yīng)參數(shù)干事情

func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
        
        let changeURLString:String = navigationAction.request.url?.absoluteString ?? "未知"
        print("要跳轉(zhuǎn)的URL:\(changeURLString)")
        
        /***方法1 攔截關(guān)鍵標(biāo)識(shí) bridge://doingSomething猫十,根據(jù)自己和H5定了規(guī)則览濒,去判斷取相應(yīng)的參數(shù) ***/
        
        //方法1:js調(diào)用native
        if changeURLString.contains("bridge://doingSomething"){
            decisionHandler(.cancel)
            
            let list:[String] = changeURLString.split(separator: "&").compactMap { "\($0)" }
            print(list)
            //TODO
        }else{
            decisionHandler(.allow)
        }
    }
1.1.2 H5調(diào)Native_WKScriptMessageHandler 回調(diào)方式

這是系統(tǒng)JavaScriptCore提供的功能,用的時(shí)候需要引用類庫(kù)import JavaScriptCore拖云。

H5那邊代碼:
Native那邊通過(guò)WKScriptMessageHandler注冊(cè)的js對(duì)象暴露出來(lái)贷笛,這邊想向原生傳遞消息,固定寫(xiě)法:window.webkit.messageHandlers.(JS對(duì)象XXXX字符串).postMessage({})
其中 window.webkit.messageHandlers 都是固定寫(xiě)法宙项,.postMessage({})也是固定寫(xiě)法乏苦。這里例子完整寫(xiě)法就是
window.webkit.messageHandlers.DIYJSBridge.postMessage({})
記住,如果沒(méi)有參數(shù)傳遞尤筐,({})小括號(hào)里面的大括號(hào)也不能省略汇荐,如果省略{},native那邊會(huì)無(wú)響應(yīng)!如果有參數(shù)傳遞盆繁,那么{}里面就是鍵值對(duì),如:{"param1":1,"param2":2}

/**
* native那邊通過(guò)WKScriptMessageHandler注冊(cè)的js對(duì)象暴露出來(lái)掀淘,這邊想向原生傳遞消息,固定寫(xiě)法:window.webkit.messageHandlers.原生那邊暴露的js對(duì)象XXXX.postMessage({})
* 其中 window.webkit.messageHandlers 都是固定寫(xiě)法油昂,.postMessage({})也是固定寫(xiě)法革娄。記住倾贰,如果沒(méi)有參數(shù)傳遞,({})小括號(hào)里面的大括號(hào)也不能省略拦惋,如果省略{},native那邊會(huì)無(wú)響應(yīng)
*/
h5AskNativeDoSomething2(){
   window.webkit.messageHandlers.DIYJSBridge.postMessage({})
}

Navtive代碼:
我們?cè)赪KWebView初始化的時(shí)候匆浙,給它配置WKWebViewConfiguration的里面添加一個(gè)類似監(jiān)聽(tīng)的東西:

let configuration:WKWebViewConfiguration = WKWebViewConfiguration()

//加監(jiān)聽(tīng)
let wkUserContnetController:WKUserContentController = WKUserContentController.init()
wkUserContnetController.add(self, name: "DIYJSBridge")//加監(jiān)聽(tīng)
configuration.userContentController = wkUserContnetController

webView = WKWebView.init(frame:self.view.bounds, configuration:configuration)
webView.scrollView.decelerationRate = UIScrollView.DecelerationRate.normal
webView.navigationDelegate = self
self.view.addSubview(webView)

wkUserContnetController.add(self, name: "DIYJSBridge")這里給self添加一個(gè)監(jiān)聽(tīng),監(jiān)聽(tīng)到DIYJSBridge這個(gè)JS對(duì)象發(fā)送的消息(先這么理解)架忌。
然后實(shí)現(xiàn)WKScriptMessageHandler 協(xié)議的方法:

//MARK: - WKScriptMessageHandler
func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
   print("方法2:\(message.name)")
   print("方法2:\(message.body)")
}

這里面可以根據(jù)WKScriptMessage 讀到方法的名稱name和參數(shù)body吞彤,然后就可以干事情。

1.1.3 H5調(diào)Native_三方庫(kù) WebViewJavascriptBridge

這個(gè)庫(kù)本質(zhì)還是攔截URL叹放,只是進(jìn)行了封裝饰恕,讓方法定義和傳參更加對(duì)象化。具體源碼井仰,有機(jī)會(huì)再寫(xiě)篇文章解析一下埋嵌!

H5那邊的代碼:
H5那邊需要在index.html定義一個(gè)方法,并調(diào)用俱恶。掛載對(duì)應(yīng)的JS對(duì)象雹嗦,完成相應(yīng)的配置初始化。

//初始化方法
function setupWebViewJavascriptBridge(callback) {
    if (window.WebViewJavascriptBridge) { return callback(WebViewJavascriptBridge); }
    if (window.WVJBCallbacks) { return window.WVJBCallbacks.push(callback); }
    window.WVJBCallbacks = [callback];
    var WVJBIframe = document.createElement('iframe');
    WVJBIframe.style.display = 'none';
    WVJBIframe.src = 'https://__bridge_loaded__';
    document.documentElement.appendChild(WVJBIframe);
    setTimeout(function() { document.documentElement.removeChild(WVJBIframe); }, 0);
}

//調(diào)用完成配置初始化
setupWebViewJavascriptBridge((bridge) => {});

現(xiàn)在H5想調(diào)用Native方法

h5AskNativeDoSomething3(){
  this.toast("方法3:H5調(diào)用Native")     
  window.WebViewJavascriptBridge.callHandler('NativeToDingSomething',{"Pa1":"11111"}, function responseCallback(responseData) {
         console.log(responseData)
  })
}

window.WebViewJavascriptBridge.callHandler()固定寫(xiě)法合是,其中參數(shù)

第一個(gè)參數(shù):Native那邊定義標(biāo)記字符串了罪,如:`NativeToDingSomething`
第二個(gè)參數(shù):傳參,為{}括起來(lái)的鍵值對(duì)聪全,如:`{"Pa1":"11111"}`
第三個(gè)參數(shù):回調(diào)函數(shù)泊藕,如果Native那邊實(shí)現(xiàn)(下面Native實(shí)現(xiàn)了)了對(duì)應(yīng)的回調(diào)Block,所帶的回調(diào)參數(shù)會(huì)在這個(gè)responseData里面

Native代碼:

Native這邊引入了三方庫(kù)WebViewJavascriptBridge难礼,如后導(dǎo)入娃圆。
代碼初始化一個(gè)WebViewJavascriptBridge實(shí)例,并關(guān)聯(lián)上WebView

/**方法3 通過(guò)三方類庫(kù)**/
self.webViewBridge = WebViewJavascriptBridge.init(webView)
self.webViewBridge.setWebViewDelegate(self)

這時(shí)設(shè)置注冊(cè)一個(gè)標(biāo)記NativeToDingSomething蛾茉,發(fā)現(xiàn)H5 callHandler這個(gè)標(biāo)記后干事情....

self.webViewBridge.registerHandler("NativeToDingSomething") { (ret:Any?, callBack:WVJBResponseCallback?) in
   print(ret)
   //TODO
  if callBack != nil{
     callBack!(["result":true,"data":["One","Two","Three"]])
  }
}

這樣H5那邊如果觸發(fā)了

window.WebViewJavascriptBridge.callHandler('NativeToDingSomething',{"Pa1":"11111"}, function responseCallback(responseData) {
        console.log("Native傳遞過(guò)來(lái)的參數(shù):",responseData)
  })
}

Native這邊收到消息會(huì)打印

Optional({
    Pa1 = 11111;
})

同時(shí)Native這邊執(zhí)行了回調(diào)callBack!(["result":true,"data":["One","Two","Three"]])

H5那邊的控制臺(tái)會(huì)打铀夏亍:


image.png

這樣就完成了一次H5發(fā)起的雙向通信!

1.2 Native調(diào)H5(Native向H5傳遞消息)

1.2.1 Native調(diào)H5_直接WebView調(diào)用evaluateJavaScript方法
open func evaluateJavaScript(_ javaScriptString: String, completionHandler: ((Any?, Error?) -> Void)? = nil)

如:

self.webView.evaluateJavaScript("nativeAskH5DoSomething('小孩')") { (any:Any?, err:Error?) in
  if err == nil{
    print("調(diào)用成功谦炬,返回結(jié)果\(any)")
  }else{
    print("調(diào)用失敗\(err)")
  }
}

其中nativeAskH5DoSomething('小孩')是H5定義的js方法

window.nativeAskH5DoSomething = function(mess){
    console.log(mess)
    return "我們都是好孩子"
}

注意:這種方法調(diào)用的js方法悦屏,必須掛載在Web的window對(duì)象上,不然會(huì)找不到!
evaluateJavaScript去調(diào)用h5中的方法之后键思,這個(gè)function要執(zhí)行完成之后才會(huì)回調(diào)的app這邊窜管,如果這個(gè)function中報(bào)錯(cuò)了,回調(diào)到app這邊就會(huì)一直報(bào)錯(cuò)稚机。所以建議function中的東西可以先延遲執(zhí)行幕帆。

1.2.2 Native調(diào)H5_通過(guò) WebViewJavascriptBridge三方庫(kù)

這個(gè)H5調(diào)Native反過(guò)來(lái)相當(dāng)于H5注冊(cè)一個(gè)標(biāo)記,Native去調(diào)用赖条。

H5代碼失乾,注冊(cè)方法標(biāo)識(shí)

window.WebViewJavascriptBridge.registerHandler("nativeAskH5DoSomething_WebViewJavascriptBridge",(data,callBack) => {
    this.toast("原生調(diào)用我了")
    console.log("原生調(diào)用我了")
    console.log("傳入?yún)?shù):",data)
    callBack({"p1":1,"p2":90})
})

第一個(gè)參數(shù)data:表示原生傳入的參數(shù)
第二個(gè)參數(shù)callBack:是回調(diào)方法常熙,執(zhí)行后可穿參數(shù)給原生,告訴執(zhí)行情況

Native代碼 事件觸發(fā)執(zhí)行

self.webViewBridge.callHandler("nativeAskH5DoSomething_WebViewJavascriptBridge", data: "10086") { (ret:Any?) in
      print("H5傳遞回來(lái)的參數(shù)結(jié)果:\(ret)")
}

執(zhí)行結(jié)果:
H5打印結(jié)果

image.png

Native打印結(jié)果

H5傳遞回來(lái)的參數(shù)結(jié)果:Optional({
    p1 = 1;
    p2 = 90;
})

這樣也完成了一次Native發(fā)起的雙向通信碱茁!

2. UIWebView

H5調(diào)Native: 1.1.1裸卫,1.1.3 和WKWebView相同!
Native調(diào)H5: 1.2.1纽竣,1.2.2 和WKWebView相同墓贿,多了一種方式!

唯一的區(qū)別就是在1.1.2上蜓氨,同時(shí)Native調(diào)H5多了使用JSContext方式
WKWebView用的是:WKScriptMessageHandler
UIWebView用的是:JSContext
因?yàn)楝F(xiàn)在都用WKWebView了聋袋,具體使用參照老鐵博客:http://www.reibang.com/p/88345985fe94

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市穴吹,隨后出現(xiàn)的幾起案子幽勒,更是在濱河造成了極大的恐慌,老刑警劉巖港令,帶你破解...
    沈念sama閱讀 219,589評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件啥容,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡顷霹,警方通過(guò)查閱死者的電腦和手機(jī)咪惠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,615評(píng)論 3 396
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)淋淀,“玉大人硝逢,你說(shuō)我怎么就攤上這事∩鸷恚” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,933評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵叫乌,是天一觀的道長(zhǎng)柴罐。 經(jīng)常有香客問(wèn)我,道長(zhǎng)憨奸,這世上最難降的妖魔是什么革屠? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,976評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮排宰,結(jié)果婚禮上似芝,老公的妹妹穿的比我還像新娘。我一直安慰自己板甘,他們只是感情好党瓮,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,999評(píng)論 6 393
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著盐类,像睡著了一般寞奸。 火紅的嫁衣襯著肌膚如雪呛谜。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,775評(píng)論 1 307
  • 那天枪萄,我揣著相機(jī)與錄音隐岛,去河邊找鬼。 笑死瓷翻,一個(gè)胖子當(dāng)著我的面吹牛聚凹,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播齐帚,決...
    沈念sama閱讀 40,474評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼妒牙,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了童谒?” 一聲冷哼從身側(cè)響起单旁,我...
    開(kāi)封第一講書(shū)人閱讀 39,359評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎饥伊,沒(méi)想到半個(gè)月后象浑,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,854評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡琅豆,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,007評(píng)論 3 338
  • 正文 我和宋清朗相戀三年愉豺,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片茫因。...
    茶點(diǎn)故事閱讀 40,146評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡蚪拦,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出冻押,到底是詐尸還是另有隱情驰贷,我是刑警寧澤,帶...
    沈念sama閱讀 35,826評(píng)論 5 346
  • 正文 年R本政府宣布洛巢,位于F島的核電站括袒,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏稿茉。R本人自食惡果不足惜锹锰,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,484評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望漓库。 院中可真熱鬧恃慧,春花似錦、人聲如沸渺蒿。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,029評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)茂装。三九已至良瞧,卻和暖如春陪汽,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背褥蚯。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,153評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工挚冤, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人赞庶。 一個(gè)月前我還...
    沈念sama閱讀 48,420評(píng)論 3 373
  • 正文 我出身青樓训挡,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親歧强。 傳聞我的和親對(duì)象是個(gè)殘疾皇子澜薄,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,107評(píng)論 2 356

推薦閱讀更多精彩內(nèi)容