OC與JS交互—— UIWebView,WKWebView

iOS中用來加載網頁,有兩個控件UIWebView(iOS8之前),WKWebView(iOS8誕生)。
此篇博文暫不討論由UIWebView轉到WKWebView適配的那些坑(Cookie問題,請求攔截問題等等)。
只是記錄一下webView與JS交互的方案調研。

UIWebView交互

  • URL Scheme

攔截鏈接,從而獲取鏈接里的信息,只能進行JS專遞信息到OC。可以利用代理方法實現。

-(BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType{
    
    //    NSLog(@"Scheme: %@", [url scheme]);  //方案
    //    NSLog(@"Host: %@", [url host]);    //主機
    //    NSLog(@"Port: %@", [url port]);    //端口
    //    NSLog(@"Path: %@", [url path]);    //路徑
    //    NSLog(@"Relative path: %@", [url relativePath]); //相關路徑
    //    NSLog(@"Path components as array: %@", [url pathComponents]);  //路徑的每個組成部分
    //    NSLog(@"Parameter string: %@", [url parameterString]);   //參數
    //    NSLog(@"Query: %@", [url query]);                        //查詢
    //    NSLog(@"Fragment: %@", [url fragment]);                  //分段

    //當前的鏈接
    NSURL* url = [request URL];
    NSString * string = [url absoluteString];
    //鏈接轉為字典,獲取我們想要的信息
    NSDictionary* dic = [self dictionaryFromQuery:string usingEncoding:NSUTF8StringEncoding];
    NSLog(@"%@",dic);
    //return NO 不允許跳轉鏈接
    return YES;
    
}

鏈接轉字典的工具方法

- (NSDictionary*)dictionaryFromQuery:(NSString*)query usingEncoding:(NSStringEncoding)encoding {
    NSCharacterSet* delimiterSet = [NSCharacterSet characterSetWithCharactersInString:@"&;"];
    NSMutableDictionary* pairs = [NSMutableDictionary dictionary];
    NSScanner* scanner = [[NSScanner alloc] initWithString:query];
    while (![scanner isAtEnd]) {
        NSString* pairString = nil;
        [scanner scanUpToCharactersFromSet:delimiterSet intoString:&pairString];
        [scanner scanCharactersFromSet:delimiterSet intoString:NULL];
        NSArray* kvPair = [pairString componentsSeparatedByString:@"="];
        if (kvPair.count == 2) {
            NSString* key = [[kvPair objectAtIndex:0]
                             stringByReplacingPercentEscapesUsingEncoding:encoding];
            NSString* value = [[kvPair objectAtIndex:1]
                               stringByReplacingPercentEscapesUsingEncoding:encoding];
            [pairs setObject:value forKey:key];
        }
    }
    
    return [NSDictionary dictionaryWithDictionary:pairs];
}
  • stringByEvaluatingJavaScriptFromString
    這個是OC調用JS方法
self.navigationItem.title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];

直接調用就可以,缺點就是返回值只能是NSString,不夠靈活。

  • JavaScriptCore框架(iOS7)
    如果你的項目最低支持到iOS7,那么可以嘗試一下這個。
    JavaScriptCore框架其實就是基于webkit中以C/C++實現的JavaScriptCore的一個包裝
    而JavaScriptCore是webkit中用來渲染JS的引擎,Safari瀏覽器和iOS開發中的webView都是由webkit來驅動的。
    所以利用這個框架可以讓 Objective-C 和 JavaScript 代碼直接的交互變得更加的簡單方便。

使用(注意:一定要在webView加載完成之后使用)

// OC調用JS方法
- (void)webViewDidFinishLoad:(UIWebView *)webView
{
    JSContext *jsContext = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    JSValue *value = [self.jsContext evaluateScript:@"document.title"];
    self.navigationItem.title = value.toString;
}
// JS調用OC方法
- (void)webViewDidFinishLoad:(UIWebView *)webView
{
    self.jsContext = [self.webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
    self.jsContext[@"share"] = ^() {
        //獲取到share里的所有參數
        NSArray *args = [JSContext currentArguments];
        //args中的元素是JSValue,需要轉成OC的對象
        NSMutableArray *messages = [NSMutableArray array];
        for (JSValue *obj in args) {
            [messages addObject:[obj toObject]];
        }
        NSLog(@"點擊分享js傳回的參數:\n%@", messages);
    };
}
  • WebViewJavascriptBridge(第三方框架)
    傳送門:https://github.com/marcuswestin/WebViewJavascriptBridge/
    如果你的項目想要支持iOS7之前的版本,同時想要擁有更好的交互體驗,建議集成一下這個第三方框架。
    作為github上8千多star的第三方框架,WebViewJavascriptBridge作為UIWebView交互工具無疑是極為優秀的,使用也極為簡單。

OC代碼

JS調用OC方法(JS給OC傳數據)

//基本設置
[WebViewJavascriptBridge enableLogging];//開啟日志
WebViewJavascriptBridge* bridge = [WebViewJavascriptBridge bridgeForWebView:self.webView];
[bridge setWebViewDelegate:self];
//方法注冊,AppClosePrice為方法名
[bridge registerHandler:@"AppClosePrice" handler:^(id data, WVJBResponseCallback responseCallback) {
        //data為JS傳來的數據
        NSLog(@"%@", data);
        [self.navigationController popViewControllerAnimated:YES];
        
    }];

** OC調用JS方法(OC給JS傳數據,也可以接收JS傳過來的數據)**
基本設置同上,就是注冊方法的時候不同。

//AppHidden:方法名,appCurVersion:OC給JS傳的數據
[self.bridge callHandler:@"AppHidden" data:appCurVersion responseCallback:^(id responseData) {

        NSLog(@"----JS傳過來的數據----%@",responseData);
    }];

JS代碼(由后臺實現,可以給后臺參考)

 /*這段代碼是固定的,必須要放到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 = 'wvjbscheme://__BRIDGE_LOADED__';
        document.documentElement.appendChild(WVJBIframe);
        setTimeout(function() { document.documentElement.removeChild(WVJBIframe) }, 0)
      }
    
      /*與OC交互的所有JS方法都要放在此處注冊,才能調用通過JS調用OC或者讓OC調用這里的JS*/
      setupWebViewJavascriptBridge(function(bridge) {
       var uniqueId = 1
                                   
       function log(message, data) {
         var log = document.getElementById('log')
         var el = document.createElement('div')
         el.className = 'logLine'
         el.innerHTML = uniqueId++ + '. ' + message + ':<br/>' + JSON.stringify(data)
         if (log.children.length) {
            log.insertBefore(el, log.children[0])
         } else {
           log.appendChild(el)
         }
       }
       /* Initialize your app here */
       
       /*注冊一個JS調用OC的方法,不帶參數*/
       bridge.registerHandler('AppClosePrice', function() {
          log("JS調用OC")
       })
        /*注冊一個JS調用OC的方法,帶參數*/
       bridge.registerHandler('AppClosePrice', function(data, responseCallback) {
                              
         log("Get user information from OC: ", data)
       })
       /*注冊一個OC調用JS的方法*/
       bridge.callHandler('AppHidden', function(responseData) {
         log("JS call ObjC's getUserIdFromObjC function, and js received response:", responseData)
       })

WKWebView交互

WKWebView作為UIWebView的替代品,采用自身的機制進行與JS的交互。
注意:WKWebView不支持JavaScriptCore框架的方式進行JS的交互。

JS調用OC方法(JS給OC傳數據)

交互的方法名定義好,OC,JS都需要進行注冊。

OC端注冊方法(jsCallNative)

    WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc] init];
    config.userContentController = [[WKUserContentController alloc] init];
    [config.userContentController addScriptMessageHandler:self name:@"jsCallNative"];
    WKWebView *webView = [[WKWebView alloc]initWithFrame:CGRectMake(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT - 14) configuration:config];

JS端注冊方法(jsCallNative)

function payClick() {
        window.webkit.messageHandlers.jsCallNative.postMessage({order_no:'201511120981234',channel:'wx',amount:1,subject:'粉色外套'});
    }

OC端接收數據

WKWebView的協議WKScriptMessageHandler里面的方法,JS向OC傳任何數據,OC都可以通過該方法接收,通過JS傳過來的數據,我們可以調用相應OC的方法,從而達到JS調用OC方法的目的。

/**
 * JS給原生傳數據:JS調用原生的方法
 */
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message{
    
    NSDictionary *dict = message.body;
    NSString* action = [dict valueForKey:@"action"];
    NSLog(@"----WKWebView交互----%@",action);
    
    if ([action isEqualToString:@"action1"]) {
        [self action1];
    }

    if ([action isEqualToString:@"action2"]) {
        [self action2];
    }
}

OC調JS的方法(OC向JS傳數據)

OC端代碼

payResult 為規定好的方法名,JS端用這個方法名接收數據。

    // 將支付結果返回給js
    //這里的payResult()是JS的語言
    NSString *jsStr = [NSString stringWithFormat:@"payResult('%@')",@"支付成功"];
    [self.webView evaluateJavaScript:jsStr completionHandler:^(id _Nullable result, NSError * _Nullable error) {
        NSLog(@"%@----%@",result, error);
    }];

注意:當JS調用OC的時候也可以用這個方法給JS進行回傳數據,和上面的WKScriptMessageHandler代理中的方法相結合,就是實現了OC和JS的數據交流

JS端代碼

function payResult(str) {
                asyncAlert(str);
                document.getElementById("returnValue").value = str;
            }

總結

UIWebView與JS交互:

  • 需要支持iOS6系統及其以下的系統:
    集成WebViewJavascriptBridge三方框架就好,方便更加靈活的交互。
  • 只需要支持iOS7及其以上的系統:
    采用系統自帶的JavaScriptCore框架就能滿足日常的交互功能了。

WKWebView與JS交互:

根據WKWebView所提供的API代理方法就可以實現靈活的交互了。

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

推薦閱讀更多精彩內容