iOS原生與H5交互

一、WKWebView

WKWebView 初始化時,有一個參數叫configuration,它是WKWebViewConfiguration類型的參數,而WKWebViewConfiguration有一個屬性叫userContentController,它又是WKUserContentController類型的參數。

   WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc] init];
    config.preferences = [[WKPreferences alloc] init];
    config.preferences.minimumFontSize = 10;
    config.preferences.javaScriptEnabled = YES;
    config.preferences.javaScriptCanOpenWindowsAutomatically = NO;
    config.userContentController = [[WKUserContentController alloc] init];
    config.processPool = [[WKProcessPool alloc] init];
    config.userContentController = [WKUserContentController new]; //在創建wkWebView時,需要將被js調用的方法注冊進去,oc與js端對應實現
    [config.userContentController addScriptMessageHandler:self name:@"callFunciton"];

    WKWebView *wkWebView = [[WKWebView alloc]initWithFrame:self.view.frame configuration:config];
    self.wkWebView = wkWebView;
    wkWebView.navigationDelegate = self;
    wkWebView.UIDelegate = self;
    NSURLRequest *request = [[NSURLRequest alloc]initWithURL:self.url];
    [wkWebView loadRequest:request];
    [self.view addSubview:wkWebView];</pre>

1.JS調用原生MessageHandler

WKUserContentController對象有一個方法

- (void)addScriptMessageHandler:(id <WKScriptMessageHandler>)scriptMessageHandler name:(NSString *)name;

JS調用OC時,這句代碼非常重要

// 在創建wkWebView時,需要將被js調用的方法注冊進去,oc與js端對應實現
[self.wkWebView.configuration.userContentControlle addScriptMessageHandler:self name:@"callFunciton"];

addScriptMessageHandler:name:有兩個參數,第一個參數是userContentController的代理對象,第二個參數是JS里發送postMessage的對象。
所以要使用MessageHandler功能,就必須要實現WKScriptMessageHandler協議。

1.1.實現WKScriptMessageHandler代理方法

當js調用callFunction方法時,會回調此代理方法:

- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{ if ([message.name isEqualToString:@"callFunction"]) { 
    NSString *methodName = message.name;
    id params = message.body;
    //調用原生掃碼 
    if ([methodName isEqualToString:@"scan"]) {
       
    }
 } 
}

Tip: addScriptMessageHandler很容易引起循環引用,導致控制器無法被釋放

- (void)dealloc{
    [self.wkWebView.configuration.userContentController removeScriptMessageHandlerForName:@"callFunction"];
}

1.2.JS中使用方法:

window.webkit.messageHandlers.<name>.postMessage(<messageBody>) 
//其中<name>,就是上面方法里的第二個參數`name`。
//例如我們調用API的時候第二個參數填@"callFunction",那么在JS里就是:
window.webkit.messageHandlers.callFunction.postMessage(<messageBody>) 
//<messageBody>是一個鍵值對,鍵是body,值可以有多種類型的參數,body 的類型:Allowed types are NSNumber, NSString, NSDate, NSArray, NSDictionary, and NSNull messageBody可以為NULL或者其他參數,不能什么都不寫,否則不走代理方法

2.原生調用JS

 //1無參數
[self.webView evaluateJavaScript:@"show()" completionHandler:^(id _Nullable response, NSError * _Nullable error) {                
  //TODO
 }];

 //2有參數
 NSDictionary *dict = @{ @"userAgent": @“userAgentM”, @"custMac":@“custMac”};
 NSString *callBackString = [NSString stringWithFormat:@"show(%@)",[self jsonToString:dict]];
 [self.webView evaluateJavaScript:callBackString completionHandler:^(id _Nullable obj, NSError * _Nullable error) {
        
 }];

3.WKNavigationDelegate

可以在此通過連接的方式傳遞一些簡單的參數,也是一種H5與原生交互

- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {    
   NSString *url = navigationAction.request.URL.absoluteString;   
   if(![url isEqualToString:self.strURL]) {          
    // 頁面跳轉
   }
   decisionHandler(WKNavigationActionPolicyAllow);
}

二、WebViewJavaScriptBridge

WebViewJavaScriptBridge 用于 WKWebView & UIWebView 中 OC 和 JS 交互。
它的基本原理是: 把 OC 的方法注冊到橋梁中,讓 JS 去調用;把 JS 的方法注冊在橋梁中,讓 OC 去調用。

1. 初始化

1.導入頭文件 #import <WebViewJavascriptBridge.h>

2.建立 WebViewJavaScriptBridge 和 WebView 之間的關系

_jsBridge = [WebViewJavascriptBridge bridgeForWebView:_webView];

3.在HTML 文件中,復制粘貼這兩段 JS 函數

function setupWebViewJavascriptBridge(callback) {
  if (window.WebViewJavascriptBridge) { 
    return callback(WebViewJavascriptBridge); 
  }if (window.WVJBCallbacks) { 
    return window.WVJBCallbacks.push(callback); 
  }
  window.WVJBCallbacks = [callback]; // 創建一個 WVJBCallbacks 全局屬性數組,并將 callback 插入到數組中。
  var WVJBIframe = document.createElement('iframe'); // 創建一個 iframe 元素
  WVJBIframe.style.display = 'none'; // 不顯示   WVJBIframe.src = 'wvjbscheme://__BRIDGE_LOADED__'; // 設置 iframe 的 src 屬性
  document.documentElement.appendChild(WVJBIframe); // 把 iframe 添加到當前文導航上。
  setTimeout(function() { 
    document.documentElement.removeChild(WVJBIframe) 
  }, 0)
} 
// 這里主要是注冊OC將要調用的JS方法。
 setupWebViewJavascriptBridge(function(bridge){

 });

2. 注入OC、JS方法

往橋梁中注入 OC 方法

/* scanClick 是 OC block 的一個別名
*  block本身,是JS通過某種方式調用到scanClick的時候,執行的代碼塊
*  data,由于OC這端由JS調用,所以data是JS端傳遞過來的數據
*  responseCallback OC端的block 執行完畢之后,往JS端傳遞的數據 
*/ 
[_jsBridge registerHandler:@"scanClick" handler:^(id data, WVJBResponseCallback responseCallback) {
  NSLog(@"dataFrom JS : %@",data[@"data"]);
  responseCallback(@"掃描結果 : www.baidu.com");
}];

往橋梁中注入 JS 函數

/* testJavaScriptFunction: 是注入到橋梁中JS函數的別名,以供OC端調用。
*  data: 回調函數的data,既然JS函數由OC調用,所以data是OC端傳遞過來的數據。
*  responseCallback: JS調用在被OC調用完畢之后,向OC端傳遞的數據 
*/
// 這里主要是注冊 OC 將要調用的 JS 方法。
 setupWebViewJavascriptBridge(function(bridge){ 
        // 聲明 OC 需要調用的 JS 方法。
        bridge.registerHanlder('testJavaScriptFunction',function(data,responseCallback){ 
            // data 是 OC 傳遞過來的數據.
            // responseCallback 是 JS 調用完畢之后傳遞給 OC 的數據
            alert("JS 被 OC 調用了.");
            responseCallback({data: "js 的數據",from : "JS"});
        })
 });

3. 調用OC、JS方法

OC調用JS

// 單純的調用 JSFunction,不往 JS 傳遞參數,也不需要 JSFunction 的返回值。
[_jsBridge callHandler:@"changeBGColor"]; 
// 調用 JSFunction,并向 JS 傳遞參數,但不需要 JSFunciton 的返回值。
[_jsBridge callHandler:@"changeBGColor" data:@"把 HTML 的背景顏色改成橙色!!!!"];
// 調用 JSFunction ,并向 JS 傳遞參數,也需要 JSFunction 的返回值。
[_jsBridge callHandler:@"changeBGColor" data:@"傳遞給 JS 的參數" responseCallback:^(id responseData) {
  NSLog(@"JS 的返回值: %@",responseData);
}];

JS調用OC

// JS單純的調用OC的block
WebViewJavascriptBridge.callHandler('scanClick'); 
// JS調用OC的block,并傳遞JS參數
WebViewJavascriptBridge.callHandler('scanClick',"JS 參數"); 
// JS調用OC的block,傳遞JS參數,并接受OC的返回值。
WebViewJavascriptBridge.callHandler('scanClick',{data : "這是JS傳遞到OC的掃描數據"},function(dataFromOC){
  alert("JS 調用了 OC 的掃描方法!");
  document.getElementById("returnValue").value = dataFromOC;
});

4. OC釋放Block

OC中,在當前控制器消失的時候,要記得把注入到橋梁中的 OC block,從橋梁中刪除,否則,可能會出現控制器無法釋放的情況。

[_jsBridge removeHandler:@"scanClick"];

5.示例

1.JS -> OC 的交互

在 OC 中,通過 WebViewJavascriptBridge 注冊一個修改 navigationBar 顏色的 Block

[_jsBridge registerHandler:@"colorClick" handler:^(id data, WVJBResponseCallback responseCallback) {
  self.navigationController.navigationBar.barTintColor = [UIColor colorWithRed:arc4random_uniform(256) / 255.0 green:arc4random_uniform(256) / 255.0 blue:arc4random_uniform(256) / 255.0 alpha:1.0];
  responseCallback(@"顏色修改完畢!");
}];

在 JS 中,通過某種方式去調用這個 OC 的 block。

WebViewJavascriptBridge.callHandler('colorClick',function(dataFromOC) {
  alert("JS 調用了 OC 注冊的 colorClick 方法");
  document.getElementById("returnValue").value = dataFromOC;
})

OC -> JS 的交互

往橋梁中,注入一個修改 HTML body 顏色的 JSFunction。

// 在這里聲明OC需要主動調用JS的方法。
setupWebViewJavascriptBridge(function(bridge) {
  bridge.registerHandler('changeBGColor',function(data,responseCallback){
    // alert('aaaaaa');
    document.body.style.backgroundColor = "orange";
    document.getElementById("returnValue").value = data;
   });
}); 

然后在 OC 端通過橋梁調用這個 changeBGColor

 [_jsBridge callHandler:@"changeBGColor" data:@"把 HTML 的背景顏色改成橙色!!!!"];

轉載: iOS原生與H5交互
參考: WebViewJavaScriptBridge 基本使用

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

推薦閱讀更多精彩內容