iOS Native和H5交互

1. WKWebView

1.1 H5調Native(H5向Native傳遞消息)

1.1.1 H5調Native_直接攔截URL方式

顧名思義就是在WebView加載URL的時候,讀出URL字符串,看看里面有沒有你和H5特別約定的標記,如果有說明H5想向你傳遞消息,就干H5讓你干的事,忽略這個URL的定向跳轉。
攔截方法在下面回調:

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

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

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

Native代碼
發現URL中包含bridge://doingSomething,可以判斷H5要讓Native干事情,那么就decisionHandler(.cancel)忽略這個URL的跳轉,取出對應參數干事情

func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
        
        let changeURLString:String = navigationAction.request.url?.absoluteString ?? "未知"
        print("要跳轉的URL:\(changeURLString)")
        
        /***方法1 攔截關鍵標識 bridge://doingSomething,根據自己和H5定了規則,去判斷取相應的參數 ***/
        
        //方法1:js調用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調Native_WKScriptMessageHandler 回調方式

這是系統JavaScriptCore提供的功能,用的時候需要引用類庫import JavaScriptCore。

H5那邊代碼:
Native那邊通過WKScriptMessageHandler注冊的js對象暴露出來,這邊想向原生傳遞消息,固定寫法:window.webkit.messageHandlers.(JS對象XXXX字符串).postMessage({})
其中 window.webkit.messageHandlers 都是固定寫法,.postMessage({})也是固定寫法。這里例子完整寫法就是
window.webkit.messageHandlers.DIYJSBridge.postMessage({})
記住,如果沒有參數傳遞,({})小括號里面的大括號也不能省略,如果省略{},native那邊會無響應!如果有參數傳遞,那么{}里面就是鍵值對,如:{"param1":1,"param2":2}

/**
* native那邊通過WKScriptMessageHandler注冊的js對象暴露出來,這邊想向原生傳遞消息,固定寫法:window.webkit.messageHandlers.原生那邊暴露的js對象XXXX.postMessage({})
* 其中 window.webkit.messageHandlers 都是固定寫法,.postMessage({})也是固定寫法。記住,如果沒有參數傳遞,({})小括號里面的大括號也不能省略,如果省略{},native那邊會無響應
*/
h5AskNativeDoSomething2(){
   window.webkit.messageHandlers.DIYJSBridge.postMessage({})
}

Navtive代碼:
我們在WKWebView初始化的時候,給它配置WKWebViewConfiguration的里面添加一個類似監聽的東西:

let configuration:WKWebViewConfiguration = WKWebViewConfiguration()

//加監聽
let wkUserContnetController:WKUserContentController = WKUserContentController.init()
wkUserContnetController.add(self, name: "DIYJSBridge")//加監聽
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添加一個監聽,監聽到DIYJSBridge這個JS對象發送的消息(先這么理解)。
然后實現WKScriptMessageHandler 協議的方法:

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

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

1.1.3 H5調Native_三方庫 WebViewJavascriptBridge

這個庫本質還是攔截URL,只是進行了封裝,讓方法定義和傳參更加對象化。具體源碼,有機會再寫篇文章解析一下!

H5那邊的代碼:
H5那邊需要在index.html定義一個方法,并調用。掛載對應的JS對象,完成相應的配置初始化。

//初始化方法
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);
}

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

現在H5想調用Native方法

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

window.WebViewJavascriptBridge.callHandler()固定寫法,其中參數

第一個參數:Native那邊定義標記字符串,如:`NativeToDingSomething`
第二個參數:傳參,為{}括起來的鍵值對,如:`{"Pa1":"11111"}`
第三個參數:回調函數,如果Native那邊實現(下面Native實現了)了對應的回調Block,所帶的回調參數會在這個responseData里面

Native代碼:

Native這邊引入了三方庫WebViewJavascriptBridge,如后導入。
代碼初始化一個WebViewJavascriptBridge實例,并關聯上WebView

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

這時設置注冊一個標記NativeToDingSomething,發現H5 callHandler這個標記后干事情....

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

這樣H5那邊如果觸發了

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

Native這邊收到消息會打印

Optional({
    Pa1 = 11111;
})

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

H5那邊的控制臺會打?。?/p>

image.png

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

1.2 Native調H5(Native向H5傳遞消息)

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

如:

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

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

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

注意:這種方法調用的js方法,必須掛載在Web的window對象上,不然會找不到!
evaluateJavaScript去調用h5中的方法之后,這個function要執行完成之后才會回調的app這邊,如果這個function中報錯了,回調到app這邊就會一直報錯。所以建議function中的東西可以先延遲執行。

1.2.2 Native調H5_通過 WebViewJavascriptBridge三方庫

這個H5調Native反過來相當于H5注冊一個標記,Native去調用。

H5代碼,注冊方法標識

window.WebViewJavascriptBridge.registerHandler("nativeAskH5DoSomething_WebViewJavascriptBridge",(data,callBack) => {
    this.toast("原生調用我了")
    console.log("原生調用我了")
    console.log("傳入參數:",data)
    callBack({"p1":1,"p2":90})
})

第一個參數data:表示原生傳入的參數
第二個參數callBack:是回調方法,執行后可穿參數給原生,告訴執行情況

Native代碼 事件觸發執行

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

執行結果:
H5打印結果

image.png

Native打印結果

H5傳遞回來的參數結果:Optional({
    p1 = 1;
    p2 = 90;
})

這樣也完成了一次Native發起的雙向通信!

2. UIWebView

H5調Native: 1.1.1,1.1.3 和WKWebView相同!
Native調H5: 1.2.1,1.2.2 和WKWebView相同,多了一種方式!

唯一的區別就是在1.1.2上,同時Native調H5多了使用JSContext方式
WKWebView用的是:WKScriptMessageHandler
UIWebView用的是:JSContext
因為現在都用WKWebView了,具體使用參照老鐵博客:http://www.lxweimin.com/p/88345985fe94

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,606評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,582評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,540評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,028評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,801評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,223評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,294評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,442評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,976評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,800評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,996評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,543評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,233評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,926評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,702評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,991評論 2 374

推薦閱讀更多精彩內容