App与Js交互(一)iOS

目录

示例代码

Demo: https://github.com/gwpp/jsinterface

前言

不论是在创业团队中快速试错,还是在成熟团队中快速迭代复杂需求,还或者是其他原因,WebView在APP中的大量使用已经成为了一个明显的趋势,这也应该算是大前端融合的一个表象吧。笔者在工作中也遇到过很多App&Js交互的问题,粗浅的研究了一下,这里也分享给大家,如果有错误的地方还请下方留言指出,共同进步。

iOS系统中的交互

众所周知,iOS有UIWebViewWKWebView两个组件可以用来渲染嵌入页面。前者使用甚广,出生的也早,后者是iOS8推出的,优化了加载速度和内存,安全性上也有所提升。具体的两者比较百度、简书上都很多,这里不做赘述。

方案一,拦截跳转

  • WebView:UIWebView
  • 原生调用JS:
    UIWebView直接调用Js方法,示例代码如下:
    [self.webView stringByEvaluatingJavaScriptFromString:@"showResponse('123')"];
    
  • JS调用原生:
    拦截跳转是我们最常见的一种方式,也是最简单,最容易理解的一种。我们可以在UIWebView的代理方法中拦截每一个请求,如果是特殊的链接就可以做一些事情,比如跳转、执行某些方法等。示例如下:
    // JS端
    window.location = 'app://login?account=13011112222&password=123456';
    
    // iOS端
    - (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType {  
      NSString *scheme = request.URL.scheme;
      NSString *host = request.URL.host;
      
      // 一般用作交互的链接都会有一个固定的协议头,这里我们一“app”作为协议头为了,实际项目中可以修改
      if ([scheme isEqualToString:@"app"]) { // scheme为“app”说明是做交互的链接
          if ([host isEqualToString:@"login"]) { // host为“login”对应的就是登录操作
              NSDictionary *paramsDict = [request.URL getURLParams];
              NSString *account = paramsDict[@"account"];
              NSString *password = paramsDict[@"password"];
              NSString *msg = [NSString stringWithFormat:@"执行登录操作,账号为:%@,密码为:%@", account, password];
              UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"原生弹窗" message:msg delegate:nil cancelButtonTitle:@"好" otherButtonTitles:nil, nil];
              [alert show];
          }
      return YES;
    }
    

方案二,拦截跳转

  • WebView:WKWebView

  • 原生调用JS:
    WKWebView直接调用Js方法,示例代码如下:

    [self.webView evaluateJavaScript:@"showResponse('点击了原生的按钮22222222222')" completionHandler:^(id _Nullable response, NSError * _Nullable error) {
          if (error) {
              NSLog(@"%@", error);
          } else {
              NSLog(@"%@", response);
          }
     }];
    

    它相对于UIWebView而言最大的优点就是支持callback,不想UIWebView那样只能一去不复返。

  • JS调用原生:
    类似UIWebView,在WK中我们同样可以拦截跳转,原理相同,代码不同。示例如下:

    // JS端
    window.location = 'app://login?account=13011112222&password=123456';
    
    // iOS端
    - (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {
      NSURLRequest *request = navigationAction.request;
      NSString *scheme = request.URL.scheme;
      NSString *host = request.URL.host;
      
      // 一般用作交互的链接都会有一个固定的协议头,这里我们一“app”作为协议头为了,实际项目中可以修改
      if ([scheme isEqualToString:@"app"]) { // scheme为“app”说明是做交互的链接
          if ([host isEqualToString:@"login"]) { // host为“login”对应的就是登录操作
              NSDictionary *paramsDict = [request.URL getURLParams];
              NSString *account = paramsDict[@"account"];
              NSString *password = paramsDict[@"password"];
              NSString *msg = [NSString stringWithFormat:@"执行登录操作,账号为:%@,密码为:%@", account, password];
              UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"原生弹窗" message:msg delegate:nil cancelButtonTitle:@"好" otherButtonTitles:nil, nil];
              [alert show];
          }
          
          // ... 这里可以继续加 else if
          
          decisionHandler(WKNavigationActionPolicyCancel);
          return;
      }
      decisionHandler(WKNavigationActionPolicyAllow);
    }
    

阶段小结

前两种方法到此就介绍完了,很简单,但是在项目大了之后拦截跳转的代理方法中会有非常多的判断。冗余、可维护性差,硬编码重。所以我们会有下面的其他方法。

方案三,JSContext

JSContext即JavaScriptContext,这个东西在UIWebView中可以拿到,但是在WKWebView中却是取不到了,所以只能用在UIWebView中。除此以外Android里也有类似的一个东西,所以使用JSContext就有了在JS端多平台统一的可能,这里不多说,在《App与Js交互(三)》中会有详细说明。
JSContext的原理就是iOS暴露出去一个遵守<JSExport>协议的对象给JS,JS可以直接调用该对象的public方法。

  • WebView:UIWebView
  • 原生调用JS:
    // 有两种方式。jsContext 是一个【JSContext *】变量,需要在【webViewDidFinishLoad: 】方法中每次赋值
    // 方式1:
    [self.jsContext evaluateScript:@"showResponse('点击了按钮1111111111111111')"];
    
    // 方式2:
    JSValue *value = self.jsContext[@"showResponse"];
    [value callWithArguments:@[@"点击了按钮222222222"]];
    
  • JS调用原生:
    // JS端,app是iOS中注册的一个对象
    app.login("13011112222", "123456");
    
    // iOS端
    // 每次嵌入页面加载完毕都要给jsContext赋值,否则在js端调用可能会失效。
    - (void)webViewDidFinishLoad:(UIWebView *)webView {
      self.jsContext = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];
      self.jsContext.exceptionHandler = ^(JSContext *context, JSValue *exception) {
          context.exception = exception;
          NSLog(@"异常信息:%@", exception);
      };
      // app是随便取的名字,可以改,改了之后JS要同步修改。如果Android端使用@JavaScriptInterface的形式,那么还要保证Android、iOS两端同步,建议都用app
      self.jsContext[@"app"] = [[JSContextModel alloc] init];
    }
    
    // JSContextModel,
    @protocol JsContextExport<JSExport>
    /**
     * 登出方法,js调用的方法名也是logout
     */
    - (void)logout;
    /**
     * 登录方法,JSExportAs的作用就是给OC方法导出一个js方法名,例如下面的方法js调用就是 login("your account", "your password")。在多参数的方法声明时必须使用这种方式
     */
    JSExportAs(login, - (void)loginWithAccount:(NSString *)account   password:(NSString *)password);
    /**
     * 获取登录信息
     * @return 当前登录用户的身份信息。JSContext方式调用OC时,方法的返回值只能是NSString、NSArray、NSDictionary、NSNumber、BooL,其他类型不能解析
     */
    - (NSDictionary *)getLoginUser;
    @end
    
    @interface JSContextModel : NSObject<JsContextExport>
    @end
    

方案四,WebKit

window.webkit.messagehandlers.<name>.postMessage是apple推荐使用的WKWebView的JS交互方式,使用起来比较简单,不支持callback回调。

  • WebView:WKWebView
  • 原生调用JS:
    参考【方案二】的原生调用JS
  • JS调用原生:
    // js
    window.webkit.messageHandlers.login.postMessage({
      'account': '13000000000',
      'password': '123456'
    });
    
    // iOS - 初始化WKWebView时设置 configuration
    self.webView = [[WKWebView alloc] initWithFrame:CGRectZero configuration:[[WKWebViewConfiguration alloc] init]];
    WKUserContentController *confVc = self.webView.configuration.userContentController;
      [confVc addScriptMessageHandler:self name:@"login"];
      
    // iOS - 在ScriptMessageHandler 的代理方法中处理
    - (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message {
      if ([message.name isEqualToString:@"login"]) {
          if (![message.body isKindOfClass:[NSDictionary class]]) {
              return;
          }
          NSDictionary *data = message.body;
          NSString *account = data[@"account"];
          NSString *password = data[@"password"];
          
          NSString *msg = [NSString stringWithFormat:@"执行登录操作,账号为:%@,密码为:%@", account, password];
          UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"原生弹窗" message:msg delegate:nil cancelButtonTitle:@"好" otherButtonTitles:nil, nil];
          [alert show];
          return;
      }
    }
    

方案五,JsBridge

  • WebView:UIWebView、WKWebView同时支持,且方法名完全没有差异,但是特别要注意的一点就是:iOS原生是不支持这种方式的,我们需要依赖于一个三方库 —— WebViewJavascriptBridge,这是一个很有名的库,具体有多牛逼这里也不做过多需求,百度一下你就知道。
  • 初始化代码:
    // JS初始化代码
    /**
     * 初始化jsbridge
     * @param readyCallback 初始化完成后的回调
     */
     function initJsBridge(readyCallback) {
         var u = navigator.userAgent;
         var isAndroid = u.indexOf('Android') > -1 || u.indexOf('Adr') > -1; //android终端
         var isiOS = !!u.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/); //ios终端
         // 注册jsbridge
         function connectWebViewJavascriptBridge(callback) {
             if (isAndroid) {
                 if (window.WebViewJavascriptBridge) {
                     callback(WebViewJavascriptBridge)
                 } else {
                     document.addEventListener(
                         'WebViewJavascriptBridgeReady'
                         , function () {
                             callback(WebViewJavascriptBridge)
                         },
                         false
                     );
                 }
                 return;
             }
             if (isiOS) {
                 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)
             }
         }
         // 调用注册方法
         connectWebViewJavascriptBridge(function (bridge) {
             if (isAndroid) {
                 bridge.init(function (message, responseCallback) {
                     console.log('JS got a message', message);
                     responseCallback(data);
                 });
             }
              
             // 只有在这里注册过的方法,在原声代码里才能用callHandler的方式调用
             bridge.registerHandler('jsbridge_showMessage', function (data, responseCallback) {
                 showResponse(data);
             });
             bridge.registerHandler('jsbridge_getJsMessage', function (data, responseCallback) {
                  showResponse(data);
                  responseCallback('native 传过来的是:' + data);
              });
    
              readyCallback();
          });
      }
    
    // iOS初始化代码
     - (void)setupJsBridge {
      if (self.bridge) return;
      // self.webview既可以是UIWebView,又可以是WKWebView
      self.bridge = [WebViewJavascriptBridge bridgeForWebView:self.webView];
      
      [self.bridge registerHandler:@"getOS" handler:^(id data, WVJBResponseCallback responseCallback) {
          // 这里Response的回调可以传id类型数据,但是为了保持Android、iOS的统一,全部使用json字符串作为返回数据
          NSDictionary *response = @{@"error": @(0), @"message": @"", @"data": @{@"os": @"ios"}};
          responseCallback([response jsonString]);
      }];
      
      [self.bridge registerHandler:@"login" handler:^(id data, WVJBResponseCallback responseCallback) {
          if (data == nil || ![data isKindOfClass:[NSDictionary class]]) {
              NSDictionary *response = @{@"error": @(-1), @"message": @"调用参数有误"};
              responseCallback([response jsonString]);
              return;
          }
          
          NSString *account = data[@"account"];
          NSString *passwd = data[@"password"];
          NSDictionary *response = @{@"error": @(0), @"message": @"登录成功", @"data" : [NSString stringWithFormat:@"执行登录操作,账号为:%@、密码为:@%@", account, passwd]};
          responseCallback([response jsonString]);
      }];
    }
    
  • 原生调用JS
    [self.bridge callHandler:@"jsbridge_getJsMessage" data:@"点击了原生的按钮222222222" responseCallback:^(id responseData) {
      UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"显示jsbridge返回值" message:responseData delegate:nil cancelButtonTitle:@"好" otherButtonTitles:nil, nil];
       [alert show];
    }];
    
  • JS调用原生
    // 首先调用JSBridge初始化代码,完成后再设置其他
    initJsBridge(function () {
      $("#getOS").click(function () {
            // 通过JsBridge调用原生方法,写法固定,第一个参数时方法名,第二个参数时传入参数,第三个参数时响应回调
            window.WebViewJavascriptBridge.callHandler('getOS', null, function (response) {
            showResponse(response);
          });
      });
     });
    
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352