WKWebView 与js的交互

自从iOS8之后苹果开始建议替换掉笨重的UIWebView,WKWebView 自从iOS9.0以后,更趋于完善。本文主要介绍WKwebView和js的使用WebViewJavascriptBridge的交互方法

OC 端代码

WebViewJavascriptBridge 早在很久以前就对WK进行了支持,不过本文使用的WK是6.0.2的版本,
pod 'WebViewJavascriptBridge', '~> 6.0.2'
  1. 关于WK的初始化 需要注意的是 在初始化的过程中不必在设置navigationDelegate的代理的。
    WKWebViewConfiguration * config = [[WKWebViewConfiguration alloc] init];
//     设置进程池 用来配置同一个进程池webView的共享数据 如cookies 用户凭证等
    WKProcessPool * pool = [[WKProcessPool alloc] init];
    config.processPool = pool;
    // 进程偏好设置
    WKPreferences * prefer = [[WKPreferences alloc] init];
    prefer.javaScriptEnabled = YES;
//     是否可以不进过用户交互由javascript自动打开窗口
    prefer.javaScriptCanOpenWindowsAutomatically = YES;
    config.preferences = prefer;
    //设置内容交互控制器 用于处理JavaScript与native交互
    WKUserContentController * userController = [[WKUserContentController alloc]init];
    //设置处理代理并且注册要被js调用的方法名称
    [userController addScriptMessageHandler:self name:@"name"];  
    //js注入,注入一个测试方法。
    NSString *javaScriptSource = @"function userFunc(){window.webkit.messageHandlers.name.postMessage( {\"name\":\"HS\"})}";
    WKUserScript *userScript = [[WKUserScript alloc] initWithSource:javaScriptSource injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:YES];
    // forMainFrameOnly:NO(全局窗口),yes(只限主窗口)
    [userController addUserScript:userScript];
    config.userContentController = userController;
    self.webView = [[WKWebView alloc] initWithFrame:(CGRect){0 ,0, [UIScreen mainScreen].bounds.size.width ,[UIScreen mainScreen].bounds.size.height } configuration:config];
    [self.view addSubview:self.webView];

    self.webView.UIDelegate = self;
    /**
     *   这个地方如果使用啦jsbrage 就不再需设置navigationDelegate 啦
     */
//    self.webView.navigationDelegate = self;
  1. 初始化bridge 方法
    _bridge = [WKWebViewJavascriptBridge bridgeForWebView:_webView];
// 因为这个地方设置了代理方法  所以切记在WK初始化的时候不要在设置代理了
    [_bridge setWebViewDelegate:self];
  1. 注册js调用oc的方法
-(void)registerNativeFunction
{
      [self shareInit];
}
-(void)shareInit{
    //     handler 中的block 参数是js要调用的oc的实现    handlerName 是oc的实现别名
    [_bridge registerHandler:@"shareInit" handler:^(id data, WVJBResponseCallback responseCallback) {
        // block 中是 oc中的实现方法
//  data 是 js传递给oc的数据
        NSLog(@"webView  使用 wkbridge 交互完成之后 返回的数据 data =%@",data);
        NSString * string = @"WHAHAH";
// responseCallback 是 方法执行完毕之后返回给js的
        responseCallback(string);
    }];
}
  1. oc 调用js的方法
//     oc 调用js的方法
//     不需要参数 不需要回掉方法
//    [_webViewBridge callHandler:@"testJSFunction"];
    // oc 传递给js有参数   不需要回掉方法
//    [_webViewBridge callHandler:@"testJSFunction" data:@"miaowu"];
//     oc 传递给js有参数  而且有回掉方法
    [_webViewBridge callHandler:@"testJSFunction" data:@"一个字符串" responseCallback:^(id responseData) {
        NSLog(@"调用完JS后的回调:%@",responseData);
    }];

JS 端代码

js 在调用oc的方法之前需要首先添加一个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)
            }
// js 调用了一次方法注入js代码 (如果不调用一次会导致oc和js互相调用不会响应) setupWebViewJavascriptBridge(function(bridge) {
                 bridge.registerHandler('testJSFunction', function(data, responseCallback) {
                    alert('JS方法被调用:'+data);
                    responseCallback('js执行过了');
                 })
            })
      
  1. JS 调用oc方法
   function jsCallOCMeathed(){
                WebViewJavascriptBridge.callHandler('shareViewShow',{'foor':'miaowu'},function(response){
                    alert(response);
                })
            }
  1. oc 调用js方法
     setupWebViewJavascriptBridge(function(bridge) {
// 如果有其他的oc调用js的 方法 依照下面的格式进行添加就可以啦
                 bridge.registerHandler('testJSFunction', function(data, responseCallback) {
                    alert('JS方法被调用:'+data);
                    responseCallback('js执行过了');
                 })
            })
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 之前用UIWebView的时候,非常的爽,用JSContext这个类就可以做想做的事,现在要改成WKWebView...
    AgoniNemo阅读 11,080评论 30 12
  • 用普通的UIWebView和JS交互 大概核心代码是这样的 代理方法 想调用的OC方法 然后JS代码 接下来是WK...
    波妞和酱豆子阅读 3,088评论 2 4
  • 事情的起因还是因为项目需求驱动。折腾了两天,由于之前没有UIWebView与JS交互的经历,并且觉得这次在功能上有...
    CocoLeo阅读 723评论 0 3
  • 远0阅读 343评论 0 1
  • 念头,即大脑里出现的想法,情绪,记忆等等以文字,画面表现的瞬间,往往转瞬即逝,但总会被我们知道。 一个普通人每天约...
    2a3705116a40阅读 353评论 0 0